Update Copywrite year to be 2019
[civicrm-core.git] / CRM / Core / Invoke.php
CommitLineData
6a488035
TO
1<?php
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 * Given an argument list, invoke the appropriate CRM function
31 * Serves as a wrapper between the UserFrameWork and Core CRM
32 *
33 * @package CRM
6b83d5bd 34 * @copyright CiviCRM LLC (c) 2004-2019
6a488035
TO
35 */
36class CRM_Core_Invoke {
37
38 /**
be7dea3f
TO
39 * This is the main front-controller that integrates with the CMS. Any
40 * page-request that is sent to the CMS and intended for CiviCRM should
41 * be processed by invoke().
6a488035 42 *
be7dea3f
TO
43 * @param array $args
44 * The parts of the URL which identify the intended CiviCRM page
45 * (e.g. array('civicrm', 'event', 'register')).
46 * @return string
47 * HTML. For non-HTML content, invoke() may call print() and exit().
6a488035 48 *
6a488035 49 */
00be9182 50 public static function invoke($args) {
6a488035
TO
51 try {
52 return self::_invoke($args);
dcc4f6a7 53 }
dcc4f6a7 54 catch (Exception $e) {
be7dea3f 55 CRM_Core_Error::handleUnhandledException($e);
6a488035
TO
56 }
57 }
58
a0ee3941 59 /**
be7dea3f
TO
60 * This is the same as invoke(), but it does *not* include exception
61 * handling.
62 *
63 * @param array $args
64 * The parts of the URL which identify the intended CiviCRM page
65 * (e.g. array('civicrm', 'event', 'register')).
66 * @return string
67 * HTML. For non-HTML content, invoke() may call print() and exit().
a0ee3941 68 */
be7dea3f 69 public static function _invoke($args) {
6a488035 70 if ($args[0] !== 'civicrm') {
408b79bf 71 return NULL;
6a488035 72 }
1ebbf8bf
NG
73 // CRM-15901: Turn off PHP errors display for all ajax calls
74 if (CRM_Utils_Array::value(1, $args) == 'ajax' || CRM_Utils_Array::value('snippet', $_REQUEST)) {
75 ini_set('display_errors', 0);
76 }
6a488035
TO
77
78 if (!defined('CIVICRM_SYMFONY_PATH')) {
be7dea3f
TO
79 // Traditional Civi invocation path
80 self::hackMenuRebuild($args); // may exit
81 self::init($args);
82 self::hackStandalone($args);
83 $item = self::getItem($args);
84 return self::runItem($item);
0db6c3e1
TO
85 }
86 else {
6a488035
TO
87 // Symfony-based invocation path
88 require_once CIVICRM_SYMFONY_PATH . '/app/bootstrap.php.cache';
89 require_once CIVICRM_SYMFONY_PATH . '/app/AppKernel.php';
4eeb9a5b 90 $kernel = new AppKernel('dev', TRUE);
6a488035
TO
91 $kernel->loadClassCache();
92 $response = $kernel->handle(Symfony\Component\HttpFoundation\Request::createFromGlobals());
c24c4679
TO
93 if (preg_match(':^text/html:', $response->headers->get('Content-Type'))) {
94 // let the CMS handle the trappings
95 return $response->getContent();
0db6c3e1
TO
96 }
97 else {
c24c4679
TO
98 $response->send();
99 exit();
100 }
6a488035
TO
101 }
102 }
353ffa53 103
6a488035
TO
104 /**
105 * Hackish support /civicrm/menu/rebuild
106 *
6a0b768e
TO
107 * @param array $args
108 * List of path parts.
6a488035
TO
109 * @void
110 */
111 static public function hackMenuRebuild($args) {
d3e86119 112 if (array('civicrm', 'menu', 'rebuild') == $args || array('civicrm', 'clearcache') == $args) {
6a488035
TO
113 // ensure that the user has a good privilege level
114 if (CRM_Core_Permission::check('administer CiviCRM')) {
115 self::rebuildMenuAndCaches();
116 CRM_Core_Session::setStatus(ts('Cleared all CiviCRM caches (database, menu, templates)'), ts('Complete'), 'success');
117 return CRM_Utils_System::redirect(); // exits
118 }
119 else {
120 CRM_Core_Error::fatal('You do not have permission to execute this url');
121 }
122 }
123 }
124
125 /**
d09edf64 126 * Perform general setup.
6a488035 127 *
6a0b768e
TO
128 * @param array $args
129 * List of path parts.
6a488035
TO
130 * @void
131 */
132 static public function init($args) {
133 // first fire up IDS and check for bad stuff
134 $config = CRM_Core_Config::singleton();
6a488035
TO
135
136 // also initialize the i18n framework
137 require_once 'CRM/Core/I18n.php';
138 $i18n = CRM_Core_I18n::singleton();
139 }
140
141 /**
142 * Hackish support for /standalone/*
143 *
6a0b768e
TO
144 * @param array $args
145 * List of path parts.
6a488035
TO
146 * @void
147 */
148 static public function hackStandalone($args) {
149 $config = CRM_Core_Config::singleton();
150 if ($config->userFramework == 'Standalone') {
151 $session = CRM_Core_Session::singleton();
152 if ($session->get('new_install') !== TRUE) {
153 CRM_Core_Standalone::sidebarLeft();
154 }
155 elseif ($args[1] == 'standalone' && $args[2] == 'register') {
156 CRM_Core_Menu::store();
157 }
158 }
159 }
160
161 /**
162 * Determine which menu $item corresponds to $args
163 *
6a0b768e
TO
164 * @param array $args
165 * List of path parts.
6a488035
TO
166 * @return array; see CRM_Core_Menu
167 */
168 static public function getItem($args) {
169 if (is_array($args)) {
170 // get the menu items
171 $path = implode('/', $args);
0db6c3e1
TO
172 }
173 else {
6a488035
TO
174 $path = $args;
175 }
176 $item = CRM_Core_Menu::get($path);
177
178 // we should try to compute menus, if item is empty and stay on the same page,
179 // rather than compute and redirect to dashboard.
180 if (!$item) {
181 CRM_Core_Menu::store(FALSE);
182 $item = CRM_Core_Menu::get($path);
183 }
184
185 return $item;
186 }
187
188 /**
189 * Given a menu item, call the appropriate controller and return the response
190 *
6a0b768e
TO
191 * @param array $item
192 * See CRM_Core_Menu.
6a488035
TO
193 * @return string, HTML
194 */
195 static public function runItem($item) {
76adcecc
TO
196 $ids = new CRM_Core_IDS();
197 $ids->check($item);
198
6a488035
TO
199 $config = CRM_Core_Config::singleton();
200 if ($config->userFramework == 'Joomla' && $item) {
201 $config->userFrameworkURLVar = 'task';
202
203 // joomla 1.5RC1 seems to push this in the POST variable, which messes
204 // QF and checkboxes
205 unset($_POST['option']);
206 CRM_Core_Joomla::sidebarLeft();
207 }
208
209 // set active Component
210 $template = CRM_Core_Smarty::singleton();
211 $template->assign('activeComponent', 'CiviCRM');
212 $template->assign('formTpl', 'default');
213
214 if ($item) {
215 // CRM-7656 - make sure we send a clean sanitized path to create printer friendly url
216 $printerFriendly = CRM_Utils_System::makeURL(
353ffa53
TO
217 'snippet', FALSE, FALSE,
218 CRM_Utils_Array::value('path', $item)
219 ) . '2';
6a488035
TO
220 $template->assign('printerFriendly', $printerFriendly);
221
222 if (!array_key_exists('page_callback', $item)) {
223 CRM_Core_Error::debug('Bad item', $item);
224 CRM_Core_Error::fatal(ts('Bad menu record in database'));
225 }
226
227 // check that we are permissioned to access this page
228 if (!CRM_Core_Permission::checkMenuItem($item)) {
229 CRM_Utils_System::permissionDenied();
408b79bf 230 return NULL;
6a488035
TO
231 }
232
233 // check if ssl is set
a7488080 234 if (!empty($item['is_ssl'])) {
6a488035
TO
235 CRM_Utils_System::redirectToSSL();
236 }
237
238 if (isset($item['title'])) {
239 CRM_Utils_System::setTitle($item['title']);
240 }
241
242 if (isset($item['breadcrumb']) && !isset($item['is_public'])) {
243 CRM_Utils_System::appendBreadCrumb($item['breadcrumb']);
244 }
245
246 $pageArgs = NULL;
a7488080 247 if (!empty($item['page_arguments'])) {
6a488035
TO
248 $pageArgs = CRM_Core_Menu::getArrayForPathArgs($item['page_arguments']);
249 }
250
251 $template = CRM_Core_Smarty::singleton();
252 if (!empty($item['is_public'])) {
253 $template->assign('urlIsPublic', TRUE);
254 }
255 else {
256 $template->assign('urlIsPublic', FALSE);
06576a03 257 self::statusCheck($template);
6a488035
TO
258 }
259
260 if (isset($item['return_url'])) {
261 $session = CRM_Core_Session::singleton();
262 $args = CRM_Utils_Array::value(
263 'return_url_args',
264 $item,
265 'reset=1'
266 );
267 $session->pushUserContext(CRM_Utils_System::url($item['return_url'], $args));
268 }
269
270 $result = NULL;
c8074a93
TO
271 // WISHLIST: Refactor this. Instead of pattern-matching on page_callback, lookup
272 // page_callback via Civi\Core\Resolver and check the implemented interfaces. This
273 // would require rethinking the default constructor.
274 if (is_array($item['page_callback']) || strpos($item['page_callback'], ':')) {
275 $result = call_user_func(Civi\Core\Resolver::singleton()->get($item['page_callback']));
6a488035
TO
276 }
277 elseif (strstr($item['page_callback'], '_Form')) {
278 $wrapper = new CRM_Utils_Wrapper();
279 $result = $wrapper->run(
280 CRM_Utils_Array::value('page_callback', $item),
281 CRM_Utils_Array::value('title', $item),
282 isset($pageArgs) ? $pageArgs : NULL
283 );
284 }
285 else {
286 $newArgs = explode('/', $_GET[$config->userFrameworkURLVar]);
6a488035
TO
287 $mode = 'null';
288 if (isset($pageArgs['mode'])) {
289 $mode = $pageArgs['mode'];
290 unset($pageArgs['mode']);
291 }
292 $title = CRM_Utils_Array::value('title', $item);
99218b4b 293 if (strstr($item['page_callback'], '_Page') || strstr($item['page_callback'], '\\Page\\')) {
408b79bf 294 $object = new $item['page_callback']($title, $mode);
6c2473d5 295 $object->urlPath = explode('/', $_GET[$config->userFrameworkURLVar]);
6a488035 296 }
99218b4b 297 elseif (strstr($item['page_callback'], '_Controller') || strstr($item['page_callback'], '\\Controller\\')) {
6a488035
TO
298 $addSequence = 'false';
299 if (isset($pageArgs['addSequence'])) {
300 $addSequence = $pageArgs['addSequence'];
301 $addSequence = $addSequence ? 'true' : 'false';
302 unset($pageArgs['addSequence']);
303 }
408b79bf 304 $object = new $item['page_callback']($title, TRUE, $mode, NULL, $addSequence);
6a488035
TO
305 }
306 else {
307 CRM_Core_Error::fatal();
308 }
309 $result = $object->run($newArgs, $pageArgs);
310 }
311
312 CRM_Core_Session::storeSessionObjects();
313 return $result;
314 }
315
316 CRM_Core_Menu::store();
317 CRM_Core_Session::setStatus(ts('Menu has been rebuilt'), ts('Complete'), 'success');
318 return CRM_Utils_System::redirect();
319 }
320
321 /**
d09edf64 322 * This function contains the default action.
6a488035
TO
323 *
324 * @param $action
325 *
77b97be7
EM
326 * @param $contact_type
327 * @param $contact_sub_type
328 *
6a488035 329 */
00be9182 330 public static function form($action, $contact_type, $contact_sub_type) {
6a488035
TO
331 CRM_Utils_System::setUserContext(array('civicrm/contact/search/basic', 'civicrm/contact/view'));
332 $wrapper = new CRM_Utils_Wrapper();
333
334 $properties = CRM_Core_Component::contactSubTypeProperties($contact_sub_type, 'Edit');
335 if ($properties) {
336 $wrapper->run($properties['class'], ts('New %1', array(1 => $contact_sub_type)), $action, TRUE);
337 }
338 else {
339 $wrapper->run('CRM_Contact_Form_Contact', ts('New Contact'), $action, TRUE);
340 }
341 }
342
6a488035 343 /**
f55dd135 344 * Show status in the footer (admin only)
6a488035 345 *
fa8dc18c 346 * @param CRM_Core_Smarty $template
6a488035 347 */
097c681e 348 public static function statusCheck($template) {
f55dd135 349 if (CRM_Core_Config::isUpgradeMode() || !CRM_Core_Permission::check('administer CiviCRM')) {
097c681e
AH
350 return;
351 }
f55dd135 352 // always use cached results - they will be refreshed by the session timer
b1fc1ab0 353 $status = Civi::cache('checks')->get('systemStatusCheckResult');
f55dd135 354 $template->assign('footer_status_severity', $status);
f608a24a 355 $template->assign('footer_status_message', CRM_Utils_Check::toStatusLabel($status));
097c681e 356 }
6a488035 357
a0ee3941
EM
358 /**
359 * @param bool $triggerRebuild
360 * @param bool $sessionReset
361 *
362 * @throws Exception
363 */
00be9182 364 public static function rebuildMenuAndCaches($triggerRebuild = FALSE, $sessionReset = FALSE) {
6a488035
TO
365 $config = CRM_Core_Config::singleton();
366 $config->clearModuleList();
367
ae2cab23
TO
368 // also cleanup all caches
369 $config->cleanupCaches($sessionReset || CRM_Utils_Request::retrieve('sessionReset', 'Boolean', CRM_Core_DAO::$_nullObject, FALSE, 0, 'GET'));
370
6a488035
TO
371 CRM_Core_Menu::store();
372
373 // also reset navigation
374 CRM_Core_BAO_Navigation::resetNavigation();
375
6a488035
TO
376 // also cleanup module permissions
377 $config->cleanupPermissions();
378
9762f6ff
CW
379 // rebuild word replacement cache - pass false to prevent operations redundant with this fn
380 CRM_Core_BAO_WordReplacement::rebuild(FALSE);
76dca235 381
76bd16ab 382 Civi::service('settings_manager')->flush();
9762f6ff 383 // Clear js caches
4cc9b813 384 CRM_Core_Resources::singleton()->flushStrings()->resetCacheCode();
ab89fdde 385 CRM_Case_XMLRepository::singleton(TRUE);
1fcf16cc 386
6a488035
TO
387 // also rebuild triggers if requested explicitly
388 if (
389 $triggerRebuild ||
390 CRM_Utils_Request::retrieve('triggerRebuild', 'Boolean', CRM_Core_DAO::$_nullObject, FALSE, 0, 'GET')
391 ) {
392 CRM_Core_DAO::triggerRebuild();
393 }
95a90cba 394 CRM_Core_DAO_AllCoreTables::reinitializeCache(TRUE);
6a488035
TO
395 CRM_Core_ManagedEntities::singleton(TRUE)->reconcile();
396 }
96025800 397
6a488035 398}