Implement xKerman/restricted-unserialize package to guard against unsafe unserialize
[civicrm-core.git] / CRM / Report / Utils / Report.php
CommitLineData
6a488035 1<?php
6a488035
TO
2/*
3 +--------------------------------------------------------------------+
fee14197 4 | CiviCRM version 5 |
6a488035 5 +--------------------------------------------------------------------+
6b83d5bd 6 | Copyright CiviCRM LLC (c) 2004-2019 |
6a488035
TO
7 +--------------------------------------------------------------------+
8 | This file is a part of CiviCRM. |
9 | |
10 | CiviCRM is free software; you can copy, modify, and distribute it |
11 | under the terms of the GNU Affero General Public License |
12 | Version 3, 19 November 2007 and the CiviCRM Licensing Exception. |
13 | |
14 | CiviCRM is distributed in the hope that it will be useful, but |
15 | WITHOUT ANY WARRANTY; without even the implied warranty of |
16 | MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. |
17 | See the GNU Affero General Public License for more details. |
18 | |
19 | You should have received a copy of the GNU Affero General Public |
20 | License and the CiviCRM Licensing Exception along |
21 | with this program; if not, contact CiviCRM LLC |
22 | at info[AT]civicrm[DOT]org. If you have questions about the |
23 | GNU Affero General Public License or the licensing of CiviCRM, |
24 | see the CiviCRM license FAQ at http://civicrm.org/licensing |
25 +--------------------------------------------------------------------+
d25dd0ee 26 */
6a488035
TO
27
28/**
29 *
30 * @package CRM
6b83d5bd 31 * @copyright CiviCRM LLC (c) 2004-2019
6a488035
TO
32 * $Id$
33 *
34 */
35class CRM_Report_Utils_Report {
36
74cf4551 37 /**
100fef9d 38 * @param int $instanceID
74cf4551
EM
39 *
40 * @return null|string
41 */
00be9182 42 public static function getValueFromUrl($instanceID = NULL) {
6a488035 43 if ($instanceID) {
0b25329b 44 $optionVal = CRM_Core_DAO::getFieldValue('CRM_Report_DAO_ReportInstance',
6a488035
TO
45 $instanceID,
46 'report_id'
47 );
48 }
49 else {
50 $config = CRM_Core_Config::singleton();
51 $args = explode('/', $_GET[$config->userFrameworkURLVar]);
52
53 // remove 'civicrm/report' from args
54 array_shift($args);
55 array_shift($args);
56
b44e3f84 57 // put rest of argument back in the form of url, which is how value
6a488035
TO
58 // is stored in option value table
59 $optionVal = implode('/', $args);
60 }
61 return $optionVal;
62 }
63
74cf4551 64 /**
100fef9d 65 * @param int $instanceID
74cf4551
EM
66 *
67 * @return array|bool
68 */
00be9182 69 public static function getValueIDFromUrl($instanceID = NULL) {
6a488035
TO
70 $optionVal = self::getValueFromUrl($instanceID);
71
72 if ($optionVal) {
73 $templateInfo = CRM_Core_OptionGroup::getRowValues('report_template', "{$optionVal}", 'value');
be2fb01f 74 return [CRM_Utils_Array::value('id', $templateInfo), $optionVal];
6a488035
TO
75 }
76
77 return FALSE;
78 }
79
74cf4551
EM
80 /**
81 * @param $optionVal
82 *
83 * @return mixed
84 */
00be9182 85 public static function getInstanceIDForValue($optionVal) {
be2fb01f 86 static $valId = [];
6a488035
TO
87
88 if (!array_key_exists($optionVal, $valId)) {
89 $sql = "
90SELECT MIN(id) FROM civicrm_report_instance
91WHERE report_id = %1";
92
be2fb01f 93 $params = [1 => [$optionVal, 'String']];
6a488035
TO
94 $valId[$optionVal] = CRM_Core_DAO::singleValueQuery($sql, $params);
95 }
96 return $valId[$optionVal];
97 }
98
74cf4551
EM
99 /**
100 * @param null $path
101 *
102 * @return mixed
103 */
00be9182 104 public static function getInstanceIDForPath($path = NULL) {
be2fb01f 105 static $valId = [];
6a488035
TO
106
107 // if $path is null, try to get it from url
108 $path = self::getInstancePath();
109
110 if ($path && !array_key_exists($path, $valId)) {
111 $sql = "
112SELECT MIN(id) FROM civicrm_report_instance
113WHERE TRIM(BOTH '/' FROM CONCAT(report_id, '/', name)) = %1";
114
be2fb01f 115 $params = [1 => [$path, 'String']];
6a488035
TO
116 $valId[$path] = CRM_Core_DAO::singleValueQuery($sql, $params);
117 }
118 return CRM_Utils_Array::value($path, $valId);
119 }
120
74cf4551
EM
121 /**
122 * @param $urlValue
123 * @param string $query
124 * @param bool $absolute
100fef9d 125 * @param int $instanceID
74cf4551
EM
126 * @param array $drilldownReport
127 *
128 * @return bool|string
129 */
be2fb01f 130 public static function getNextUrl($urlValue, $query = 'reset=1', $absolute = FALSE, $instanceID = NULL, $drilldownReport = []) {
6a488035 131 if ($instanceID) {
84178120
TO
132 $drilldownInstanceID = FALSE;
133 if (array_key_exists($urlValue, $drilldownReport)) {
0b25329b 134 $drilldownInstanceID = CRM_Core_DAO::getFieldValue('CRM_Report_DAO_ReportInstance', $instanceID, 'drilldown_id', 'id');
84178120 135 }
6a488035 136
84178120 137 if (!$drilldownInstanceID) {
6a488035 138 $drilldownInstanceID = self::getInstanceIDForValue($urlValue);
84178120 139 }
6a488035
TO
140
141 if ($drilldownInstanceID) {
142 return CRM_Utils_System::url("civicrm/report/instance/{$drilldownInstanceID}",
143 "{$query}", $absolute
144 );
145 }
146 else {
147 return FALSE;
148 }
149 }
150 else {
151 return CRM_Utils_System::url("civicrm/report/" . trim($urlValue, '/'),
152 $query, $absolute
153 );
154 }
155 }
156
74cf4551 157 /**
fe482240 158 * get instance count for a template.
74cf4551
EM
159 * @param $optionVal
160 *
161 * @return int|null|string
162 */
00be9182 163 public static function getInstanceCount($optionVal) {
4f99ca55
TO
164 if (empty($optionVal)) {
165 return 0;
84178120 166 }
ae555e90 167
6a488035
TO
168 $sql = "
169SELECT count(inst.id)
170FROM civicrm_report_instance inst
171WHERE inst.report_id = %1";
172
be2fb01f 173 $params = [1 => [$optionVal, 'String']];
6a488035
TO
174 $count = CRM_Core_DAO::singleValueQuery($sql, $params);
175 return $count;
176 }
177
74cf4551
EM
178 /**
179 * @param $fileContent
100fef9d 180 * @param int $instanceID
74cf4551
EM
181 * @param string $outputMode
182 * @param array $attachments
183 *
184 * @return bool
185 */
be2fb01f 186 public static function mailReport($fileContent, $instanceID = NULL, $outputMode = 'html', $attachments = []) {
6a488035
TO
187 if (!$instanceID) {
188 return FALSE;
189 }
190
191 list($domainEmailName,
192 $domainEmailAddress
353ffa53 193 ) = CRM_Core_BAO_Domain::getNameAndEmail();
6a488035 194
be2fb01f
CW
195 $params = ['id' => $instanceID];
196 $instanceInfo = [];
0b25329b 197 CRM_Core_DAO::commonRetrieve('CRM_Report_DAO_ReportInstance',
6a488035
TO
198 $params,
199 $instanceInfo
200 );
201
be2fb01f 202 $params = [];
6a488035 203 $params['groupName'] = 'Report Email Sender';
353ffa53 204 $params['from'] = '"' . $domainEmailName . '" <' . $domainEmailAddress . '>';
6a488035 205 //$domainEmailName;
353ffa53 206 $params['toName'] = "";
6a488035 207 $params['toEmail'] = CRM_Utils_Array::value('email_to', $instanceInfo);
353ffa53 208 $params['cc'] = CRM_Utils_Array::value('email_cc', $instanceInfo);
6a488035 209 $params['subject'] = CRM_Utils_Array::value('email_subject', $instanceInfo);
a7488080 210 if (empty($instanceInfo['attachments'])) {
be2fb01f 211 $instanceInfo['attachments'] = [];
6a488035
TO
212 }
213 $params['attachments'] = array_merge(CRM_Utils_Array::value('attachments', $instanceInfo), $attachments);
214 $params['text'] = '';
215 $params['html'] = $fileContent;
216
217 return CRM_Utils_Mail::send($params);
218 }
219
74cf4551 220 /**
c490a46a 221 * @param CRM_Core_Form $form
74cf4551
EM
222 * @param $rows
223 */
00be9182 224 public static function export2csv(&$form, &$rows) {
6a488035 225 //Mark as a CSV file.
d42a224c 226 CRM_Utils_System::setHttpHeader('Content-Type', 'text/csv');
6a488035
TO
227
228 //Force a download and name the file using the current timestamp.
229 $datetime = date('Ymd-Gi', $_SERVER['REQUEST_TIME']);
d42a224c 230 CRM_Utils_System::setHttpHeader('Content-Disposition', 'attachment; filename=Report_' . $datetime . '.csv');
6a488035
TO
231 echo self::makeCsv($form, $rows);
232 CRM_Utils_System::civiExit();
233 }
234
235 /**
236 * Utility function for export2csv and CRM_Report_Form::endPostProcess
237 * - make CSV file content and return as string.
ea3ddccf 238 *
239 * @param CRM_Core_Form $form
240 * @param array $rows
241 *
242 * @return string
6a488035 243 */
00be9182 244 public static function makeCsv(&$form, &$rows) {
6a488035
TO
245 $config = CRM_Core_Config::singleton();
246 $csv = '';
247
248 // Add headers if this is the first row.
249 $columnHeaders = array_keys($form->_columnHeaders);
250
251 // Replace internal header names with friendly ones, where available.
252 foreach ($columnHeaders as $header) {
253 if (isset($form->_columnHeaders[$header])) {
254 $headers[] = '"' . html_entity_decode(strip_tags($form->_columnHeaders[$header]['title'])) . '"';
255 }
256 }
257 // Add the headers.
258 $csv .= implode($config->fieldSeparator,
353ffa53
TO
259 $headers
260 ) . "\r\n";
6a488035 261
be2fb01f 262 $displayRows = [];
6a488035
TO
263 $value = NULL;
264 foreach ($rows as $row) {
265 foreach ($columnHeaders as $k => $v) {
266 $value = CRM_Utils_Array::value($v, $row);
267 if (isset($value)) {
268 // Remove HTML, unencode entities, and escape quotation marks.
269 $value = str_replace('"', '""', html_entity_decode(strip_tags($value)));
270
271 if (CRM_Utils_Array::value('type', $form->_columnHeaders[$v]) & 4) {
272 if (CRM_Utils_Array::value('group_by', $form->_columnHeaders[$v]) == 'MONTH' ||
273 CRM_Utils_Array::value('group_by', $form->_columnHeaders[$v]) == 'QUARTER'
274 ) {
275 $value = CRM_Utils_Date::customFormat($value, $config->dateformatPartial);
276 }
277 elseif (CRM_Utils_Array::value('group_by', $form->_columnHeaders[$v]) == 'YEAR') {
278 $value = CRM_Utils_Date::customFormat($value, $config->dateformatYear);
279 }
908a11e7
EM
280 elseif ($form->_columnHeaders[$v]['type'] == 12) {
281 // This is a datetime format
282 $value = CRM_Utils_Date::customFormat($value, '%Y-%m-%d %H:%i');
283 }
6a488035
TO
284 else {
285 $value = CRM_Utils_Date::customFormat($value, '%Y-%m-%d');
286 }
287 }
182f5081 288 // Note the reference to a specific field does not belong in this generic class & does not work on all reports.
289 // @todo - fix this properly rather than just supressing the en-otice. Repeat transaction report is a good example.
290 elseif (CRM_Utils_Array::value('type', $form->_columnHeaders[$v]) == 1024 && !empty($row['civicrm_contribution_currency'])) {
39d06e38 291 $value = CRM_Utils_Money::format($value, $row['civicrm_contribution_currency']);
6a488035
TO
292 }
293 $displayRows[$v] = '"' . $value . '"';
294 }
295 else {
b5be5afe 296 $displayRows[$v] = "";
6a488035
TO
297 }
298 }
299 // Add the data row.
300 $csv .= implode($config->fieldSeparator,
353ffa53
TO
301 $displayRows
302 ) . "\r\n";
6a488035
TO
303 }
304
305 return $csv;
306 }
307
74cf4551
EM
308 /**
309 * @return mixed
310 */
00be9182 311 public static function getInstanceID() {
6a488035
TO
312
313 $config = CRM_Core_Config::singleton();
314 $arg = explode('/', $_GET[$config->userFrameworkURLVar]);
315
316 if ($arg[1] == 'report' &&
317 CRM_Utils_Array::value(2, $arg) == 'instance'
318 ) {
319 if (CRM_Utils_Rule::positiveInteger($arg[3])) {
320 return $arg[3];
321 }
322 }
323 }
324
74cf4551
EM
325 /**
326 * @return string
327 */
00be9182 328 public static function getInstancePath() {
6a488035
TO
329 $config = CRM_Core_Config::singleton();
330 $arg = explode('/', $_GET[$config->userFrameworkURLVar]);
331
332 if ($arg[1] == 'report' &&
333 CRM_Utils_Array::value(2, $arg) == 'instance'
334 ) {
335 unset($arg[0], $arg[1], $arg[2]);
336 $path = trim(CRM_Utils_Type::escape(implode('/', $arg), 'String'), '/');
337 return $path;
338 }
339 }
340
74cf4551 341 /**
100fef9d 342 * @param int $instanceId
74cf4551
EM
343 *
344 * @return bool
345 */
00be9182 346 public static function isInstancePermissioned($instanceId) {
6a488035
TO
347 if (!$instanceId) {
348 return TRUE;
349 }
350
be2fb01f
CW
351 $instanceValues = [];
352 $params = ['id' => $instanceId];
0b25329b 353 CRM_Core_DAO::commonRetrieve('CRM_Report_DAO_ReportInstance',
6a488035
TO
354 $params,
355 $instanceValues
356 );
357
358 if (!empty($instanceValues['permission']) &&
359 (!(CRM_Core_Permission::check($instanceValues['permission']) ||
353ffa53
TO
360 CRM_Core_Permission::check('administer Reports')
361 ))
6a488035
TO
362 ) {
363 return FALSE;
364 }
365
366 return TRUE;
367 }
368
369 /**
370 * Check if the user can view a report instance based on their role(s)
371 *
372 * @instanceId string $str the report instance to check
373 *
100fef9d 374 * @param int $instanceId
fd31fa4c 375 *
acb1052e 376 * @return bool
a6c01b45 377 * true if yes, else false
6a488035 378 */
00be9182 379 public static function isInstanceGroupRoleAllowed($instanceId) {
6a488035
TO
380 if (!$instanceId) {
381 return TRUE;
382 }
383
be2fb01f
CW
384 $instanceValues = [];
385 $params = ['id' => $instanceId];
0b25329b 386 CRM_Core_DAO::commonRetrieve('CRM_Report_DAO_ReportInstance',
6a488035
TO
387 $params,
388 $instanceValues
389 );
390 //transform grouprole to array
391 if (!empty($instanceValues['grouprole'])) {
392 $grouprole_array = explode(CRM_Core_DAO::VALUE_SEPARATOR,
393 $instanceValues['grouprole']
394 );
395 if (!CRM_Core_Permission::checkGroupRole($grouprole_array) &&
396 !CRM_Core_Permission::check('administer Reports')
397 ) {
398 return FALSE;
399 }
400 }
401 return TRUE;
402 }
403
74cf4551 404 /**
c490a46a 405 * @param array $params
74cf4551
EM
406 *
407 * @return array
408 */
00be9182 409 public static function processReport($params) {
6a488035
TO
410 $instanceId = CRM_Utils_Array::value('instanceId', $params);
411
412 // hack for now, CRM-8358
413 $_REQUEST['instanceId'] = $instanceId;
414 $_REQUEST['sendmail'] = CRM_Utils_Array::value('sendmail', $params, 1);
884605ca 415
6a488035
TO
416 // if cron is run from terminal --output is reserved, and therefore we would provide another name 'format'
417 $_REQUEST['output'] = CRM_Utils_Array::value('format', $params, CRM_Utils_Array::value('output', $params, 'pdf'));
418 $_REQUEST['reset'] = CRM_Utils_Array::value('reset', $params, 1);
419
420 $optionVal = self::getValueFromUrl($instanceId);
be2fb01f 421 $messages = ["Report Mail Triggered..."];
6a488035
TO
422
423 $templateInfo = CRM_Core_OptionGroup::getRowValues('report_template', $optionVal, 'value');
353ffa53
TO
424 $obj = new CRM_Report_Page_Instance();
425 $is_error = 0;
6a488035 426 if (strstr(CRM_Utils_Array::value('name', $templateInfo), '_Form')) {
be2fb01f
CW
427 $instanceInfo = [];
428 CRM_Report_BAO_ReportInstance::retrieve(['id' => $instanceId], $instanceInfo);
6a488035
TO
429
430 if (!empty($instanceInfo['title'])) {
431 $obj->assign('reportTitle', $instanceInfo['title']);
432 }
433 else {
434 $obj->assign('reportTitle', $templateInfo['label']);
435 }
436
437 $wrapper = new CRM_Utils_Wrapper();
be2fb01f
CW
438 $arguments = [
439 'urlToSession' => [
440 [
884605ca
DL
441 'urlVar' => 'instanceId',
442 'type' => 'Positive',
443 'sessionVar' => 'instanceId',
444 'default' => 'null',
be2fb01f
CW
445 ],
446 ],
21dfd5f5 447 'ignoreKey' => TRUE,
be2fb01f 448 ];
6a488035
TO
449 $messages[] = $wrapper->run($templateInfo['name'], NULL, $arguments);
450 }
451 else {
452 $is_error = 1;
453 if (!$instanceId) {
454 $messages[] = 'Required parameter missing: instanceId';
455 }
456 else {
457 $messages[] = 'Did not find valid instance to execute';
458 }
459 }
460
be2fb01f 461 $result = [
6a488035
TO
462 'is_error' => $is_error,
463 'messages' => implode("\n", $messages),
be2fb01f 464 ];
6a488035
TO
465 return $result;
466 }
467
468 /**
469 * Build a URL query string containing all report filter criteria that are
470 * stipulated in $_GET or in a report Preview, but which haven't yet been
471 * saved in the report instance.
472 *
7e06c9f5
TO
473 * @param array $defaults
474 * The report criteria that aren't coming in as submitted form values, as in CRM_Report_Form::_defaults.
475 * @param array $params
476 * All effective report criteria, as in CRM_Report_Form::_params.
6a488035 477 *
a6c01b45
CW
478 * @return string
479 * URL query string
6a488035 480 */
be2fb01f 481 public static function getPreviewCriteriaQueryParams($defaults = [], $params = []) {
6a488035
TO
482 static $query_string;
483 if (!isset($query_string)) {
484 if (!empty($params)) {
be2fb01f 485 $url_params = $op_values = $string_values = $process_params = [];
6a488035
TO
486
487 // We'll only use $params that are different from what's in $default.
488 foreach ($params as $field_name => $field_value) {
489 if (!array_key_exists($field_name, $defaults) || $defaults[$field_name] != $field_value) {
490 $process_params[$field_name] = $field_value;
491 }
492 }
493 // Criteria stipulated in $_GET will be in $defaults even if they're not
494 // saved, so we can't easily tell if they're saved or not. So just include them.
495 $process_params += $_GET;
496
497 // All $process_params should be passed on if they have an effective value
498 // (in other words, there's no point in propagating blank filters).
499 foreach ($process_params as $field_name => $field_value) {
500 $suffix_position = strrpos($field_name, '_');
353ffa53
TO
501 $suffix = substr($field_name, $suffix_position);
502 $basename = substr($field_name, 0, $suffix_position);
6a488035
TO
503 if ($suffix == '_min' || $suffix == '_max' ||
504 $suffix == '_from' || $suffix == '_to' ||
505 $suffix == '_relative'
506 ) {
507 // For these types, we only keep them if they have a value.
508 if (!empty($field_value)) {
509 $url_params[$field_name] = $field_value;
510 }
511 }
512 elseif ($suffix == '_value') {
513 // These filters can have an effect even without a value
514 // (e.g., values for 'nll' and 'nnll' ops are blank),
515 // so store them temporarily and examine below.
516 $string_values[$basename] = $field_value;
517 $op_values[$basename] = CRM_Utils_Array::value("{$basename}_op", $params);
518 }
519 elseif ($suffix == '_op') {
520 // These filters can have an effect even without a value
521 // (e.g., values for 'nll' and 'nnll' ops are blank),
522 // so store them temporarily and examine below.
523 $op_values[$basename] = $field_value;
524 $string_values[$basename] = $params["{$basename}_value"];
525 }
526 }
527
528 // Check the *_value and *_op criteria and include them if
529 // they'll have an effective value.
530 foreach ($op_values as $basename => $field_value) {
531 if ($field_value == 'nll' || $field_value == 'nnll') {
532 // 'nll' and 'nnll' filters should be included even with empty values.
533 $url_params["{$basename}_op"] = $field_value;
534 }
535 elseif ($string_values[$basename]) {
536 // Other filters are only included if they have a value.
537 $url_params["{$basename}_op"] = $field_value;
538 $url_params["{$basename}_value"] = (is_array($string_values[$basename]) ? implode(',', $string_values[$basename]) : $string_values[$basename]);
539 }
540 }
541 $query_string = http_build_query($url_params);
542 }
543 else {
544 $query_string = '';
545 }
546 }
547 return $query_string;
548 }
549
74cf4551
EM
550 /**
551 * @param $reportUrl
552 *
553 * @return mixed
554 */
00be9182 555 public static function getInstanceList($reportUrl) {
be2fb01f 556 static $instanceDetails = [];
6001af1f 557
481a74f4 558 if (!array_key_exists($reportUrl, $instanceDetails)) {
be2fb01f 559 $instanceDetails[$reportUrl] = [];
6a488035
TO
560
561 $sql = "
562SELECT id, title FROM civicrm_report_instance
563WHERE report_id = %1";
be2fb01f 564 $params = [1 => [$reportUrl, 'String']];
6a488035 565 $result = CRM_Core_DAO::executeQuery($sql, $params);
22e263ad 566 while ($result->fetch()) {
6a488035
TO
567 $instanceDetails[$reportUrl][$result->id] = $result->title . " (ID: {$result->id})";
568 }
569 }
570 return $instanceDetails[$reportUrl];
571 }
96025800 572
6a488035 573}