/worker-tools

WorkerTools is a collection of modules meant to speed up how we write background tasks following a few basic patterns.

Primary LanguageRubyMIT LicenseMIT

WorkerTools

Build Status MIT License Release Maintenance


Table of Contents
  1. About The Project
  2. Installation
  3. Conventions
  4. Module 'Basics'
  5. Module 'Recorder'
  6. Module 'SlackErrorNotifier'
  7. Wrappers
  8. Module 'Notes'
  9. Attachments
  10. Complete Examples
  11. Changelog
  12. Requirements
  13. Contributing
  14. License
  15. Acknowledgement

About The Project

WorkerTools is a collection of modules meant to speed up how we write background tasks following a few basic patterns. The structure of plain independent modules with limited abstraction allows to define and override a few methods according to your needs without requiring a deep investment in the library.

These modules provide some features and conventions to address the following points with little configuration on your part:

  • How to save the state the task.
  • How to save notes relevant to the admins / customers.
  • How to log the details
  • How to handle exceptions and send notifications
  • How to process CSV files (as input and output)
  • How to process XLXS files (as input, output coming next)
  • How to set options

Installation

Add this line to your application's Gemfile:

gem 'worker_tools'

And then execute:

$ bundle

Or install it yourself as:

$ gem install worker_tools

Conventions

Most of the modules require an ActiveRecord model to keep track of the state, notes, and files related to the job. The class of this model is typically an Import, Export, Report.. or something more generic like a JobEntry.

An example of this model for an Import using Paperclip would be something like this:

class Import < ApplicationRecord
  enum state: %w[
    waiting
    complete
    complete_with_warnings
    failed
    running
  ].map { |e| [e, e] }.to_h

  enum kind: { foo: 0, bar: 1 }

  has_attached_file :attachment

  validates :kind, presence: true
  validates :state, presence: true
  end

The state complete and failed are used by the modules. Both state and kind could be an enum or just a string field. Whether you have one, none or many attachments, and which library you use to handle it's up to you.

The state complete_with_warnings indicates that the model contains notes that did not lead to a failure but should get some attention. By default those levels are warning and errors and can be customized.

In this case the migration would be something like this:

  def change
    create_table :imports do |t|
      t.integer :kind, null: false
      t.string :state, default: 'waiting', null: false
      t.json :notes, default: []
      t.json  :options, default: {}
      t.json :meta, default: {}

      t.string :attachment_file_name
      t.integer :attachment_file_size
      t.string :attachment_content_type

      t.timestamps
    end

Module 'Basics'

the basics module takes care of finding or creating the model, marking it as completed or failed, and calling any flow control wrappers around run that had been specified. (See wrappers)

A simple example would be as follows:

class MyImporter
  include WorkerTools::Basic
  wrappers :basics

  def model_class
    Import
  end

  def model_kind
    'foo'
  end

  def run
    # do stuff
  end
end

The basics module contains a perform method, which is the usual entry point for ApplicationJob and Sidekiq. It can receive the id of the model, the model instance, or nothing, in which case it will attempt to create this model on its own.

By default errors subclassed from WorkerTools::Errors::Silent (such as those related to wrong headers in the input modules) will mark the model as failed but not raise. The method silent_error? lets you modifiy this behaviour.

Module 'Recorder'

Provides some methods to manage a log and the notes field of the model. The main methods are add_info, add_log, and record (which both logs and appends the message to the notes field). See all methods in recorder

This module has a recoder wrapper that will register the exception details into the log and notes field in case of error:

class MyImporter
  include WorkerTools::Basic
  wrappers :basics, :recorder
  # ...
end

If you only want the logger functions, without worrying about persisting a model, you can use the logger wrapper and include the module as a stand alone component (without the basics module), like this:

  class StandAloneWithLogging
    include WorkerTools::Recorder

    def perform
      with_wrapper_logger do
        # do stuff
      end
    end
  end

Module SlackErrorNotifier

Provides a Slack error notifier wrapper. To do this, you need to define SLACK_NOTIFIER_WEBHOOK as well as SLACK_NOTIFIER_CHANNEL. Then you need to include the SlackErrorNotifier module in your class and append slack_error_notifier to your wrappers. Below you can see an example.

  class MyImporter
    include WorkerTools::SlackErrorNotifier

    wrappers :slack_error_notifier

    def perform
      with_wrapper_logger do
        # do stuff
      end
    end
  end

See all methods in slack_error_notifier

Module CSV Input

See all methods in csv_input

Module CSV Output

See all methods in csv_output

Module XLSX Input

See all methods in xlsx_input

Module XLSX Output

See all methods in xlsx_output

Wrappers

In the basics module, perform calls your custom method run to do the actual work of the task, and wraps it around any methods expecting a block that you might have had defined using wrappers. That gives us a systematic way to add logic depending on the output of run and any exceptions that might arise, such as logging the error and context, sending a chat notification, retrying under some circumstances, etc.

The following code

class MyImporter
  include WorkerTools::Basic
  wrappers :basics

  def run
    # do stuff
  end

  # ..
end

is internally handled as

def perform(model_id)
  # set model
  with_wrapper_basics do
    run
  end
end

where this wrapper method looks like

def with_wrapper_basics(&block)
  block.yield # calls run
  # marks the import as complete
  rescue Exception
  # marks the import as failed
  raise
end

if we also add a wrapper to send notifications, such as wrappers :basics, :rocketchat_error_notifier

the resulting nested calls would look like

def perform(model_id)
  # set model
  with_wrapper_basics do
    with_wrapper_rocketchat_error_notifier do
      run
    end
  end
end

Run Modes

It is possible to run the same task in different modes by providing the key run_mode inside options. For example, by setting it to :destroy, the method run_in_destroy_mode will get called instead of the usual run

    # options[:run_mode] = :destroy

    def run_in_destroy_mode
      # add_some note
      # delete plenty of stuff, take your time
    end

As a convention, use options[:run_mode_option] to provide more context:

    # options[:run_mode] = :destroy
    # options[:run_mode_option] = :all / :only_foos / :only_bars
    def run_in_destroy_mode
      case run_mode_option
      when :all then kaboom
      when :only_foos then delete_foos
      when :only_bars then delete_bars
      end
    end

If the corresponding run method is not available an exeception will be raised. A special case is the run_mode :repeat which will try to use the method :run_in_repeat_mode and fallback to run if not present.

Counter

There is a counter wrapper that you can use to add custom counters to the meta attribute. To do this, you need to complete the following tasks:

  • include WorkerTools::Counters to your class
  • add :counters to the wrappers method props
  • call counters method with your custom counters You can see an example below. After that, you can access your custom counters via the meta attribute.
class MyImporter
  include WorkerTools::Counters
  wrappers :counters
  counters :foo, :bar

  def run
    example_foo_counter_methods
  end

  def example_foo_counter_methods
    # you can use the increment helper
    10.times { increment_foo } # +1
    increment_foo(5) # +5

    # the counter works like a regular accessor, you can read it and modify it
    # directly
    self.bar = 100
    puts bar # => 100
  end

  # ..
end

Benchmark

There is a benchmark wrapper that you can use to record the benchmark. The only thing you need to do is to include the benchmark module and append the name to the wrapper array. Below you can see an example of the integration.

class MyImporter
  include WorkerTools::Benchmark
  wrappers :benchmark

  def run
    # do stuff
  end

  # ..
end

Module 'Notes'

If you use ActiveRecord you may need to modify the serializer as well as deserializer from the note attribute. After that you can easily serialize hashes and array of hashes with indifferent access. For that purpose the gem provides two utility methods. (HashWithIndifferentAccessType, SerializedArrayType). There is an example of how you can use it.

  class ServiceTask < ApplicationRecord

    attribute :notes, SerializedArrayType.new(type: HashWithIndifferentAccessType.new)
  end

See all methods in utils

Attachments

The modules that generate a file expect the model to provide an add_attachment method with following signature:

  def add_attachment(file, file_name: nil, content_type: nil)
    # your logic
  end

You can skip this convention by overwriting the module related method, for example after including CsvOutput

def csv_output_add_attachment
  # default implementation
  # model.add_attachment(csv_output_tmp_file, file_name: csv_output_file_name, content_type: 'text/csv')

  # your method
  ftp_upload(csv_output_tmp_file)
end

Complete Examples

XLSX Input Example

class XlsxInputExample
  include Sidekiq::Worker
  include WorkerTools::Basics
  include WorkerTools::Recorder
  include WorkerTools::XlsxInput

  wrappers %i[basics recorder]

  def model_class
    Import
  end

  def model_kind
    'xlsx_input_example'
  end

  def run
    xlsx_input_foreach.each { |row| SomeModel.create!(row) }
  end

  def xlsx_input_columns
    {
      foo: 'Your Foo',
      bar: 'Your Bar'
    }
  end
end

CSV Input Example

class CsvInputExample
  include Sidekiq::Worker
  include WorkerTools::Basics
  include WorkerTools::Recorder
  include WorkerTools::CsvInput

  wrappers %i[basics recorder]

  def model_class
    Import
  end

  def model_kind
    'csv_input_example'
  end

  def csv_input_columns
    {
      flavour: 'Flavour',
      number: 'Number'
    }
  end

  def run
    csv_input_foreach.map { |row| do_something row_to_attributes(row) }
  end

  def row_to_attributes(row)
    {
      flavour: row['flavour'].downcase,
      number: row['number'].to_i * 10
    }
  end
end

CSV Output Example

# More complex example with CsvOutput
class CsvOutputExample
  include Sidekiq::Worker
  include WorkerTools::Basics
  include WorkerTools::CsvOutput
  include WorkerTools::Recorder

  wrappers %i[basics recorder]

  def model_class
    Report
  end

  def model_kind
    'csv_out_example'
  end

  def model_file_name
    "#{model_kind}-#{Date.current}.csv"
  end

  def run
    csv_output_write_file
  end

  def csv_output_column_headers
    @csv_output_column_headers ||= {
      foo: 'Foo',
      bar: 'Bar'
    }
  end

  def csv_output_entries
    @csv_output_entries ||= User.includes(...).find_each do |user|
      {
        foo: user.foo,
        bar: user.bar
      }
    end
  end

end

XLSX Output Example

# ExampleXlsxOutput
class XlsxOutputExample
  include Sidekiq::Worker
  include WorkerTools::Basics
  include WorkerTools::Recorder
  include WorkerTools::XlsxOutput

  wrappers %i[basics recorder]

  def model_class
    Export
  end

  def model_kind
    'xlsx_output_example'
  end

  def run
    xlsx_output_write_file
  end

  def xlsx_output_column_headers
    @xlsx_output_column_headers ||= {
      foo: 'Foo',
      bar: 'Bar'
    }
  end

  def xlsx_output_entries
    @xlsx_output_entries ||= SomeArray.lazy.map do |entry|
      {
        foo: user.foo,
        bar: user.bar
      }
    end
  end
end

Changelog

See CHANGELOG

Requirements

  • ruby > 2.3.1

Contributing

Contributions are what make the open source community such an amazing place to be learn, inspire, and create. Any contributions you make are greatly appreciated.

  1. Fork the Project
  2. Create your Feature Branch (git checkout -b feature/new_feature)
  3. Commit your Changes (git commit -m 'feat: Add new feature')
  4. Push to the Branch (git push origin feature/new_feature)
  5. Open a Pull Request

License

The gem is available under the MIT License. See LICENSE for more information.

Acknowledgement