| Code Coverage | ||||||||||
| Lines | Functions and Methods | Classes and Traits | ||||||||
| Total |  | 0.00% | 0 / 237 |  | 0.00% | 0 / 17 | CRAP |  | 0.00% | 0 / 1 | 
| tao_install_Installator |  | 0.00% | 0 / 237 |  | 0.00% | 0 / 17 | 3306 |  | 0.00% | 0 / 1 | 
| __construct |  | 0.00% | 0 / 9 |  | 0.00% | 0 / 1 | 12 | |||
| install |  | 0.00% | 0 / 167 |  | 0.00% | 0 / 1 | 702 | |||
| getServiceManager |  | 0.00% | 0 / 1 |  | 0.00% | 0 / 1 | 2 | |||
| retryInstallation |  | 0.00% | 0 / 15 |  | 0.00% | 0 / 1 | 20 | |||
| isWindows |  | 0.00% | 0 / 1 |  | 0.00% | 0 / 1 | 2 | |||
| generateSessionName |  | 0.00% | 0 / 1 |  | 0.00% | 0 / 1 | 2 | |||
| checkInstallData |  | 0.00% | 0 / 10 |  | 0.00% | 0 / 1 | 20 | |||
| escapeCheck |  | 0.00% | 0 / 4 |  | 0.00% | 0 / 1 | 2 | |||
| getEscapedChecks |  | 0.00% | 0 / 1 |  | 0.00% | 0 / 1 | 2 | |||
| setEscapedChecks |  | 0.00% | 0 / 1 |  | 0.00% | 0 / 1 | 2 | |||
| isEscapedCheck |  | 0.00% | 0 / 1 |  | 0.00% | 0 / 1 | 2 | |||
| log |  | 0.00% | 0 / 17 |  | 0.00% | 0 / 1 | 56 | |||
| getLog |  | 0.00% | 0 / 1 |  | 0.00% | 0 / 1 | 2 | |||
| getGenerisConfig |  | 0.00% | 0 / 1 |  | 0.00% | 0 / 1 | 2 | |||
| getConfigPath |  | 0.00% | 0 / 3 |  | 0.00% | 0 / 1 | 6 | |||
| setInstallationFinished |  | 0.00% | 0 / 3 |  | 0.00% | 0 / 1 | 2 | |||
| recreateDependencyInjectionContainerCache |  | 0.00% | 0 / 1 |  | 0.00% | 0 / 1 | 2 | |||
| 1 | <?php | 
| 2 | |
| 3 | /** | 
| 4 | * This program is free software; you can redistribute it and/or | 
| 5 | * modify it under the terms of the GNU General Public License | 
| 6 | * as published by the Free Software Foundation; under version 2 | 
| 7 | * of the License (non-upgradable). | 
| 8 | * | 
| 9 | * This program is distributed in the hope that it will be useful, | 
| 10 | * but WITHOUT ANY WARRANTY; without even the implied warranty of | 
| 11 | * MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the | 
| 12 | * GNU General Public License for more details. | 
| 13 | * | 
| 14 | * You should have received a copy of the GNU General Public License | 
| 15 | * along with this program; if not, write to the Free Software | 
| 16 | * Foundation, Inc., 51 Franklin Street, Fifth Floor, Boston, MA 02110-1301, USA. | 
| 17 | * | 
| 18 | * Copyright (c) 2002-2008 (original work) Public Research Centre Henri Tudor & University of Luxembourg | 
| 19 | * (under the project TAO & TAO2); | 
| 20 | * 2008-2010 (update and modification) Deutsche Institut für Internationale Pädagogische Forschung | 
| 21 | * (under the project TAO-TRANSFER); | 
| 22 | * 2009-2012 (update and modification) Public Research Centre Henri Tudor | 
| 23 | * (under the project TAO-SUSTAIN & TAO-DEV); | 
| 24 | * 2013-2017 (update and modification) Open Assessment Technologies SA (under the project TAO-PRODUCT); | 
| 25 | */ | 
| 26 | |
| 27 | use oat\generis\persistence\DriverConfigurationFeeder; | 
| 28 | use oat\oatbox\cache\SetupFileCache; | 
| 29 | use oat\tao\helpers\InstallHelper; | 
| 30 | use oat\oatbox\install\Installer; | 
| 31 | use oat\oatbox\service\ServiceManager; | 
| 32 | use oat\tao\model\OperatedByService; | 
| 33 | use oat\generis\persistence\sql\DbCreator; | 
| 34 | use oat\generis\persistence\sql\SetupDb; | 
| 35 | use oat\generis\persistence\PersistenceManager; | 
| 36 | use oat\generis\model\data\Ontology; | 
| 37 | use oat\tao\model\TaoOntology; | 
| 38 | use oat\generis\model\GenerisRdf; | 
| 39 | use oat\tao\model\user\TaoRoles; | 
| 40 | use oat\tao\model\service\ApplicationService; | 
| 41 | use oat\oatbox\service\ServiceNotFoundException; | 
| 42 | |
| 43 | /** | 
| 44 | * | 
| 45 | * | 
| 46 | * Installation main class | 
| 47 | * | 
| 48 | * @access public | 
| 49 | * @author Jérôme Bogaerts, <jerome@taotesting.com> | 
| 50 | * @package tao | 
| 51 | */ | 
| 52 | |
| 53 | class tao_install_Installator | 
| 54 | { | 
| 55 | // Adding container and logger. | 
| 56 | use \oat\oatbox\log\ContainerLoggerTrait; | 
| 57 | |
| 58 | /** | 
| 59 | * Installator related dependencies will be reached under this offset. | 
| 60 | */ | 
| 61 | public const CONTAINER_INDEX = 'taoInstallInstallator'; | 
| 62 | |
| 63 | protected $options = []; | 
| 64 | |
| 65 | private $log = []; | 
| 66 | |
| 67 | private $escapedChecks = []; | 
| 68 | |
| 69 | private $oatBoxInstall = null; | 
| 70 | |
| 71 | public function __construct($options) | 
| 72 | { | 
| 73 | // Using the container if it's necessary with automatic dependency returning. | 
| 74 | $options = $this->initContainer($options, static::CONTAINER_INDEX); | 
| 75 | |
| 76 | if (!isset($options['root_path'])) { | 
| 77 | throw new tao_install_utils_Exception("root_path option must be defined to perform installation."); | 
| 78 | } | 
| 79 | if (!isset($options['install_path'])) { | 
| 80 | throw new tao_install_utils_Exception("install_path option must be defined to perform installation."); | 
| 81 | } | 
| 82 | |
| 83 | $this->options = $options; | 
| 84 | |
| 85 | $this->options['root_path'] = rtrim($this->options['root_path'], '/\\') . DIRECTORY_SEPARATOR; | 
| 86 | $this->options['install_path'] = rtrim($this->options['install_path'], '/\\') . DIRECTORY_SEPARATOR; | 
| 87 | |
| 88 | $this->oatBoxInstall = new Installer(); | 
| 89 | } | 
| 90 | |
| 91 | /** | 
| 92 | * Run the TAO install from the given data | 
| 93 | * @throws tao_install_utils_Exception | 
| 94 | * @param $installData array data coming from the install form | 
| 95 | * @param $callback callable|null post install callback | 
| 96 | */ | 
| 97 | public function install(array $installData, callable $callback = null) | 
| 98 | { | 
| 99 | try { | 
| 100 | /** | 
| 101 | * It's a quick hack for solving reinstall issue. | 
| 102 | * Should be a better option. | 
| 103 | */ | 
| 104 | @unlink($this->options['root_path'] . 'config/generis.conf.php'); | 
| 105 | |
| 106 | /* | 
| 107 | * 0 - Check input parameters. | 
| 108 | */ | 
| 109 | $this->log('i', "Checking install data"); | 
| 110 | self::checkInstallData($installData); | 
| 111 | |
| 112 | $this->log('i', "Starting TAO install"); | 
| 113 | |
| 114 | // Sanitize $installData if needed. | 
| 115 | if (!preg_match("/\/$/", $installData['module_url'])) { | 
| 116 | $installData['module_url'] .= '/'; | 
| 117 | } | 
| 118 | |
| 119 | // Define the ROOT_URL constant if not defined (can be used in manifest files) | 
| 120 | if (!defined('ROOT_URL')) { | 
| 121 | define('ROOT_URL', $installData['module_url']); | 
| 122 | } | 
| 123 | |
| 124 | if (isset($installData['extensions'])) { | 
| 125 | $extensionIDs = is_array($installData['extensions']) | 
| 126 | ? $installData['extensions'] | 
| 127 | : explode(',', $installData['extensions']); | 
| 128 | } else { | 
| 129 | $extensionIDs = ['taoCe']; | 
| 130 | } | 
| 131 | |
| 132 | $this->log('d', 'Extensions to be installed: ' . var_export($extensionIDs, true)); | 
| 133 | |
| 134 | $installData['file_path'] = rtrim($installData['file_path'], DIRECTORY_SEPARATOR) . DIRECTORY_SEPARATOR; | 
| 135 | |
| 136 | /* | 
| 137 | * 1 - Check configuration with checks described in the manifest. | 
| 138 | */ | 
| 139 | $configChecker = tao_install_utils_ChecksHelper::getConfigChecker($extensionIDs); | 
| 140 | |
| 141 | // Silence checks to have to be escaped. | 
| 142 | foreach ($configChecker->getComponents() as $c) { | 
| 143 | if (method_exists($c, 'getName') && in_array($c->getName(), $this->getEscapedChecks())) { | 
| 144 | $configChecker->silent($c); | 
| 145 | } | 
| 146 | } | 
| 147 | |
| 148 | $reports = $configChecker->check(); | 
| 149 | foreach ($reports as $r) { | 
| 150 | $msg = $r->getMessage(); | 
| 151 | $component = $r->getComponent(); | 
| 152 | $this->log('i', $msg); | 
| 153 | |
| 154 | if ($r->getStatus() !== common_configuration_Report::VALID && !$component->isOptional()) { | 
| 155 | throw new tao_install_utils_Exception($msg); | 
| 156 | } | 
| 157 | } | 
| 158 | |
| 159 | /* | 
| 160 | * X - Setup Oatbox | 
| 161 | */ | 
| 162 | |
| 163 | $this->log('d', 'Removing old config'); | 
| 164 | $consistentOptions = array_merge($installData, $this->options); | 
| 165 | $consistentOptions['config_path'] = $this->getConfigPath(); | 
| 166 | $this->oatBoxInstall->setOptions($consistentOptions); | 
| 167 | $this->oatBoxInstall->install(); | 
| 168 | $this->log('d', 'Oatbox was installed!'); | 
| 169 | |
| 170 | ServiceManager::setServiceManager($this->getServiceManager()); | 
| 171 | |
| 172 | /* | 
| 173 | * 2 - Setup RDS persistence | 
| 174 | */ | 
| 175 | if (!$this->getServiceManager()->has(DriverConfigurationFeeder::SERVICE_ID)) { | 
| 176 | $this->getServiceManager()->register( | 
| 177 | DriverConfigurationFeeder::SERVICE_ID, | 
| 178 | new DriverConfigurationFeeder( | 
| 179 | [ | 
| 180 | DriverConfigurationFeeder::OPTION_DRIVER_OPTIONS => [] | 
| 181 | ] | 
| 182 | ) | 
| 183 | ); | 
| 184 | } | 
| 185 | |
| 186 | if ($this->getServiceManager()->has(PersistenceManager::SERVICE_ID)) { | 
| 187 | $persistenceManager = $this->getServiceManager()->get(PersistenceManager::SERVICE_ID); | 
| 188 | } else { | 
| 189 | $this->log('i', "Spawning new PersistenceManager"); | 
| 190 | $persistenceManager = new PersistenceManager(); | 
| 191 | } | 
| 192 | if (!$persistenceManager->hasPersistence('default')) { | 
| 193 | $this->log('i', "Register default Persistence"); | 
| 194 | $dbalConfigCreator = new tao_install_utils_DbalConfigCreator(); | 
| 195 | $persistenceManager->registerPersistence('default', $dbalConfigCreator->createDbalConfig($installData)); | 
| 196 | $this->getServiceManager()->register(PersistenceManager::SERVICE_ID, $persistenceManager); | 
| 197 | } | 
| 198 | |
| 199 | $dbCreator = new SetupDb(); | 
| 200 | $dbCreator->setLogger($this->logger); | 
| 201 | $dbCreator->setupDatabase($persistenceManager->getPersistenceById('default')); | 
| 202 | |
| 203 | /* | 
| 204 | * 4 - Create the generis config files | 
| 205 | */ | 
| 206 | |
| 207 | $this->log('d', 'Writing generis config'); | 
| 208 | $generisConfigWriter = new tao_install_utils_ConfigWriter( | 
| 209 | $this->options['root_path'] . 'generis/config/sample/generis.conf.php', | 
| 210 | $this->getGenerisConfig() | 
| 211 | ); | 
| 212 | |
| 213 | $session_name = $installData['session_name'] ?? self::generateSessionName(); | 
| 214 | $generisConfigWriter->createConfig(); | 
| 215 | $constants = [ | 
| 216 | 'LOCAL_NAMESPACE' => $installData['module_namespace'], | 
| 217 | 'GENERIS_INSTANCE_NAME' => $installData['instance_name'], | 
| 218 | 'GENERIS_SESSION_NAME' => $session_name, | 
| 219 | 'ROOT_PATH' => $this->options['root_path'], | 
| 220 | 'FILES_PATH' => $installData['file_path'], | 
| 221 | 'ROOT_URL' => $installData['module_url'], | 
| 222 | 'DEFAULT_LANG' => $installData['module_lang'], | 
| 223 | 'DEBUG_MODE' => ($installData['module_mode'] == 'debug') ? true : false, | 
| 224 | 'TIME_ZONE' => $installData['timezone'] | 
| 225 | ]; | 
| 226 | |
| 227 | $constants['DEFAULT_ANONYMOUS_INTERFACE_LANG'] = $installData['anonymous_lang'] | 
| 228 | ?? $installData['module_lang']; | 
| 229 | |
| 230 | $generisConfigWriter->writeConstants($constants); | 
| 231 | $this->log( | 
| 232 | 'd', | 
| 233 | 'The following constants were written in generis config:' . PHP_EOL . var_export($constants, true) | 
| 234 | ); | 
| 235 | |
| 236 | /* | 
| 237 | * 4b - Prepare the file/cache folder (FILES_PATH) not yet defined) | 
| 238 | * @todo solve this more elegantly | 
| 239 | */ | 
| 240 | $file_path = $installData['file_path']; | 
| 241 | if (is_dir($file_path)) { | 
| 242 | $this->log('i', 'Data from previous install found and will be removed'); | 
| 243 | if (!helpers_File::emptyDirectory($file_path, true)) { | 
| 244 | throw new common_exception_Error('Unable to empty ' . $file_path . ' folder.'); | 
| 245 | } | 
| 246 | } else { | 
| 247 | if (mkdir($file_path, 0700, true)) { | 
| 248 | $this->log('d', $file_path . ' directory was created!'); | 
| 249 | } else { | 
| 250 | throw new Exception($file_path . ' directory creation was failed!'); | 
| 251 | } | 
| 252 | } | 
| 253 | |
| 254 | $setupFileCache = $this->getServiceManager()->get(SetupFileCache::class); | 
| 255 | $cachePath = $file_path . 'generis' . DIRECTORY_SEPARATOR . 'cache'; | 
| 256 | $setupFileCache->createDirectory($cachePath); | 
| 257 | $this->log('d', $cachePath . ' directory was created!'); | 
| 258 | |
| 259 | foreach ((array)$installData['extra_persistences'] as $k => $persistence) { | 
| 260 | $persistenceManager->registerPersistence($k, $persistence); | 
| 261 | } | 
| 262 | |
| 263 | /* | 
| 264 | * 5 - Run the extensions bootstrap | 
| 265 | */ | 
| 266 | $this->log('d', 'Running the extensions bootstrap'); | 
| 267 | common_Config::load($this->getGenerisConfig()); | 
| 268 | |
| 269 | /* | 
| 270 | * 5b - Create cache persistence | 
| 271 | */ | 
| 272 | $this->log('d', 'Creating cache persistence..'); | 
| 273 | $setupFileCache->createPersistence(); | 
| 274 | |
| 275 | /* | 
| 276 | * 6 - Finish Generis Install | 
| 277 | */ | 
| 278 | |
| 279 | $this->log('d', 'Finishing generis install..'); | 
| 280 | $generis = common_ext_ExtensionsManager::singleton()->getExtensionById('generis'); | 
| 281 | |
| 282 | $generisInstaller = new common_ext_GenerisInstaller($generis, true); | 
| 283 | $generisInstaller->initContainer($this->getContainer()); | 
| 284 | $generisInstaller->install(); | 
| 285 | |
| 286 | /* | 
| 287 | * 7 - Add languages | 
| 288 | */ | 
| 289 | $this->log('d', 'Adding languages..'); | 
| 290 | $ontology = $this->getServiceManager()->get(Ontology::SERVICE_ID); | 
| 291 | $langModel = \tao_models_classes_LanguageService::singleton()->getLanguageDefinition(); | 
| 292 | $rdfModel = $ontology->getRdfInterface(); | 
| 293 | foreach ($langModel as $triple) { | 
| 294 | $rdfModel->add($triple); | 
| 295 | } | 
| 296 | |
| 297 | /* | 
| 298 | * 8 - Install the extensions | 
| 299 | */ | 
| 300 | InstallHelper::initContainer($this->container); | 
| 301 | $installed = InstallHelper::installRecursively($extensionIDs, $installData); | 
| 302 | $this->log('ext', $installed); | 
| 303 | |
| 304 | /* | 
| 305 | * 8b - Generates client side translation bundles (depends on extension install) | 
| 306 | */ | 
| 307 | $this->log('i', 'Generates client side translation bundles'); | 
| 308 | |
| 309 | tao_models_classes_LanguageService::singleton()->generateAll(); | 
| 310 | |
| 311 | /* | 
| 312 | * 9 - Insert Super User | 
| 313 | */ | 
| 314 | $this->log('i', 'Spawning SuperUser ' . $installData['user_login']); | 
| 315 | |
| 316 | $userClass = $ontology->getClass(TaoOntology::CLASS_URI_TAO_USER); | 
| 317 | $userid = $installData['module_namespace'] . TaoOntology::DEFAULT_USER_URI_SUFFIX; | 
| 318 | $userpwd = core_kernel_users_Service::getPasswordHash()->encrypt($installData['user_pass1']); | 
| 319 | $userLang = 'http://www.tao.lu/Ontologies/TAO.rdf#Lang' . $installData['module_lang']; | 
| 320 | |
| 321 | $superUser = $userClass->createInstance( | 
| 322 | 'Super User', | 
| 323 | 'super user created during the TAO installation', | 
| 324 | $userid | 
| 325 | ); | 
| 326 | $superUser->setPropertiesValues([ | 
| 327 | GenerisRdf::PROPERTY_USER_ROLES => [ | 
| 328 | TaoRoles::GLOBAL_MANAGER, | 
| 329 | TaoRoles::SYSTEM_ADMINISTRATOR | 
| 330 | ], | 
| 331 | TaoOntology::PROPERTY_USER_FIRST_TIME => GenerisRdf::GENERIS_TRUE, | 
| 332 | GenerisRdf::PROPERTY_USER_LOGIN => $installData['user_login'], | 
| 333 | GenerisRdf::PROPERTY_USER_PASSWORD => $userpwd, | 
| 334 | GenerisRdf::PROPERTY_USER_LASTNAME => $installData['user_lastname'], | 
| 335 | GenerisRdf::PROPERTY_USER_FIRSTNAME => $installData['user_firstname'], | 
| 336 | GenerisRdf::PROPERTY_USER_MAIL => $installData['user_email'], | 
| 337 | GenerisRdf::PROPERTY_USER_DEFLG => $userLang, | 
| 338 | GenerisRdf::PROPERTY_USER_UILG => $userLang, | 
| 339 | GenerisRdf::PROPERTY_USER_TIMEZONE => TIME_ZONE | 
| 340 | ]); | 
| 341 | |
| 342 | /* | 
| 343 | * 10 - Secure the install for production mode | 
| 344 | */ | 
| 345 | if ($installData['module_mode'] == 'production') { | 
| 346 | $extensions = common_ext_ExtensionsManager::singleton()->getInstalledExtensions(); | 
| 347 | $this->log('i', 'Securing tao for production'); | 
| 348 | |
| 349 | // 11.0 Protect TAO dist | 
| 350 | $shield = new tao_install_utils_Shield(array_keys($extensions)); | 
| 351 | $shield->disableRewritePattern(["!/test/", "!/doc/"]); | 
| 352 | $shield->denyAccessTo([ | 
| 353 | 'views/sass', | 
| 354 | 'views/js/test', | 
| 355 | 'views/build' | 
| 356 | ]); | 
| 357 | $shield->protectInstall(); | 
| 358 | } | 
| 359 | |
| 360 | /* | 
| 361 | * 11 - Create the version file | 
| 362 | */ | 
| 363 | $this->log('d', 'Creating TAO version file'); | 
| 364 | file_put_contents($installData['file_path'] . 'version', TAO_VERSION); | 
| 365 | |
| 366 | /* | 
| 367 | * 12 - Register Information about organization operating the system | 
| 368 | */ | 
| 369 | $this->log('t', 'Registering information about the organization operating the system'); | 
| 370 | $operatedByService = $this->getServiceManager()->get(OperatedByService::SERVICE_ID); | 
| 371 | |
| 372 | if (!empty($installData['operated_by_name'])) { | 
| 373 | $operatedByService->setName($installData['operated_by_name']); | 
| 374 | } | 
| 375 | |
| 376 | if (!empty($installData['operated_by_email'])) { | 
| 377 | $operatedByService->setEmail($installData['operated_by_email']); | 
| 378 | } | 
| 379 | |
| 380 | $this->getServiceManager()->register(OperatedByService::SERVICE_ID, $operatedByService); | 
| 381 | if ($callback) { | 
| 382 | $callback(); | 
| 383 | } | 
| 384 | |
| 385 | $this->recreateDependencyInjectionContainerCache(); | 
| 386 | $this->setInstallationFinished(); | 
| 387 | } catch (Exception $e) { | 
| 388 | if ($this->retryInstallation($e)) { | 
| 389 | return; | 
| 390 | } | 
| 391 | |
| 392 | // In any case, we transmit a single exception type (at the moment) | 
| 393 | // for a clearer API for client code. | 
| 394 | $this->log('e', 'Error Occurs : ' . $e->getMessage() . PHP_EOL . $e->getTraceAsString()); | 
| 395 | throw new tao_install_utils_Exception($e->getMessage(), 0, $e); | 
| 396 | } | 
| 397 | } | 
| 398 | |
| 399 | public function getServiceManager() | 
| 400 | { | 
| 401 | return $this->oatBoxInstall->setupServiceManager($this->getConfigPath()); | 
| 402 | } | 
| 403 | |
| 404 | private function retryInstallation($exception) | 
| 405 | { | 
| 406 | $returnValue = false; | 
| 407 | $err = $exception->getMessage(); | 
| 408 | |
| 409 | if (strpos($err, 'cannot construct the resource because the uri cannot be empty') === 0 && $this->isWindows()) { | 
| 410 | /* | 
| 411 | * a known issue | 
| 412 | * @see http://forge.taotesting.com/issues/3014 | 
| 413 | * this issue can only be fixed by an administrator | 
| 414 | * changing the thread_stack system variable in my.ini as following: | 
| 415 | * '256K' on 64bit windows | 
| 416 | * '192K' on 32bit windows | 
| 417 | */ | 
| 418 | |
| 419 | $this->log('e', 'Error Occurs : ' . $err . PHP_EOL . $exception->getTraceAsString()); | 
| 420 | throw new tao_install_utils_Exception( | 
| 421 | "Error in mysql system variable 'thread_stack':<br>It is required to change its value in " | 
| 422 | . "my.ini as following<br>'192K' on 32bit windows<br>'256K' on 64bit windows.<br><br>Note that " | 
| 423 | . "such configuration changes will only take effect after server restart.<br><br>", | 
| 424 | 0, | 
| 425 | $exception | 
| 426 | ); | 
| 427 | } | 
| 428 | |
| 429 | if (!$returnValue) { | 
| 430 | return false; | 
| 431 | } | 
| 432 | |
| 433 | // it is a known issue, go ahead to retry with the issue fixer | 
| 434 | $this->install($this->config); | 
| 435 | return true; | 
| 436 | } | 
| 437 | |
| 438 | private function isWindows() | 
| 439 | { | 
| 440 | return strtoupper(substr(PHP_OS, 0, 3)) == 'WIN'; | 
| 441 | } | 
| 442 | |
| 443 | /** | 
| 444 | * Generate an alphanum token to be used as a PHP session name. | 
| 445 | * | 
| 446 | * @access public | 
| 447 | * @author Jerome Bogaerts, <jerome.bogaerts@tudor.lu> | 
| 448 | * @return string | 
| 449 | */ | 
| 450 | public static function generateSessionName() | 
| 451 | { | 
| 452 | return 'tao_' . helpers_Random::generateString(8); | 
| 453 | } | 
| 454 | |
| 455 | /** | 
| 456 | * Check the install data information such as | 
| 457 | * - instance name | 
| 458 | * - database driver | 
| 459 | * - ... | 
| 460 | * | 
| 461 | * If a parameter of the $installData is not valid regarding the install | 
| 462 | * business rules, an MalformedInstall | 
| 463 | * | 
| 464 | * @param array $installData | 
| 465 | */ | 
| 466 | public static function checkInstallData(array $installData) | 
| 467 | { | 
| 468 | // instance name | 
| 469 | if (empty($installData['instance_name'])) { | 
| 470 | $msg = "Missing install parameter 'instance_name'."; | 
| 471 | throw new tao_install_utils_MalformedParameterException($msg); | 
| 472 | } elseif (!is_string($installData['instance_name'])) { | 
| 473 | $msg = "Malformed install parameter 'instance_name'. It must be a string."; | 
| 474 | throw new tao_install_utils_MalformedParameterException($msg); | 
| 475 | } elseif (1 === preg_match('/\s/u', $installData['instance_name'])) { | 
| 476 | $msg = "Malformed install parameter 'instance_name'. It cannot contain spacing characters " | 
| 477 | . "(tab, backspace)."; | 
| 478 | throw new tao_install_utils_MalformedParameterException($msg); | 
| 479 | } | 
| 480 | } | 
| 481 | |
| 482 | /** | 
| 483 | * Tell the Installator instance to not take into account | 
| 484 | * a Configuration Check with ID = $id. | 
| 485 | * | 
| 486 | * @param string $id The identifier of the check to escape. | 
| 487 | */ | 
| 488 | public function escapeCheck($id) | 
| 489 | { | 
| 490 | $checks = $this->getEscapedChecks(); | 
| 491 | array_push($checks, $id); | 
| 492 | $checks = array_unique($checks); | 
| 493 | $this->setEscapedChecks($checks); | 
| 494 | } | 
| 495 | |
| 496 | /** | 
| 497 | * Obtain an array of Configuration Check IDs to be escaped by | 
| 498 | * the Installator. | 
| 499 | * | 
| 500 | * @return array | 
| 501 | */ | 
| 502 | public function getEscapedChecks() | 
| 503 | { | 
| 504 | return $this->escapedChecks; | 
| 505 | } | 
| 506 | |
| 507 | /** | 
| 508 | * Set the array of Configuration Check IDs to be escaped by | 
| 509 | * the Installator. | 
| 510 | * | 
| 511 | * @param array $escapedChecks An array of strings. | 
| 512 | * @return void | 
| 513 | */ | 
| 514 | public function setEscapedChecks(array $escapedChecks) | 
| 515 | { | 
| 516 | $this->escapedChecks = $escapedChecks; | 
| 517 | } | 
| 518 | |
| 519 | /** | 
| 520 | * Informs you if a given Configuration Check ID corresponds | 
| 521 | * to a Check that has to be escaped. | 
| 522 | */ | 
| 523 | public function isEscapedCheck($id) | 
| 524 | { | 
| 525 | return in_array($id, $this->getEscapedChecks()); | 
| 526 | } | 
| 527 | |
| 528 | /** | 
| 529 | * Log message and add it to $this->log array; | 
| 530 | * @see common_Logger class | 
| 531 | * @param string $logLevel | 
| 532 | * <ul> | 
| 533 | * <li>'w' - warning</li> | 
| 534 | * <li>'t' - trace</li> | 
| 535 | * <li>'d' - debug</li> | 
| 536 | * <li>'i' - info</li> | 
| 537 | * <li>'e' - error</li> | 
| 538 | * <li>'f' - fatal</li> | 
| 539 | * <li>'ext' - installed extensions</li> | 
| 540 | * </ul> | 
| 541 | * @param string $message | 
| 542 | * @param array $tags | 
| 543 | */ | 
| 544 | public function log($logLevel, $message, $tags = []) | 
| 545 | { | 
| 546 | if (!is_array($tags)) { | 
| 547 | $tags = [$tags]; | 
| 548 | } | 
| 549 | if ($this->getLogger() instanceof \Psr\Log\LoggerInterface) { | 
| 550 | if ($logLevel === 'ext') { | 
| 551 | $this->logNotice('Installed extensions: ' . implode(', ', $message)); | 
| 552 | } else { | 
| 553 | $this->getLogger()->log( | 
| 554 | common_log_Logger2Psr::getPsrLevelFromCommon($logLevel), | 
| 555 | $message | 
| 556 | ); | 
| 557 | } | 
| 558 | } | 
| 559 | if (method_exists('common_Logger', $logLevel)) { | 
| 560 | call_user_func('common_Logger::' . $logLevel, $message, $tags); | 
| 561 | } | 
| 562 | if (is_array($message)) { | 
| 563 | $this->log[$logLevel] = (isset($this->log[$logLevel])) | 
| 564 | ? array_merge($this->log[$logLevel], $message) | 
| 565 | : $message; | 
| 566 | } else { | 
| 567 | $this->log[$logLevel][] = $message; | 
| 568 | } | 
| 569 | } | 
| 570 | |
| 571 | /** | 
| 572 | * Get array of log messages | 
| 573 | * @return array | 
| 574 | */ | 
| 575 | public function getLog() | 
| 576 | { | 
| 577 | return $this->log; | 
| 578 | } | 
| 579 | |
| 580 | /** | 
| 581 | * Get the config file platform e.q. generis.conf.php | 
| 582 | * | 
| 583 | * @return string | 
| 584 | */ | 
| 585 | protected function getGenerisConfig() | 
| 586 | { | 
| 587 | return $this->getConfigPath() . 'generis.conf.php'; | 
| 588 | } | 
| 589 | |
| 590 | /** | 
| 591 | * Get the config path for installation | 
| 592 | * If options have installation_config_path, it's taken otherwise it's root_path | 
| 593 | * | 
| 594 | * @return string | 
| 595 | */ | 
| 596 | protected function getConfigPath() | 
| 597 | { | 
| 598 | if (isset($this->options['installation_config_path'])) { | 
| 599 | return $this->options['installation_config_path']; | 
| 600 | } else { | 
| 601 | return $this->options['root_path'] . 'config' . DIRECTORY_SEPARATOR; | 
| 602 | } | 
| 603 | } | 
| 604 | |
| 605 | /** | 
| 606 | * Mark application as ready to be used (all extensions installed and post scripts executed) | 
| 607 | * @throws common_Exception | 
| 608 | */ | 
| 609 | private function setInstallationFinished() | 
| 610 | { | 
| 611 | $applicationService = $this->getServiceManager()->get(ApplicationService::SERVICE_ID); | 
| 612 | $applicationService->setOption(ApplicationService::OPTION_INSTALLATION_FINISHED, true); | 
| 613 | $this->getServiceManager()->register(ApplicationService::SERVICE_ID, $applicationService); | 
| 614 | } | 
| 615 | |
| 616 | private function recreateDependencyInjectionContainerCache(): void | 
| 617 | { | 
| 618 | ServiceManager::getServiceManager()->rebuildContainer(); | 
| 619 | } | 
| 620 | } |