# pyramid_scheduler

The ``pyramid-scheduler`` package is a pyramid plugin that allows
asynchronous and deferred task scheduling and management. It uses
APScheduler_ for actual task management and Kombu_ for messaging
between processes.

**IMPORTANT**: the pyramid-scheduler is very new, but fully
functional. The following features are considered "beta", and
should probably not be used in production:

* Multiple queues. For now, just use a single queue (and thus you are
  limited to a single background process).

* Intuitive sequence of job events. Currently, the events that can be
  listened to (from pyramid_scheduler.api.Event, such as JOB_CREATED,
  JOB_EXECUTED, JOB_CANCELED, and JOB_REMOVED) do not reliably fire in
  an intuitive order. For example, you may get a JOB_REMOVED event
  before the JOB_EXECUTED event for a deferred job.


## TL;DR

Install:

```
$ pip install pyramid-scheduler
```

Use:

``` python
# ini file settings:
#   [app:main]
#   scheduler.combined   = true | false   ## should execution be in-process?
#   scheduler.queues     = jobs           ## space-separated list of queues
#   scheduler.broker.url = %(dburl)s      ## the URL used for kombu messaging
#   ## other optional settings:
#   ##   scheduler.housekeeping
#   ##   scheduler.jobstore.default.class
#   ##   scheduler.misfire_grace_time

# enabling the plugin adds a `scheduler` attribute to the registry
def main(global_config, **settings):
  # ... (the usual pyramid startup calls) ...
  config.include('pyramid-scheduler)

# create an asynchronous task
def slow_process(name, id):
  # ...a slow asynchronous job...
def handle_request_quickly(request):
  request.registry.scheduler.add_async_job(slow_process, args=('my-first-arg', 2))

# schedule a deferred task for one hour from now
def delayed_process():
  # ...something that should happen later...
def handle_request_now(request):
  import time
  request.registry.scheduler.add_date_job(delayed_process, time.time() + 3600)

# do something every 10 minutes
def interval_process(reason=None):
  # ...gets executed every 10 minutes with an optional reason...
def handle_request_often(request):
  request.registry.scheduler.add_date_job(interval_process, minutes=10)
```


## Concepts

The pyramid-scheduler package refers to asynchronous or deferred tasks
to be managed as "jobs" and these fall into the following categories:

* Asynchronous jobs:

  These jobs are executed immediately, but asynchronously.

* Deferred jobs, i.e. "date" jobs:

  Jobs that are scheduled to be performed at a specific time.

* Interval jobs:

  Similar to deferred jobs, but that are then re-executed on an
  interval.

* Cron jobs:

  Similar to interval jobs, but they use a scheduling definition
  that is similar to a unix "cron" definition.

Conceptually, there are two activities: the activity of creating the
jobs and the activity of executing the jobs. These can be performed by
the same process (with "combined" mode enabled), or they can be
performed by different processes (with "combined" mode disabled).

If there are multiple processes that are creating jobs (for example,
if your are running multiple servers or your WSGI configuration uses
multiple processes), then you **CANNOT** run pyramid-scheduler in
combined mode since then there will be multiple processes executing
the jobs leading to multiple executions.

Typically, combined mode is used during development where a single
`pserve` instance will be used. Then, in production mode, you will
have multiple servers and WSGI processes that generate jobs, that
are then executed by a single background process (managed via the
`pscheduler` daemon).

Pyramid-scheduler supports multiple "queues". The main reason to use
separate queues is that the `pscheduler` can be configured to only
process jobs for specific queues, which means that multiple
pschedulers can work in parallel as long as they are listening to
different queues. (A later enhancement is planned to allow multiple
pschedulers to handle a single queue.)

**IMPORTANT**: currently, the following limitations exist:

* The callback handler provided as the first argument to the scheduler
  add_*_job() methods must be a normal module-level defined
  function. It *cannot* be a lambda function, an internal function, a
  method, or otherwise function that is not globally resolvable using
  standard dot-notation.

* The `args` and `kwargs` parameters must all be completely
  pickle-able.

* Deferred jobs that are scheduled to occur further in the past than
  `misfire_grace_time` will be silently dropped.

* Jobs that take a `date` or `start_date` parameter can specify those
  values either as an epoch int or float or as a datetime object. If
  a datetime is provided, it **must** be timezone "naive" (see the
  documentation of datetime_).

Under the hood, pyramid-scheduler uses APScheduler to do the actual
processing and scheduling. For messaging between the job creators and
the pscheduler background process, it uses Kombu messaging, which
supports a variety of transports including Redis and SQLAlchemy. This
package was developed as an alternative to celery, due to severe
limitations found in the celery API and shortcomings in the actual
implementation.


## Installation

You can manually install it by running:

```
$ pip install pyramid-scheduler
```

However, a better approach is to use standard python distribution
utilities, and add pyramid-scheduler as a dependency in your project's
`install_requires` parameter in your ``setup.py``. Then run a ``python
setup.py develop``.

Then, enable the package either in your INI file via:

```
pyramid.includes = pyramid-scheduler
```

or in code in your package's application initialization via:

``` python
def main(global_config, **settings):
  # ...
  config.include('pyramid-scheduler')
  # ...
```


## Configuration

The following configuration options (placed in the "[app:main]"
section of your INI file):

TODO: add documentation

* scheduler.combined
* scheduler.queues
* scheduler.broker.url
* scheduler.housekeeping
* scheduler.jobstore.default.class
* scheduler.misfire_grace_time


## Background daemon

TODO: add documentation


.. _APScheduler: https://pypi.python.org/pypi/APScheduler
.. _Kombu: https://pypi.python.org/pypi/kombu
.. _datetime: http://docs.python.org/2/library/datetime.html
