Array(string $realType, string $realName)). */ private $map; private $locals, $settings; private $cache = array(); /** * CRM_Core_Config_MagicMerge constructor. */ public function __construct() { $this->map = self::getPropertyMap(); } /** * Set the map to the property map. */ public function __wakeup() { $this->map = self::getPropertyMap(); } /** * Get a list of $config properties and the entities to which they map. * * This is used for two purposes: * * 1. Runtime: Provide backward-compatible interface for reading these * properties. * 2. Upgrade: Migrate old properties of config_backend into settings. * * @return array */ public static function getPropertyMap() { // Each mapping: $propertyName => Array(0 => $type, 1 => $foreignName|NULL, ...). // If $foreignName is omitted/null, then it's assumed to match the $propertyName. // Other parameters may be specified, depending on the type. return array( // "local" properties are unique to each instance of CRM_Core_Config (each request). 'doNotResetCache' => array('local'), 'inCiviCRM' => array('local'), 'keyDisable' => array('local'), 'userFrameworkFrontend' => array('local'), 'userPermissionTemp' => array('local'), // "runtime" properties are computed from define()s, $_ENV, etc. // See also: CRM_Core_Config_Runtime. 'dsn' => array('runtime'), 'initialized' => array('runtime'), 'userFramework' => array('runtime'), 'userFrameworkClass' => array('runtime'), 'userFrameworkDSN' => array('runtime'), 'userFrameworkURLVar' => array('runtime'), 'userHookClass' => array('runtime'), 'cleanURL' => array('runtime'), 'configAndLogDir' => array('runtime'), 'templateCompileDir' => array('runtime'), 'templateDir' => array('runtime'), // "boot-svc" properties are critical services needed during init. // See also: Civi\Core\Container::getBootService(). 'userSystem' => array('boot-svc'), 'userPermissionClass' => array('boot-svc'), 'userFrameworkBaseURL' => array('user-system', 'getAbsoluteBaseURL'), 'userFrameworkVersion' => array('user-system', 'getVersion'), 'useFrameworkRelativeBase' => array('user-system', 'getRelativeBaseURL'), // ugh typo. // "setting" properties are loaded through the setting layer, esp // table "civicrm_setting" and global $civicrm_setting. // See also: Civi::settings(). 'backtrace' => array('setting'), 'contact_default_language' => array('setting'), 'countryLimit' => array('setting'), 'customTranslateFunction' => array('setting'), 'dateInputFormat' => array('setting'), 'dateformatDatetime' => array('setting'), 'dateformatFull' => array('setting'), 'dateformatPartial' => array('setting'), 'dateformatTime' => array('setting'), 'dateformatYear' => array('setting'), 'dateformatFinancialBatch' => array('setting'), 'dateformatshortdate' => array('setting'), 'debug' => array('setting', 'debug_enabled'), // renamed. 'defaultContactCountry' => array('setting'), 'defaultContactStateProvince' => array('setting'), 'defaultCurrency' => array('setting'), 'defaultSearchProfileID' => array('setting'), 'doNotAttachPDFReceipt' => array('setting'), 'empoweredBy' => array('setting'), 'enableComponents' => array('setting', 'enable_components'), // renamed. 'enableSSL' => array('setting'), 'fatalErrorHandler' => array('setting'), 'fieldSeparator' => array('setting'), 'fiscalYearStart' => array('setting'), 'geoAPIKey' => array('setting'), 'geoProvider' => array('setting'), 'includeAlphabeticalPager' => array('setting'), 'includeEmailInName' => array('setting'), 'includeNickNameInName' => array('setting'), 'includeOrderByClause' => array('setting'), 'includeWildCardInName' => array('setting'), 'inheritLocale' => array('setting'), 'languageLimit' => array('setting'), 'lcMessages' => array('setting'), 'legacyEncoding' => array('setting'), 'logging' => array('setting'), 'mailThrottleTime' => array('setting'), 'mailerBatchLimit' => array('setting'), 'mailerJobSize' => array('setting'), 'mailerJobsMax' => array('setting'), 'mapAPIKey' => array('setting'), 'mapProvider' => array('setting'), 'maxFileSize' => array('setting'), 'maxAttachments' => array('setting', 'max_attachments'), // renamed. 'monetaryDecimalPoint' => array('setting'), 'monetaryThousandSeparator' => array('setting'), 'moneyformat' => array('setting'), 'moneyvalueformat' => array('setting'), 'provinceLimit' => array('setting'), 'recaptchaOptions' => array('setting'), 'recaptchaPublicKey' => array('setting'), 'recaptchaPrivateKey' => array('setting'), 'replyTo' => array('setting'), 'secondDegRelPermissions' => array('setting'), 'smartGroupCacheTimeout' => array('setting'), 'timeInputFormat' => array('setting'), 'userFrameworkLogging' => array('setting'), 'userFrameworkUsersTableName' => array('setting'), 'verpSeparator' => array('setting'), 'wkhtmltopdfPath' => array('setting'), 'wpBasePage' => array('setting'), 'wpLoadPhp' => array('setting'), // "setting-path" properties are settings with special filtering // to return normalized file paths. // Option: `mkdir` - auto-create dir // Option: `restrict` - auto-restrict remote access 'customFileUploadDir' => array('setting-path', NULL, array('mkdir', 'restrict')), 'customPHPPathDir' => array('setting-path'), 'customTemplateDir' => array('setting-path'), 'extensionsDir' => array('setting-path', NULL, array('mkdir')), 'imageUploadDir' => array('setting-path', NULL, array('mkdir')), 'uploadDir' => array('setting-path', NULL, array('mkdir', 'restrict')), // "setting-url" properties are settings with special filtering // to return normalized URLs. // Option: `noslash` - don't append trailing slash // Option: `rel` - convert to relative URL (if possible) 'customCSSURL' => array('setting-url', NULL, array('noslash')), 'extensionsURL' => array('setting-url'), 'imageUploadURL' => array('setting-url'), 'resourceBase' => array('setting-url', 'userFrameworkResourceURL', array('rel')), 'userFrameworkResourceURL' => array('setting-url'), // "callback" properties are generated on-demand by calling a function. // @todo remove geocodeMethod. As of Feb 2018, $config->geocodeMethod works but gives a deprecation warning. 'geocodeMethod' => array('callback', 'CRM_Utils_Geocode', 'getProviderClass'), 'defaultCurrencySymbol' => array('callback', 'CRM_Core_BAO_Country', 'getDefaultCurrencySymbol'), ); } /** * Get value. * * @param string $k * * @return mixed * @throws \CRM_Core_Exception */ public function __get($k) { if (!isset($this->map[$k])) { throw new \CRM_Core_Exception("Cannot read unrecognized property CRM_Core_Config::\${$k}."); } if (isset($this->cache[$k])) { return $this->cache[$k]; } $type = $this->map[$k][0]; $name = isset($this->map[$k][1]) ? $this->map[$k][1] : $k; switch ($type) { case 'setting': return $this->getSettings()->get($name); case 'setting-path': // Array(0 => $type, 1 => $setting, 2 => $actions). $value = $this->getSettings()->get($name); $value = Civi::paths()->getPath($value); if ($value) { $value = CRM_Utils_File::addTrailingSlash($value); if (isset($this->map[$k][2]) && in_array('mkdir', $this->map[$k][2])) { if (!is_dir($value) && !CRM_Utils_File::createDir($value, FALSE)) { CRM_Core_Session::setStatus(ts('Failed to make directory (%1) at "%2". Please update the settings or file permissions.', array( 1 => $k, 2 => $value, ))); } } if (isset($this->map[$k][2]) && in_array('restrict', $this->map[$k][2])) { CRM_Utils_File::restrictAccess($value); } } $this->cache[$k] = $value; return $value; case 'setting-url': $options = !empty($this->map[$k][2]) ? $this->map[$k][2] : array(); $value = $this->getSettings()->get($name); if ($value && !(in_array('noslash', $options))) { $value = CRM_Utils_File::addTrailingSlash($value, '/'); } $this->cache[$k] = Civi::paths()->getUrl($value, in_array('rel', $options) ? 'relative' : 'absolute'); return $this->cache[$k]; case 'runtime': return \Civi\Core\Container::getBootService('runtime')->{$name}; case 'boot-svc': $this->cache[$k] = \Civi\Core\Container::getBootService($name); return $this->cache[$k]; case 'local': $this->initLocals(); return $this->locals[$name]; case 'user-system': $userSystem = \Civi\Core\Container::getBootService('userSystem'); $this->cache[$k] = call_user_func(array($userSystem, $name)); return $this->cache[$k]; case 'service': return \Civi::service($name); case 'callback': // Array(0 => $type, 1 => $obj, 2 => $getter, 3 => $setter, 4 => $unsetter). if (!isset($this->map[$k][1], $this->map[$k][2])) { throw new \CRM_Core_Exception("Cannot find getter for property CRM_Core_Config::\${$k}"); } return \Civi\Core\Resolver::singleton()->call(array($this->map[$k][1], $this->map[$k][2]), array($k)); default: throw new \CRM_Core_Exception("Cannot read property CRM_Core_Config::\${$k} ($type)"); } } /** * Set value. * * @param string $k * @param mixed $v * * @throws \CRM_Core_Exception */ public function __set($k, $v) { if (!isset($this->map[$k])) { throw new \CRM_Core_Exception("Cannot set unrecognized property CRM_Core_Config::\${$k}"); } unset($this->cache[$k]); $type = $this->map[$k][0]; // If foreign name is set, use that name (except with callback types because // their second parameter is the object, not the foreign name). $name = isset($this->map[$k][1]) && $type != 'callback' ? $this->map[$k][1] : $k; switch ($type) { case 'setting': case 'setting-path': case 'setting-url': case 'user-system': case 'runtime': case 'callback': case 'boot-svc': // In the past, changes to $config were not persisted automatically. $this->cache[$name] = $v; return; case 'local': $this->initLocals(); $this->locals[$name] = $v; return; default: throw new \CRM_Core_Exception("Cannot set property CRM_Core_Config::\${$k} ($type)"); } } /** * Is value set. * * @param string $k * * @return bool */ public function __isset($k) { return isset($this->map[$k]); } /** * Unset value. * * @param string $k * * @throws \CRM_Core_Exception */ public function __unset($k) { if (!isset($this->map[$k])) { throw new \CRM_Core_Exception("Cannot unset unrecognized property CRM_Core_Config::\${$k}"); } unset($this->cache[$k]); $type = $this->map[$k][0]; $name = isset($this->map[$k][1]) ? $this->map[$k][1] : $k; switch ($type) { case 'setting': case 'setting-path': case 'setting-url': $this->getSettings()->revert($k); return; case 'local': $this->initLocals(); $this->locals[$name] = NULL; return; case 'callback': // Array(0 => $type, 1 => $obj, 2 => $getter, 3 => $setter, 4 => $unsetter). if (!isset($this->map[$k][1], $this->map[$k][4])) { throw new \CRM_Core_Exception("Cannot find unsetter for property CRM_Core_Config::\${$k}"); } \Civi\Core\Resolver::singleton()->call(array($this->map[$k][1], $this->map[$k][4]), array($k)); return; default: throw new \CRM_Core_Exception("Cannot unset property CRM_Core_Config::\${$k} ($type)"); } } /** * @return \Civi\Core\SettingsBag */ protected function getSettings() { if ($this->settings === NULL) { $this->settings = Civi::settings(); } return $this->settings; } /** * Initialise local settings. */ private function initLocals() { if ($this->locals === NULL) { $this->locals = array( 'inCiviCRM' => FALSE, 'doNotResetCache' => 0, 'keyDisable' => FALSE, 'initialized' => FALSE, 'userFrameworkFrontend' => FALSE, 'userPermissionTemp' => NULL, ); } } }