Skip to content

nebojsac/cakephp-queue

 
 

Repository files navigation

CakePHP Queue Plugin

modified by Mark Scherer (dereuromark)

  • cake2.x support
  • some minor fixes
  • added crontasks (as a different approach on specific problems)
  • possible Tools Plugin dependencies for frontend access via /admin/queue

new:

  • config key "queue" is now "Queue" ($config['Queue'][...])

Background:

This is a very simple and minimalistic job Queue (or deferred-task) system for CakePHP.

Overall functionality is inspired by systems like Gearman, Beanstalk or dropr, but without any illusion to compete with these more advanced Systems.

The Plugin is an attempt to provide a basic, simple to use method to enable deferred job execution, without the hassle of setting up or running an extra queue daemon, while integrating nicely into CakePHP and also simplifying the creation of worker scripts.

Why use deferred execution?

deferred execution makes sense (especially in PHP) when your page wants' to execute tasks, which are not directly related to rendering the current page. For instance, in a BBS-type system, a new users post might require the creation of multiple personalized email messages, notifying other users of the new content. Creating and sending these emails is completely irrelevant to the currently active user, and should not increase page response time. Another example would be downloading, extraction and/or analyzing an external file per request of the user. The regular solution to these problems would be to create specialized cronjobs which use specific database states to determine which action should be done.

The Queue Plugin provides a simple method to create and run such non-user-interaction-critical tasks.

While you can run multiple workers, and can (to some extend) spread these workers to different machines via a shared database, you should seriously consider using a more advanced system for high volume/high number of workers systems.

Status

Build Status

Installation:

  • Copy the files in this directory into APP/Plugin/Queue

  • Enable the plugin within your APP/Config/bootstrap.php (unless you loadAll)

      CakePlugin::load('Queue');
    
  • Run the following command in the cake console to create the tables:

      cake Schema create -p Queue
    

Configuration:

The plugin allows some simple runtime configuration. You may create a file called "queue.php" inside your 'APP/Config' folder (NOT the plugins config folder) to set the following values:

  • seconds to sleep() when no executable job is found

      $config['Queue']['sleeptime'] = 10;
    
  • Propability in percent of a old job cleanup happening

      $config['Queue']['gcprop'] = 10;
    
  • Default timeout after which a job is requeued if the worker doesn't report back

      $config['Queue']['defaultworkertimeout'] = 120;
    
  • Default number of retries if a job fails or times out.

      $config['Queue']['defaultworkerretries'] = 4;
    
  • Seconds of running time after which the worker will terminate (0 = unlimited)

      $config['Queue']['workermaxruntime'] = 0;
    

    Warning: Do not use 0 if you are using a cronjob to permanantly start a new worker once in a while and if you do not exit on idle.

  • Should a Workerprocess quit when there are no more tasks for it to execute (true = exit, false = keep running)

      $config['Queue']['exitwhennothingtodo'] = false;
    

You can also drop the configuration into an existing config file that is already been loaded. The values above are the default settings which apply, when no configuration is found.

Usage:

Run the following using the CakePHP shell:

  • Display Help message:

      cake Queue.Queue
    
  • Try to call the cli add() function on a task:

      cake Queue.Queue add <TaskName>
    

    Tasks may or may not provide this functionality.

  • Run a queue worker, which will look for a pending task it can execute:

      cake Queue.Queue runworker
    

    The worker will always try to find jobs matching its installed Tasks.

Most tasks you will not trigger from the console, but the actual APP code. Use the model access for QueueTask to do that:

For sending emails, for example:

// In your controller
$this->loadModel('Queue.QueuedTask');
$this->QueuedTask->createJob('Email', array('to' => '[email protected]', ...)));

// Somewhere in the model
ClassRegistry::init('Queue.QueuedTask')->createJob('Email',
	array('to' => '[email protected]', ...)));

It will use your custom APP QueueEmailTask to send out emails via CLI.

Important: Do not forget to set your domain when sending from CLI.

Notes

<TaskName> may either be the complete classname (eg. QueueExample) or the shorthand without the leading "Queue" (eg. Example)

Also note that you dont need to add the type ("Task"): cake Queue.Queue add SpecialExample for QueueSpecialExampleTask.

Custom tasks should be placed in APP/Console/Command/Task. Tasks should be named QueueSomethingTask.php and implement a "QueueSomethingTask", keeping CakePHP naming conventions intact.

Plugin tasks go in APP/Plugin/PluginName/Console/Command/Task.

A detailed Example task can be found in /Console/Command/Task/QueueExampleTask.php inside this folder.

Setting up the trigger cronjob

As outlined in the book you can easily set up a cronjob to start a new worker:

*/10  *    *    *    *  cd /full/path/to/app && Console/cake Queue.Queue runworker

This would start a new worker every 10 minutes. If you configure your max life time of a worker to 15 minutes, you got a small overlap where two workers would run simulaniously. If you lower the 10 minutes and raise the lifetime, you get quite a few overlapping workers and thus more "parallel" processing power. Play around with it, but just don't shoot over the top.

TODO

  • Add priority
  • Cleanup and better test coverage

About

A minimalistic JobQueue Plugin for CakePHP

Resources

License

Stars

Watchers

Forks

Packages

No packages published