3 +--------------------------------------------------------------------+
4 | Copyright CiviCRM LLC. All rights reserved. |
6 | This work is published under the GNU AGPLv3 license with some |
7 | permitted exceptions and without any warranty. For full license |
8 | and copyright information, see https://civicrm.org/licensing |
9 +--------------------------------------------------------------------+
13 * This interface defines methods that need to be implemented
14 * by every scheduled job (cron task) in CiviCRM.
17 * @copyright CiviCRM LLC https://civicrm.org/licensing
19 class CRM_Core_JobManager
{
24 * Format is ($id => CRM_Core_ScheduledJob).
31 * @var CRM_Core_ScheduledJob
33 public $currentJob = NULL;
35 public $singleRunParams = [];
37 public $_source = NULL;
42 public function __construct() {
43 $config = CRM_Core_Config
::singleton();
44 $config->fatalErrorHandler
= 'CRM_Core_JobManager_scheduledJobFatalErrorHandler';
46 $this->jobs
= $this->_getJobs();
52 public function execute($auth = TRUE) {
54 $this->logEntry('Starting scheduled jobs execution');
56 if ($auth && !CRM_Utils_System
::authenticateKey(TRUE)) {
57 $this->logEntry('Could not authenticate the site key.');
59 require_once 'api/api.php';
61 // it's not asynchronous at this stage
62 CRM_Utils_Hook
::cron($this);
63 foreach ($this->jobs
as $job) {
64 if ($job->is_active
) {
65 if ($job->needsRunning()) {
66 $this->executeJob($job);
70 $this->logEntry('Finishing scheduled jobs execution.');
72 // Set last cron date for the status check
74 'name' => 'checkLastCron',
75 'check_info' => gmdate('U'),
78 CRM_Core_BAO_StatusPreference
::create($statusPref);
84 public function __destruct() {
91 public function executeJobByAction($entity, $action) {
92 $job = $this->_getJob(NULL, $entity, $action);
93 $this->executeJob($job);
99 public function executeJobById($id) {
100 $job = $this->_getJob($id);
101 $this->executeJob($job);
105 * @param CRM_Core_ScheduledJob $job
107 public function executeJob($job) {
108 $this->currentJob
= $job;
110 // CRM-18231 check if non-production environment.
112 CRM_Core_BAO_Setting
::isAPIJobAllowedToRun($job->apiParams
);
114 catch (Exception
$e) {
115 $this->logEntry('Error while executing ' . $job->name
. ': ' . $e->getMessage());
116 $this->currentJob
= FALSE;
120 $this->logEntry('Starting execution of ' . $job->name
);
123 $singleRunParamsKey = strtolower($job->api_entity
. '_' . $job->api_action
);
125 if (array_key_exists($singleRunParamsKey, $this->singleRunParams
)) {
126 $params = $this->singleRunParams
[$singleRunParamsKey];
129 $params = $job->apiParams
;
132 CRM_Utils_Hook
::preJob($job, $params);
134 $result = civicrm_api($job->api_entity
, $job->api_action
, $params);
136 catch (Exception
$e) {
137 $this->logEntry('Error while executing ' . $job->name
. ': ' . $e->getMessage());
140 CRM_Utils_Hook
::postJob($job, $params, $result);
141 $this->logEntry('Finished execution of ' . $job->name
. ' with result: ' . $this->_apiResultToMessage($result));
142 $this->currentJob
= FALSE;
144 //Disable outBound option after executing the job.
145 $environment = CRM_Core_Config
::environment(NULL, TRUE);
146 if ($environment != 'Production' && !empty($job->apiParams
['runInNonProductionEnvironment'])) {
147 Civi
::settings()->set('mailing_backend', ['outBound_option' => CRM_Mailing_Config
::OUTBOUND_OPTION_DISABLED
]);
152 * Retrieves the list of jobs from the database,
153 * populates class param.
156 * ($id => CRM_Core_ScheduledJob)
158 private function _getJobs() {
160 $dao = new CRM_Core_DAO_Job();
161 $dao->orderBy('name');
162 $dao->domain_id
= CRM_Core_Config
::domainID();
164 while ($dao->fetch()) {
166 CRM_Core_DAO
::storeValues($dao, $temp);
167 $jobs[$dao->id
] = new CRM_Core_ScheduledJob($temp);
173 * Retrieves specific job from the database by id.
174 * and creates ScheduledJob object.
177 * @param null $entity
178 * @param null $action
180 * @return CRM_Core_ScheduledJob
183 private function _getJob($id = NULL, $entity = NULL, $action = NULL) {
184 if (is_null($id) && is_null($action)) {
185 throw new CRM_Core_Exception('You need to provide either id or name to use this method');
187 $dao = new CRM_Core_DAO_Job();
189 $dao->api_entity
= $entity;
190 $dao->api_action
= $action;
192 while ($dao->fetch()) {
193 CRM_Core_DAO
::storeValues($dao, $temp);
194 $job = new CRM_Core_ScheduledJob($temp);
202 * @param array $params
203 * @param null $source
205 public function setSingleRunParams($entity, $job, $params, $source = NULL) {
206 $this->_source
= $source;
207 $key = strtolower($entity . '_' . $job);
208 $this->singleRunParams
[$key] = $params;
209 $this->singleRunParams
[$key]['version'] = 3;
213 * @param string $message
215 public function logEntry($message) {
216 $domainID = CRM_Core_Config
::domainID();
217 $dao = new CRM_Core_DAO_JobLog();
219 $dao->domain_id
= $domainID;
222 * The description is a summary of the message.
223 * HTML tags are stripped from the message.
224 * The description is limited to 240 characters
225 * and has an ellipsis added if it is truncated.
227 $maxDescription = 240;
228 $ellipsis = " (...)";
229 $description = strip_tags($message);
230 if (strlen($description) > $maxDescription) {
231 $description = substr($description, 0, $maxDescription - strlen($ellipsis)) . $ellipsis;
233 $dao->description
= $description;
235 if ($this->currentJob
) {
236 $dao->job_id
= $this->currentJob
->id
;
237 $dao->name
= $this->currentJob
->name
;
238 $dao->command
= ts("Entity:") . " " . $this->currentJob
->api_entity
. " " . ts("Action:") . " " . $this->currentJob
->api_action
;
240 if (!empty($this->currentJob
->parameters
)) {
241 $data .= "\n\nParameters raw (from db settings): \n" . $this->currentJob
->parameters
;
243 $singleRunParamsKey = strtolower($this->currentJob
->api_entity
. '_' . $this->currentJob
->api_action
);
244 if (array_key_exists($singleRunParamsKey, $this->singleRunParams
)) {
245 $data .= "\n\nParameters raw (" . $this->_source
. "): \n" . serialize($this->singleRunParams
[$singleRunParamsKey]);
246 $data .= "\n\nParameters parsed (and passed to API method): \n" . serialize($this->singleRunParams
[$singleRunParamsKey]);
249 $data .= "\n\nParameters parsed (and passed to API method): \n" . serialize($this->currentJob
->apiParams
);
252 $data .= "\n\nFull message: \n" . $message;
264 private function _apiResultToMessage($apiResult) {
265 $status = $apiResult['is_error'] ?
ts('Failure') : ts('Success');
266 $msg = CRM_Utils_Array
::value('error_message', $apiResult, 'empty error_message!');
267 $vals = CRM_Utils_Array
::value('values', $apiResult, 'empty values!');
268 if (is_array($msg)) {
269 $msg = serialize($msg);
271 if (is_array($vals)) {
272 $vals = serialize($vals);
274 $message = $apiResult['is_error'] ?
', Error message: ' . $msg : " (" . $vals . ")";
275 return $status . $message;
285 function CRM_Core_JobManager_scheduledJobFatalErrorHandler($message) {
286 throw new Exception("{$message['message']}: {$message['code']}");