| <?php␊ | 
| /* -*- tab-width: 4; indent-tabs-mode: nil; c-basic-offset: 4 -*- */␊ | 
| /*␊ | 
| # ***** BEGIN LICENSE BLOCK *****␊ | 
| # This file is part of InDefero, an open source project management application.␊ | 
| # Copyright (C) 2008 Céondo Ltd and contributors.␊ | 
| #␊ | 
| # InDefero is free software; you can redistribute it and/or modify␊ | 
| # it under the terms of the GNU General Public License as published by␊ | 
| # the Free Software Foundation; either version 2 of the License, or␊ | 
| # (at your option) any later version.␊ | 
| #␊ | 
| # InDefero is distributed in the hope that it will be useful,␊ | 
| # but WITHOUT ANY WARRANTY; without even the implied warranty of␊ | 
| # MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE.  See the␊ | 
| n# GNU General Public License for more details.␊ | 
| #␊ | 
| # You should have received a copy of the GNU General Public License␊ | 
| # along with this program; if not, write to the Free Software␊ | 
| # Foundation, Inc., 51 Franklin St, Fifth Floor, Boston, MA  02110-1301  USA␊ | 
| #␊ | 
| # ***** END LICENSE BLOCK ***** */␊ | 
| ␊ | 
| ␊ | 
| /**␊ | 
| * Queue system for the management of asynchronous operations.␊ | 
| *␊ | 
| * Anybody can add an item to the queue and any application can␊ | 
| * register itself to process an item from the queue.␊ | 
| *␊ | 
| * An item in the queue is considered as fully processed when all the␊ | 
| * handlers have processed it successfully.␊ | 
| *␊ | 
| * To push a new item in the queue:␊ | 
| *␊ | 
| * <code>␊ | 
| * $item = new IDF_Queue();␊ | 
| * $item->type = 'new_commit';␊ | 
| * $item->payload = array('what', 'ever', array('data'));␊ | 
| * $item->create();␊ | 
| * </code>␊ | 
| *␊ | 
| * To process one item from the queue, you first need to register an␊ | 
| * handler, by adding the following in your relations.php file before␊ | 
| * the return statement or in your config file.␊ | 
| *␊ | 
| * <code>␊ | 
| * Pluf_Signal::connect('IDF_Queue::processItem', ␊ | 
| *                       array('YourApp_Class', 'processItem'));␊ | 
| * </code>␊ | 
| *␊ | 
| * The processItem method will be called with two arguments, the first␊ | 
| * is the name of the signal ('IDF_Queue::processItem') and the second␊ | 
| * is an array with:␊ | 
| *␊ | 
| * <code>␊ | 
| * array('item' => $item,␊ | 
| *       'res' => array('OtherApp_Class::handler' => false,␊ | 
| *                      'FooApp_Class::processItem' => true));␊ | 
| * </code>␊ | 
| *␊ | 
| * When you process an item, you need first to check if the type is␊ | 
| * corresponding to what you want to work with, then you need to check␊ | 
| * in 'res' if you have not already processed successfully the item,␊ | 
| * that is the key 'YourApp_Class::processItem' must be set to true,␊ | 
| * and then you can process the item. At the end of your processing,␊ | 
| * you need to modify by reference the 'res' key to add your status.␊ | 
| *␊ | 
| * All the data except for the type is in the payload, this makes the␊ | 
| * queue flexible to manage many different kind of tasks.␊ | 
| *␊ | 
| */␊ | 
| class IDF_Queue extends Pluf_Model␊ | 
| {␊ | 
| public $_model = __CLASS__;␊ | 
| ␊ | 
| function init()␊ | 
| {␊ | 
| $this->_a['table'] = 'idf_queue';␊ | 
| $this->_a['model'] = __CLASS__;␊ | 
| $this->_a['cols'] = array(␊ | 
| // It is mandatory to have an "id" column.␊ | 
| 'id' =>␊ | 
| array(␊ | 
| 'type' => 'Pluf_DB_Field_Sequence',␊ | 
| 'blank' => true, ␊ | 
| ),␊ | 
| 'status' => ␊ | 
| array(␊ | 
| 'type' => 'Pluf_DB_Field_Integer',␊ | 
| 'blank' => false,␊ | 
| 'choices' => array(␊ | 
| 'pending' => 0,␊ | 
| 'in_progress' => 1,␊ | 
| 'need_retry' => 2,␊ | 
| 'done' => 3,␊ | 
| 'error' => 4,␊ | 
| ),␊ | 
| 'default' => 0,␊ | 
| ),␊ | 
| 'trials' =>␊ | 
| array(␊ | 
| 'type' => 'Pluf_DB_Field_Integer',␊ | 
| 'default' => 0,␊ | 
| ),␊ | 
| 'type' =>␊ | 
| array(␊ | 
| 'type' => 'Pluf_DB_Field_Varchar',␊ | 
| 'blank' => false,␊ | 
| 'size' => 50,␊ | 
| ),␊ | 
| 'payload' =>␊ | 
| array(␊ | 
| 'type' => 'Pluf_DB_Field_Serialized',␊ | 
| 'blank' => false,␊ | 
| ),␊ | 
| 'results' =>␊ | 
| array(␊ | 
| 'type' => 'Pluf_DB_Field_Serialized',␊ | 
| 'blank' => false,␊ | 
| ),␊ | 
| 'lasttry_dtime' =>␊ | 
| array(␊ | 
| 'type' => 'Pluf_DB_Field_Datetime',␊ | 
| 'blank' => true,␊ | 
| ),␊ | 
| 'creation_dtime' =>␊ | 
| array(␊ | 
| 'type' => 'Pluf_DB_Field_Datetime',␊ | 
| 'blank' => true,␊ | 
| ),␊ | 
| );␊ | 
| }␊ | 
| ␊ | 
| function preSave($create=false)␊ | 
| {␊ | 
| if ($create) {␊ | 
| $this->creation_dtime = gmdate('Y-m-d H:i:s');␊ | 
| $this->lasttry_dtime = gmdate('Y-m-d H:i:s');␊ | 
| $this->results = array();␊ | 
| }␊ | 
| }␊ | 
| ␊ | 
| /**␊ | 
| * The current item is going to be processed.␊ | 
| */␊ | 
| function processItem()␊ | 
| {␊ | 
| /**␊ | 
| * [signal]␊ | 
| *␊ | 
| * IDF_Queue::processItem␊ | 
| *␊ | 
| * [sender]␊ | 
| *␊ | 
| * IDF_Queue␊ | 
| *␊ | 
| * [description]␊ | 
| *␊ | 
| * This signal allows an application to run an asynchronous␊ | 
| * job. The handler gets the queue item and the results from␊ | 
| * the previous run. If the handler key is not set, then the␊ | 
| * job was not run. If set it can be either true (already done)␊ | 
| * or false (error at last run).␊ | 
| *␊ | 
| * [parameters]␊ | 
| *␊ | 
| * array('item' => $item, 'res' => $res)␊ | 
| *␊ | 
| */␊ | 
| $params = array('item' => $this, 'res' => $this->results);␊ | 
| Pluf_Signal::send('IDF_Queue::processItem',␊ | 
| 'IDF_Queue', $params);␊ | 
| $this->status = 3; // Success␊ | 
| foreach ($params['res'] as $handler=>$ok) {␊ | 
| if (!$ok) {␊ | 
| $this->status = 2; // Set to need retry␊ | 
| $this->trials += 1;␊ | 
| break;␊ | 
| }␊ | 
| }␊ | 
| $this->results = $params['res'];␊ | 
| $this->lasttry_dtime = gmdate('Y-m-d H:i:s');␊ | 
| $this->update();␊ | 
| }␊ | 
| }␊ |