Skip to main content

Coal Mine - Periodic task execution monitor

Project description

(Releases are available in PyPI.)

Track tasks that are supposed to execute periodically using “canaries” that the tasks trigger when they execute. Alert by email when a canary is late. Alert again when a late canary resumes. Keep a partial history of canary trigger times.

Uses MongoDB for storage. Pull requests to add additional storage engines are welcome.

The server notifies immediately when the deadline for an unpaused canary passes. Similarly, the server notifies immediately when a previously late canary is triggered.

All timestamps are stored and displayed in UTC.

Prerequisites

Python 3.

Requirements listed in requirements.txt.

For development, requirements listed in requirements_dev.txt.

API Usage examples

Example commands

$ coal-mine &
[1] 7564
$ curl 'http://coal-mine-server/coal-mine/v1/canary/create?name=My+First+Canary&periodicity=3600'
{
    "status": "ok",
    "canary": {
        "deadline": "2015-03-19T02:08:44.885182",
        "id": "fbkvlsby",
        "paused": false,
        "description": "",
        "periodicity": 3600,
        "name": "My First Canary",
        "slug": "my-first-canary",
        "emails": [],
        "history": [
            [
                "2015-03-19T01:08:44.885182",
                "Snitch created"
            ]
        ],
        "late": false
    }
}
$ curl 'http://coal-mine-server/fbkvlsby?comment=short+form+trigger+url'
{
    "recovered": false,
    "unpaused": false,
    "status": "ok"
}
$ curl 'http://coal-mine-server/coal-mine/v1/canary/trigger?slug=my-first-canary&comment=long+form+trigger+url'
{
    "recovered": false,
    "unpaused": false,
    "status": "ok"
}
$ curl 'http://coal-mine-server/coal-mine/v1/canary/get?name=My+First+Canary'
{
    "canary": {
        "paused": false,
        "name": "My First Canary",
        "history": [
            [
                "2015-03-19T01:11:56.408000",
                "Triggered (long form trigger url)"
            ],
            [
                "2015-03-19T01:10:42.608000",
                "Triggered (short form trigger url)"
            ],
            [
                "2015-03-19T01:08:44.885000",
                "Snitch created"
            ]
        ],
        "emails": [],
        "id": "fbkvlsby",
        "late": false,
        "slug": "my-first-canary",
        "deadline": "2015-03-19T02:11:56.408000",
        "periodicity": 3600,
        "description": ""
    },
    "status": "ok"
}

All API endpoints are fully documented below.

Watching a cron job

0 0 * * * my-backup-script.sh && (curl http://coal-mine-server/fbkvlsby &>/dev/null)

Or use use the CLI!

Use cmcli to send commands to the server from the same host or from any other host where Coal Mine is installed. You can either configure it:

cmcli configure --host coal-mine-server --port 8080 --auth_key [auth_key in coal-mine.ini]

Or specify --host, --port, and/or --auth-key on the command line of each invocation.

Some example commands:

cmcli create --help
# $((60*60*25)) is 25 hours
cmcli create --name 'My Second Canary' --periodicity $((60*60*25))
cmcli delete --slug 'my-second-canary'

Run cmcli --help for more information.

Server configuration

Create coal-mine.ini in the current directory, or /etc, or /usr/local/etc, or modify the list of directories hear the top of main() in server.py if you want to put it somewhere else.

Here’s what can go in the config file:

  • [logging] - optional

  • file - specify the log file; otherwise logging goes to stderr

  • rotate - if true, then rotate the log file when it gets too large

  • max_size - max log file size before rotating (default: 1048576)

  • backup_count - number of rotated log files to keep (default: 5)

  • [mongodb] - required

  • hosts - the first argument to pymongo’s MongoClient or MongoReplicaSetClient

  • database - database name. Coal Mine will create only one collection in the database, called “canaries”.

  • username - must be specified, but can be blank if no authentication is required

  • password - must be specified, but can be blank if no authentication is required

  • replicaSet - must be specified if using a replicaset

  • other arguments will be passed through to MongoClient or MongoReplicaSetClient

  • [email] - required

  • sender - email address to put in the From line of notification emails

  • [wsgi] - optional

  • port - port number the server should listen on (default: 80)

  • auth_key - if non-empty, then the specified key must be specified as a parameter of the same name with all API requests except “trigger”.

API

All API endpoints are submitted as http(s) GET requests. Results are returned in JSON.

All results have a “status” field which is “ok” on success or “error” on failure. Failures also return a reasonable HTTP error status code.

Boolean fields in API should be specified as “true”, “yes”, or “1” for true, or “false”, “no”, “0”, or empty string for false. Boolean fields in responses are standard JSON, i.e., “true” or “false”.

Timestamps returned by API are always UTC.

Create canary

Endpoint: /coal-mine/v1/canary/create

Side effects:

Adds canary to database. Creates history record at current time with “Canary created” as its comment. Sets deadline to current time plus periodicity, unless “paused” was specified.

Required parameters:

  • name

  • periodicity

  • auth_key (if authentication is enabled in the server)

Optional parameters:

  • description - empty if unspecified

  • email - specify multiple times for multiple addresses; no notifications if unspecified

  • paused - allows canary to be created already in paused state

Response is the same as shown for get().

Delete canary

Endpoint: /coal-mine/v1/canary/delete

Required parameters:

  • name, id, or slug

  • auth_key

Response:

{'status': 'ok'}

Update canary

Endpoint: /coal-mine/v1/canary/update

Side effects:

Updates the specified canary attributes. Updates deadline to latest history timestamp plus periodicity if periodicity is updated and canary is unpaused, and sets late state if new deadline is before now. Sends notification if canary goes from not late to late or vice versa.

Required parameters:

  • id or slug (not name, which should only be specified to update the name and slug)

  • auth_key

Optional parameters:

  • name

  • periodicity

  • description

  • email

Response is the same as shown for get().

Get canary

Endpoint: /coal-mine/v1/canary/get

Required parameters:

  • name, id, or slug

  • auth_key

Response:

{'status': 'ok',
 'canary': {'name': name,
           'description': description,
           'id': identifier,
           'slug': slug,
           'periodicity': seconds,
           'emails': [address, ...],
           'late': boolean,
           'paused': boolean,
           'deadline': 'YYYY-MM-DDTHH:MM:SSZ',
           'history': [['YYYY-MM-DDTHH:MM:SSZ', comment], ...]}}

List canaries

Endpoint: /coal-mine/v1/canary/list

Required parameters:

  • auth_key

Optional parameters:

  • verbose - include all query output for each canary

  • paused - boolean, whether to list paused / unpaused canaries only

  • late - boolean, whether to list late / timely canaries only

  • search - string, regular expression to match against name, identifier, and slug

Response:

{'status': 'ok',
 'canaries': [{'name': name,
             'id': identifier},
            ...]}

If “verbose” is true, then the JSON for each canary includes all the fields shown above, not just the name and identifier.

Trigger canary

Endpoint: /coal-mine/v1/canary/trigger

Also: /identifier, in which case the “id” parameter is implied

Side effects:

Sets late state to false. Sets deadline to now plus periodicity. Adds history record. Prunes history records. Unpauses canary. Generates notification email if canary was previously late.

Required parameters:

  • name, id, or slug

  • auth_key

Optional parameters:

  • comment - stored in history with trigger record

Response:

{'status': 'ok', 'recovered': boolean, 'unpaused': boolean}
  • recovered - indicates whether the canary was previously late before this trigger

  • unpaused - indicates whether the canary was previously paused before this trigger

Pause canary

Endpoint: /coal-mine/v1/canary/pause

Side effects:

Clears deadline. Sets late state to false if necessary. Pauses canary. Adds history record about pause. Prunes history records.

Required parameters:

  • name, id, or slug

  • auth_key

Optional parameters:

  • comment

Response is the same as shown for get().

Unpause canary

Endpoint /coal-mine/v1/canary/unpause

Side effects:

Sets deadline to now plus periodicity. Unpauses canary. Adds history record about unpause. Prunes history records.

Required parameters:

  • name, id, or slug

  • auth_key

Optional parameters:

  • comment

Response is the same as shown for get().

Quis custodiet ipsos custodes?

Obviously, if you’re relying on Coal Mine to let you know when something is wrong, you need to make sure that Coal Mine itself stays running. One way to do that is to have a cron job which periodically triggers a canary and generates output (which crond will email to you) if the trigger fails. Something like:

0 * * * * (curl http://coal-mine-server/atvywzoa | grep -q -s '"status": "ok"') || echo "Failed to trigger canary."

I also recommend using a log-monitoring service such as Papertrail to monitor and alert about errors in the Coal Mine log.

Alternatives

Alternatives to Coal Mine include:

We chose to write something new, rather than using what’s already out there, for several reasons:

  • We wanted more control over the stability and reliability of our watch service than the commercial alternatives provide.

  • We wanted fine-grained control over the periodicity of our watches, as well as assurance that we would be notified immediately when a watch is late, something that not all of the alternatives guarantee.

  • We like Python.

  • We like OSS.

Contributors

Coal Mine was created by Jonathan Kamens, with design help from the awesome folks at Quantopian. Thanks, also, to Quantopian for supporting the development and open-sourcing of this project.

Contacts

Github

Email

PyPI

Developer notes

Development philosophy

Use Python.

Do one, simple thing well. There are several similar projects out there that do more than this project attempts to do.

Make the implementation as simple and straightforward as possible. The code should be small. What everything does should be obvious from reading it.

Minimize external dependencies. If something is simple and straightforward to do ourselves, don’t use a third-party package just for the sake of using a third-party package.

Data model

For each canary, we store:

  • name

  • description

  • slug - the name, lower-cased, with spaces and underscores converted to hyphens and other non-alphanumeric characters removed

  • random identifier, guaranteed unique

  • periodicity - maximum number of seconds that can elapse before a canary is late.

  • notification email address(es)

  • late state (boolean)

  • paused state (boolean)

  • deadline for next update

  • history of triggers, pruned when (>1000 or (>100 and older than one week)

Timestamps in database are UTC.

To Do

(Pull requests welcome!)

Other storage engines.

Other notification mechanisms.

More smtplib configuration options in INI file.

Unit tests would be nice.

Web UI.

Links to Web UI in email notifications.

Repeat notifications if a canary remains late for an extended period of time? Not even sure I want this.

Better authentication?

Support time-zone localization of displayed timestamps.

SSL support in server

Project details


Download files

Download the file for your platform. If you're not sure which to choose, learn more about installing packages.

Source Distribution

coal_mine-0.2.tar.gz (15.2 kB view details)

Uploaded Source

Built Distributions

coal_mine-0.2.linux-x86_64.tar.gz (30.0 kB view details)

Uploaded Source

coal_mine-0.2-py3.4.egg (35.8 kB view details)

Uploaded Source

File details

Details for the file coal_mine-0.2.tar.gz.

File metadata

  • Download URL: coal_mine-0.2.tar.gz
  • Upload date:
  • Size: 15.2 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No

File hashes

Hashes for coal_mine-0.2.tar.gz
Algorithm Hash digest
SHA256 7bfecc7c10c9ef6d3531c1fe951bda143c6619ac1b211430d2bb59f0f66cc805
MD5 4472281d2149b254d6ed6acc1a2f9f7a
BLAKE2b-256 bfa69c89e23a93ac8ea2b0130137f33658043546eb4227be16c11c63d2c9c938

See more details on using hashes here.

File details

Details for the file coal_mine-0.2.linux-x86_64.tar.gz.

File metadata

File hashes

Hashes for coal_mine-0.2.linux-x86_64.tar.gz
Algorithm Hash digest
SHA256 9bd14d04c61025b522c6f3ee8ceda2a6504a972b997f35c6cd8d9bd6deb95159
MD5 165f515d620e43f3241c887fe113c24d
BLAKE2b-256 d8009120d337ab35a8d356944b0024504ae21ba751cb834a4fd51452e0dfb481

See more details on using hashes here.

File details

Details for the file coal_mine-0.2-py3.4.egg.

File metadata

  • Download URL: coal_mine-0.2-py3.4.egg
  • Upload date:
  • Size: 35.8 kB
  • Tags: Source
  • Uploaded using Trusted Publishing? No

File hashes

Hashes for coal_mine-0.2-py3.4.egg
Algorithm Hash digest
SHA256 f725baa8f9923607f8c07316c9fea1956aa0ded0724efbf0dddf7b8c6804c3ad
MD5 2ba75c7009d7bbb2a74ffa0d77087f4e
BLAKE2b-256 37921c5bfe513bebf860a89f29a78ffd9faf73e6cc2090d574f468bcfab0c792

See more details on using hashes here.

Supported by

AWS AWS Cloud computing and Security Sponsor Datadog Datadog Monitoring Fastly Fastly CDN Google Google Download Analytics Microsoft Microsoft PSF Sponsor Pingdom Pingdom Monitoring Sentry Sentry Error logging StatusPage StatusPage Status page