]> CyberLeo.Net >> Repos - Github/sugarcrm.git/blob - modules/SchedulersJobs/SchedulersJob.php
Release 6.5.0RC1
[Github/sugarcrm.git] / modules / SchedulersJobs / SchedulersJob.php
1 <?php
2 if(!defined('sugarEntry') || !sugarEntry) die('Not A Valid Entry Point');
3 /*********************************************************************************
4  * SugarCRM Community Edition is a customer relationship management program developed by
5  * SugarCRM, Inc. Copyright (C) 2004-2012 SugarCRM Inc.
6  * 
7  * This program is free software; you can redistribute it and/or modify it under
8  * the terms of the GNU Affero General Public License version 3 as published by the
9  * Free Software Foundation with the addition of the following permission added
10  * to Section 15 as permitted in Section 7(a): FOR ANY PART OF THE COVERED WORK
11  * IN WHICH THE COPYRIGHT IS OWNED BY SUGARCRM, SUGARCRM DISCLAIMS THE WARRANTY
12  * OF NON INFRINGEMENT OF THIRD PARTY RIGHTS.
13  * 
14  * This program is distributed in the hope that it will be useful, but WITHOUT
15  * ANY WARRANTY; without even the implied warranty of MERCHANTABILITY or FITNESS
16  * FOR A PARTICULAR PURPOSE.  See the GNU Affero General Public License for more
17  * details.
18  * 
19  * You should have received a copy of the GNU Affero General Public License along with
20  * this program; if not, see http://www.gnu.org/licenses or write to the Free
21  * Software Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA
22  * 02110-1301 USA.
23  * 
24  * You can contact SugarCRM, Inc. headquarters at 10050 North Wolfe Road,
25  * SW2-130, Cupertino, CA 95014, USA. or at email address contact@sugarcrm.com.
26  * 
27  * The interactive user interfaces in modified source and object code versions
28  * of this program must display Appropriate Legal Notices, as required under
29  * Section 5 of the GNU Affero General Public License version 3.
30  * 
31  * In accordance with Section 7(b) of the GNU Affero General Public License version 3,
32  * these Appropriate Legal Notices must retain the display of the "Powered by
33  * SugarCRM" logo. If the display of the logo is not reasonably feasible for
34  * technical reasons, the Appropriate Legal Notices must display the words
35  * "Powered by SugarCRM".
36  ********************************************************************************/
37
38
39 /**
40  * Job queue job
41  * @api
42  */
43 class SchedulersJob extends Basic
44 {
45     const JOB_STATUS_QUEUED = 'queued';
46     const JOB_STATUS_RUNNING = 'running';
47     const JOB_STATUS_DONE = 'done';
48
49     const JOB_PENDING = 'queued';
50     const JOB_PARTIAL = 'partial';
51     const JOB_SUCCESS = 'success';
52     const JOB_FAILURE = 'failure';
53
54     // schema attributes
55         public $id;
56         public $name;
57         public $deleted;
58         public $date_entered;
59         public $date_modified;
60         public $scheduler_id;
61         public $execute_time; // when to execute
62     public $status;
63     public $resolution;
64     public $message;
65         public $target; // URL or function name
66     public $data; // Data set
67     public $requeue; // Requeue on failure?
68     public $retry_count;
69     public $failure_count;
70     public $job_delay=0; // Frequency to run it
71     public $assigned_user_id; // User under which the task is running
72     public $client; // Client ID that owns this job
73     public $execute_time_db;
74     public $percent_complete; // how much of the job is done
75
76         // standard SugarBean child attrs
77         var $table_name         = "job_queue";
78         var $object_name                = "SchedulersJob";
79         var $module_dir         = "SchedulersJobs";
80         var $new_schema         = true;
81         var $process_save_dates = true;
82         // related fields
83         var $job_name;  // the Scheduler's 'name' field
84         var $job;               // the Scheduler's 'job' field
85         // object specific attributes
86         var $user; // User object
87         var $scheduler; // Scheduler parent
88         public $min_interval = 30; // minimal interval for job reruns
89         protected $job_done = true;
90
91         /**
92          * Job constructor.
93          */
94         function SchedulersJob()
95         {
96         parent::Basic();
97         if(!empty($GLOBALS['sugar_config']['jobs']['min_retry_interval'])) {
98             $this->min_interval = $GLOBALS['sugar_config']['jobs']['min_retry_interval'];
99         }
100         }
101
102         public function check_date_relationships_load()
103         {
104         // Hack to work around the mess with dates being auto-converted to user format on retrieve
105             $this->execute_time_db = $this->db->fromConvert($this->execute_time, 'datetime');
106             parent::check_date_relationships_load();
107     }
108         /**
109      * handleDateFormat
110      *
111          * This function handles returning a datetime value.  It allows a user instance to be passed in, but will default to the
112      * user member variable instance if none is found.
113      *
114          * @param string $date String value of the date to calculate, defaults to 'now'
115          * @param object $user The User instance to use in calculating the time value, if empty, it will default to user member variable
116          * @param boolean $user_format Boolean indicating whether or not to convert to user's time format, defaults to false
117      *
118          * @return string Formatted datetime value
119          */
120         function handleDateFormat($date='now', $user=null, $user_format=false) {
121                 global $timedate;
122                 
123                 if(!isset($timedate) || empty($timedate))
124         {
125                         $timedate = new TimeDate();
126                 }
127                 
128                 // get user for calculation
129                 $user = (empty($user)) ? $this->user : $user;
130
131         if($date == 'now')
132         {
133             $dbTime = $timedate->asUser($timedate->getNow(), $user);
134         } else {
135             $dbTime = $timedate->asUser($timedate->fromString($date, $user), $user);
136         }
137
138         // if $user_format is set to true then just return as th user's time format, otherwise, return as database format
139         return $user_format ? $dbTime : $timedate->fromUser($dbTime, $user)->asDb();
140         }
141
142
143         ///////////////////////////////////////////////////////////////////////////
144         ////    SCHEDULERSJOB HELPER FUNCTIONS
145
146         /**
147          * This function takes a passed URL and cURLs it to fake multi-threading with another httpd instance
148          * @param       $job            String in URI-clean format
149          * @param       $timeout        Int value in secs for cURL to timeout. 30 default.
150          */
151         public function fireUrl($job, $timeout=30)
152         {
153         // TODO: figure out what error is thrown when no more apache instances can be spun off
154             // cURL inits
155                 $ch = curl_init();
156                 curl_setopt($ch, CURLOPT_URL, $job); // set url
157                 curl_setopt($ch, CURLOPT_FAILONERROR, true); // silent failure (code >300);
158                 curl_setopt($ch, CURLOPT_FOLLOWLOCATION, true); // do not follow location(); inits - we always use the current
159                 curl_setopt($ch, CURLOPT_FORBID_REUSE, 1);
160                 curl_setopt($ch, CURLOPT_DNS_USE_GLOBAL_CACHE, false);  // not thread-safe
161                 curl_setopt($ch, CURLOPT_RETURNTRANSFER, true); // return into a variable to continue program execution
162                 curl_setopt($ch, CURLOPT_TIMEOUT, $timeout); // never times out - bad idea?
163                 curl_setopt($ch, CURLOPT_CONNECTTIMEOUT, 5); // 5 secs for connect timeout
164                 curl_setopt($ch, CURLOPT_FRESH_CONNECT, true);  // open brand new conn
165                 curl_setopt($ch, CURLOPT_HEADER, true); // do not return header info with result
166                 curl_setopt($ch, CURLOPT_NOPROGRESS, true); // do not have progress bar
167                 $urlparts = parse_url($job);
168                 if(empty($urlparts['port'])) {
169                     if($urlparts['scheme'] == 'https'){
170                                 $urlparts['port'] = 443;
171                         } else {
172                                 $urlparts['port'] = 80;
173                         }
174                 }
175                 curl_setopt($ch, CURLOPT_PORT, $urlparts['port']); // set port as reported by Server
176                 //TODO make the below configurable
177                 curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, false); // most customers will not have Certificate Authority account
178                 curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, false); // most customers will not have Certificate Authority account
179
180                 curl_setopt($ch, CURLOPT_NOSIGNAL, true); // ignore any cURL signals to PHP (for multi-threading)
181                 $result = curl_exec($ch);
182                 $cInfo = curl_getinfo($ch);     //url,content_type,header_size,request_size,filetime,http_code
183                                                                         //ssl_verify_result,total_time,namelookup_time,connect_time
184                                                                         //pretransfer_time,size_upload,size_download,speed_download,
185                                                                         //speed_upload,download_content_length,upload_content_length
186                                                                         //starttransfer_time,redirect_time
187                 if(curl_errno($ch)) {
188                     $this->errors .= curl_errno($ch)."\n";
189                 }
190                 curl_close($ch);
191
192                 if($result !== FALSE && $cInfo['http_code'] < 400) {
193                         $GLOBALS['log']->debug("----->Firing was successful: $job");
194                         $GLOBALS['log']->debug('----->WTIH RESULT: '.strip_tags($result).' AND '.strip_tags(print_r($cInfo, true)));
195                         return true;
196                 } else {
197                         $GLOBALS['log']->fatal("Job failed: $job");
198                         return false;
199                 }
200         }
201         ////    END SCHEDULERSJOB HELPER FUNCTIONS
202         ///////////////////////////////////////////////////////////////////////////
203
204
205         ///////////////////////////////////////////////////////////////////////////
206         ////    STANDARD SUGARBEAN OVERRIDES
207         /**
208          * This function gets DB data and preps it for ListViews
209          */
210         function get_list_view_data()
211         {
212                 global $mod_strings;
213
214                 $temp_array = $this->get_list_view_array();
215                 $temp_array['JOB_NAME'] = $this->job_name;
216                 $temp_array['JOB']              = $this->job;
217
218         return $temp_array;
219         }
220
221         /** method stub for future customization
222          *
223          */
224         function fill_in_additional_list_fields()
225         {
226                 $this->fill_in_additional_detail_fields();
227         }
228
229
230         /**
231          * Mark this job as failed
232          * @param string $message
233          */
234     public function failJob($message = null)
235     {
236         return $this->resolveJob(self::JOB_FAILURE, $message);
237     }
238
239         /**
240          * Mark this job as success
241          * @param string $message
242          */
243     public function succeedJob($message = null)
244     {
245         return $this->resolveJob(self::JOB_SUCCESS, $message);
246     }
247
248     /**
249      * Called if job failed but will be retried
250      */
251     public function onFailureRetry()
252     {
253         // TODO: what we do if job fails, notify somebody?
254         $this->call_custom_logic("job_failure_retry");
255     }
256
257     /**
258      * Called if job has failed and will not be retried
259      */
260     public function onFinalFailure()
261     {
262         // TODO: what we do if job fails, notify somebody?
263         $this->call_custom_logic("job_failure");
264     }
265
266     /**
267      * Resolve job as success or failure
268      * @param string $resolution One of JOB_ constants that define job status
269      * @param string $message
270      * @return bool
271      */
272     public function resolveJob($resolution, $message = null)
273     {
274         $GLOBALS['log']->info("Resolving job {$this->id} as $resolution: $message");
275         if($resolution == self::JOB_FAILURE) {
276             $this->failure_count++;
277             if($this->requeue && $this->retry_count > 0) {
278                 // retry failed job
279                 $this->status = self::JOB_STATUS_QUEUED;
280                 if($this->job_delay < $this->min_interval) {
281                     $this->job_delay = $this->min_interval;
282                 }
283                 $this->execute_time = $GLOBALS['timedate']->getNow()->modify("+{$this->job_delay} seconds")->asDb();
284                 $this->retry_count--;
285                 $GLOBALS['log']->info("Will retry job {$this->id} at {$this->execute_time} ($this->retry_count)");
286                 $this->onFailureRetry();
287             } else {
288                 // final failure
289                 $this->status = self::JOB_STATUS_DONE;
290                 $this->onFinalFailure();
291             }
292         } else {
293             $this->status = self::JOB_STATUS_DONE;
294         }
295         $this->addMessages($message);
296         $this->resolution = $resolution;
297         $this->save();
298         return true;
299     }
300
301     /**
302      * Assemle job messages
303      * Takes messages in $this->message, errors & $message and assembles them into $this->message
304      * @param string $message
305      */
306     protected function addMessages($message)
307     {
308         if(!empty($this->errors)) {
309             $this->message .= $this->errors;
310             $this->errors = '';
311         }
312         if(!empty($message)) {
313             $this->message .= "$message\n";
314         }
315     }
316
317     /**
318      * Rerun this job again
319      * @param string $message
320      * @param string $delay how long to delay (default is job's delay)
321      * @return bool
322      */
323     public function postponeJob($message = null, $delay = null)
324     {
325         $this->status = self::JOB_STATUS_QUEUED;
326         $this->addMessages($message);
327         $this->resolution = self::JOB_PARTIAL;
328         if(empty($delay)) {
329             $delay = intval($this->job_delay);
330         }
331         $this->execute_time = $GLOBALS['timedate']->getNow()->modify("+$delay seconds")->asDb();
332         $GLOBALS['log']->info("Postponing job {$this->id} to {$this->execute_time}: $message");
333
334         $this->save();
335         return true;
336     }
337
338     /**
339      * Delete a job
340      * @see SugarBean::mark_deleted($id)
341      */
342     public function mark_deleted($id)
343     {
344         return $this->db->query("DELETE FROM {$this->table_name} WHERE id=".$this->db->quoted($id));
345     }
346
347     /**
348      * Shutdown handler to be called if something breaks in the middle of the job
349      */
350     public function unexpectedExit()
351     {
352         if(!$this->job_done) {
353             // Job wasn't properly finished, fail it
354             $this->resolveJob(self::JOB_FAILURE, translate('ERR_FAILED', 'SchedulersJobs'));
355         }
356     }
357
358     /**
359      * Run the job by ID
360      * @param string $id
361      * @param string $client Client that is trying to run the job
362      * @return bool|string true on success, false on job failure, error message on failure to run
363      */
364     public static function runJobId($id, $client)
365     {
366         $job = new self();
367         $job->retrieve($id);
368         if(empty($job->id)) {
369             $GLOBALS['log']->fatal("Job $id not found.");
370             return "Job $id not found.";
371         }
372         if($job->status != self::JOB_STATUS_RUNNING) {
373             $GLOBALS['log']->fatal("Job $id is not marked as running.");
374             return "Job $id is not marked as running.";
375         }
376         if($job->client != $client) {
377             $GLOBALS['log']->fatal("Job $id belongs to client {$job->client}, can not run as $client.");
378             return "Job $id belongs to another client, can not run as $client.";
379         }
380         $job->job_done = false;
381         register_shutdown_function(array($job, "unexpectedExit"));
382         $res = $job->runJob();
383         $job->job_done = true;
384         return $res;
385     }
386
387     /**
388      * Error handler, assembles the error messages
389      * @param int $errno
390      * @param string $errstr
391      * @param string $errfile
392      * @param int $errline
393      */
394     public function errorHandler($errno, $errstr, $errfile, $errline)
395     {
396         switch($errno)
397         {
398                 case E_USER_WARNING:
399                 case E_COMPILE_WARNING:
400                 case E_CORE_WARNING:
401                 case E_WARNING:
402                         $type = "Warning";
403                         break;
404                 case E_USER_ERROR:
405                 case E_COMPILE_ERROR:
406                 case E_CORE_ERROR:
407                 case E_ERROR:
408                         $type = "Fatal Error";
409                         break;
410                 case E_PARSE:
411                         $type = "Parse Error";
412                         break;
413                 case E_RECOVERABLE_ERROR:
414                         $type = "Recoverable Error";
415                         break;
416                     default:
417                         // Ignore errors we don't know about
418                         return;
419         }
420         $errstr = strip_tags($errstr);
421         $this->errors .= sprintf(translate('ERR_PHP', 'SchedulersJobs'), $type, $errno, $errstr, $errfile, $errline)."\n";
422     }
423
424     /**
425      * Change current user to given user
426      * @param User $user
427      */
428     protected function sudo($user)
429     {
430         $GLOBALS['current_user'] = $user;
431         // Reset the session
432         if(session_id()) {
433             session_write_close();
434         }
435         if(!headers_sent()) {
436                 session_start();
437             session_regenerate_id();
438                 $_SESSION['is_valid_session']= true;
439                 $_SESSION['user_id'] = $user->id;
440                 $_SESSION['type'] = 'user';
441                 $_SESSION['authenticated_user_id'] = $user->id;
442         }
443     }
444
445     /**
446      * Run this job
447      * @return bool Was the job successful?
448      */
449     public function runJob()
450     {
451         $this->errors = "";
452         $exJob = explode('::', $this->target, 2);
453         if($exJob[0] == 'function') {
454             // set up the current user and drop session
455             if(!empty($this->assigned_user_id)) {
456                 $user = new User();
457                 $user->retrieve($this->assigned_user_id);
458                 if(empty($user->id)) {
459                     $this->resolveJob(self::JOB_FAILURE, sprintf(translate('ERR_NOSUCHUSER', 'SchedulersJobs'), $this->assigned_user_id));
460                     return;
461                 }
462                 $old_user = $GLOBALS['current_user'];
463                 $this->sudo($user);
464             } else {
465                 $this->resolveJob(self::JOB_FAILURE, translate('ERR_NOUSER', 'SchedulersJobs'));
466                 return;
467             }
468                 require_once('modules/Schedulers/_AddJobsHere.php');
469                 $func = $exJob[1];
470                         $GLOBALS['log']->debug("----->SchedulersJob calling function: $func");
471             set_error_handler(array($this, "errorHandler"), E_ALL & ~E_NOTICE & ~E_STRICT);
472                         if(!is_callable($func)) {
473                             $this->resolveJob(self::JOB_FAILURE, sprintf(translate('ERR_CALL', 'SchedulersJobs'), $func));
474                         }
475                         $data = array($this);
476                         if(!empty($this->data)) {
477                             $data[] = $this->data;
478                         }
479             $res = call_user_func_array($func, $data);
480             restore_error_handler();
481             if(isset($old_user)) {
482                 $this->sudo($old_user);
483                 unset($old_user);
484             }
485                         if($this->status == self::JOB_STATUS_RUNNING) {
486                             // nobody updated the status yet - job function could do that
487                         if($res) {
488                             $this->resolveJob(self::JOB_SUCCESS);
489                                 return true;
490                         } else {
491                             $this->resolveJob(self::JOB_FAILURE);
492                             return false;
493                         }
494                         } else {
495                             return $this->resolution != self::JOB_FAILURE;
496                         }
497                 }
498         elseif($exJob[0] == 'url')
499         {
500                         if(function_exists('curl_init')) {
501                                 $GLOBALS['log']->debug('----->SchedulersJob firing URL job: '.$exJob[1]);
502                 set_error_handler(array($this, "errorHandler"), E_ALL & ~E_NOTICE & ~E_STRICT);
503                                 if($this->fireUrl($exJob[1])) {
504                     restore_error_handler();
505                     $this->resolveJob(self::JOB_SUCCESS);
506                                         return true;
507                                 } else {
508                     restore_error_handler();
509                                     $this->resolveJob(self::JOB_FAILURE);
510                                         return false;
511                                 }
512                         } else {
513                             $this->resolveJob(self::JOB_FAILURE, translate('ERR_CURL', 'SchedulersJobs'));
514                         }
515                 }
516         else if ($exJob[0] == 'class')
517         {
518             $tmpJob = new $exJob[1]();
519             if($tmpJob instanceof RunnableSchedulerJob)
520             {
521                 $tmpJob->setJob($this);
522                 return $tmpJob->run($this->data);
523             }
524             else {
525                 $this->resolveJob(self::JOB_FAILURE, sprintf(translate('ERR_JOBTYPE', 'SchedulersJobs'), strip_tags($this->target)));
526             }
527         }
528         else {
529                     $this->resolveJob(self::JOB_FAILURE, sprintf(translate('ERR_JOBTYPE', 'SchedulersJobs'), strip_tags($this->target)));
530                 }
531                 return false;
532     }
533
534 }  // end class Job
535
536 /**
537  * Runnable job queue job
538  *
539  */
540 interface RunnableSchedulerJob
541 {
542     /**
543      * @abstract
544      * @param SchedulersJob $job
545      */
546     public function setJob(SchedulersJob $job);
547
548     /**
549      * @abstract
550      *
551      */
552     public function run($data);
553 }