Skip to content

A flexible worker pool that aims to fit all possible use-cases.

License

Notifications You must be signed in to change notification settings

cookiemonsterproject/cookie-monster

Repository files navigation

CookieMonster

Build Status GolangCI codecov Go Report Card

Purpose

CookieMonster is a flexible worker pool that can be adapted to fit (almost) any usecase.

This is accomplished with the use of Jars and Cookies. These are simply interfaces that represent a work provider and the information needed to process that work, respectively.

Usage

  1. Add CookieMonster to your project:

    go get -u github.com/cookiemonsterproject/cookie-monster

  2. Setup your Jar, by either:

    a) Checking if there's already an implementation that fits your usecase.

    b) Implementing your own Jar. For this you'll need to create implementations that fit the Jar and Cookie interfaces.

    // Represents a work provider
    type Jar interface {
        Retrieve() ([]Cookie, error) // generate units of work to distribute amongst the various workers
        Retire(Cookie) error         // mark the work as done (e.g., delete a message from a queue after it's been processed)
    }
    
    // Represents a unit of work
    type Cookie interface {
        ID() string                  // work identifier
        Content() interface{}        // data needed to process the work
        Metadata() map[string]string // optional map of metadata related to the work
    }

    Please consider sharing your implementation as an open-source project and help this baby grow (check Contributing).

  3. Configure the pool:

    digester := cookiemonster.NewDigester(jar)

    Optionally, you can also pass a list of options to tweak how the digester works internally.

    List of options:

    Function Description Default behaviour (if not called)
    SetWorkers Define the number of goroutines processing work concurrently. Sets to runtime.NumCPU().
    SetBackoff Define how the digester behaves in the interval of processing work. You can use the ConstantBackoff or ExponentialBackoff implementations provided or use your own implementation of the Backoff interface. Sets to ConstantBackoff of 10 seconds.
    SetInfoLog Define info logger where internal information is written to. No info logs.
    SetErrorLog Define error logger where errors are written to. No error logs.
    SetStopSignals Define a list of os.Signals for the digester to wait for to call Stop() automatically, once you call Start(). Start() will exit immediately, leaving the pool working in the background. It will be up to you to define how to wait for the work to be complete and call digester.Stop() to stop the pool.
  4. Start the pool:

    digester.Start(digestFn)

    Here you have to pass the function used to process the work, in the form of: func(cookie cookiemonster.Cookie) error.

  5. Stop the pool (optional):

    digester.Stop()

    If you didn't pass any signals, you'll need to explicitly call the Stop function for the pool to gracefully shutdown.

    On the other hand, if you did pass them, this will be automatically called once any of them is triggered.

Examples

Check the examples folder.

Contributing

This project aims to be generic and fit as much cases as possible. This will only be possible if you share your specific usecase to help identify where the project is still lacking.

To ease discovery, I'd love to have github.com/cookiemonsterproject as the main place to go to find the existent implementations, so if you wish to contribute feel free to open an issue or DM me on the Gophers' Slack.