currentStep = min(intval($_REQUEST['step'] ?? 1) ?: 1, count($this->steps) + 1); } public function listShouldInitializeTables() { return $this->initializeTables; } public function currentStep() { return $this->currentStep; } public function canAccessStep($step) { for ($i = 1; $i < $step; $i++) { $progressKey = $this->getProgressKey($i); if (!isset($_SESSION[$progressKey])) { $this->doLog("check step: $i, session doesn't have, session: " . json_encode($_SESSION)); return false; } } $this->doLog("check step: $step, can access, session: " . json_encode($_SESSION)); return true; } public function doneStep($step) { $progressKey = $this->getProgressKey($step); $this->doLog("doneStep: $step, $progressKey = 1"); $_SESSION[$progressKey] = 1; } private function getProgressKey($step) { return $this->progressKeyPrefix . $step; } public function getLogFile() { return sprintf('%s/nexus-install-%s.log', sys_get_temp_dir(), date('Ymd')); } public function getInsallDirectory() { return ROOT_PATH . 'public/install'; } public function doLog($log) { $log = sprintf('[%s] [%s] %s%s', date('Y-m-d H:i:s'), $this->currentStep, $log, PHP_EOL); file_put_contents($this->getLogFile(), $log, FILE_APPEND); } public function listAllTableCreate($sqlFile = '') { if (empty($sqlFile)) { $sqlFile = ROOT_PATH . '_db/dbstructure_v1.6.sql'; } $pattern = '/CREATE TABLE `(.*)`.*;/isU'; $string = file_get_contents($sqlFile); if ($string === false) { throw new \RuntimeException("sql file: $sqlFile can not read, make sure it exits and can be read."); } $count = preg_match_all($pattern, $string, $matches, PREG_SET_ORDER); if ($count == 0) { return []; } return array_column($matches, 0, 1); } public function listAllTableCreateFromMigrations() { $tables = []; foreach (glob(ROOT_PATH . "database/migrations/*.php") as $path) { $filename = basename($path); $count = preg_match('/create_(.*)_table.php/', $filename, $matches); if ($count) { $tables[$matches[1]] = $filename; } } return $tables; } public function listExistsTable() { $sql = 'show tables'; $res = sql_query($sql); $data = []; while ($row = mysql_fetch_row($res)) { $data[] = $row[0]; } return $data; } public function listRequirementTableRows() { $gdInfo = function_exists('gd_info') ? gd_info() : []; $extensions = ['ctype', 'fileinfo', 'json', 'mbstring', 'openssl', 'pdo_mysql', 'tokenizer', 'xml', 'mysqli', 'gd']; $tableRows = []; $tableRows[] = [ 'label' => 'PHP version', 'required' => '>= ' . $this->minimumPhpVersion, 'current' => PHP_VERSION, 'result' => $this->yesOrNo(version_compare(PHP_VERSION, $this->minimumPhpVersion, '>=')), ]; foreach ($extensions as $extension) { $tableRows[] = [ 'label' => "PHP extension $extension", 'required' => 'enabled', 'current' => extension_loaded($extension), 'result' => $this->yesOrNo(extension_loaded($extension)), ]; } $tableRows[] = [ 'label' => 'PHP extension gd JPEG Support', 'required' => 'true', 'current' => $gdInfo['JPEG Support'] ?? '', 'result' => $this->yesOrNo($gdInfo['JPEG Support'] ?? ''), ]; $tableRows[] = [ 'label' => 'PHP extension gd PNG Support', 'required' => 'true', 'current' => $gdInfo['PNG Support'] ?? '', 'result' => $this->yesOrNo($gdInfo['PNG Support'] ?? ''), ]; $tableRows[] = [ 'label' => 'PHP extension gd GIF Read Support', 'required' => 'true', 'current' => $gdInfo['GIF Read Support'] ?? '', 'result' => $this->yesOrNo($gdInfo['GIF Read Support'] ?? ''), ]; $tableRows[] = [ 'label' => 'PHP extension redis', 'required' => 'optional', 'current' => extension_loaded('redis'), 'result' => $this->yesOrNo(extension_loaded('redis')), ]; $fails = array_filter($tableRows, function ($value) {return in_array($value['required'], ['true', 'enabled']) && $value['result'] == 'NO';}); $pass = empty($fails); return [ 'table_rows' => $tableRows, 'pass' => $pass, ]; } public function listSettingTableRows() { $defaultSettingsFile = __DIR__ . '/settings.default.php'; $originalConfigFile = ROOT_PATH . 'config/allconfig.php'; if (!file_exists($defaultSettingsFile)) { throw new \RuntimeException("default setting file: $defaultSettingsFile not exists."); } if (!file_exists($originalConfigFile)) { throw new \RuntimeException("original setting file: $originalConfigFile not exists."); } $tableRows = [ [ 'label' => basename($defaultSettingsFile), 'required' => 'exists && readable', 'current' => $defaultSettingsFile, 'result' => $this->yesOrNo(file_exists($defaultSettingsFile) && is_readable($defaultSettingsFile)), ], [ 'label' => basename($originalConfigFile), 'required' => 'exists && readable', 'current' => $originalConfigFile, 'result' => $this->yesOrNo(file_exists($originalConfigFile) && is_readable($originalConfigFile)), ], ]; $requireDirs = [ 'main' => ['bitbucket', ], 'attachment' => ['savedirectory', ], ]; $symbolicLinks = []; require $originalConfigFile; $settings = require $defaultSettingsFile; $settingsFromDb = []; if (get_row_count('settings') > 0) { $settingsFromDb = get_setting(); } $this->doLog("settings form db: " . json_encode($settingsFromDb)); foreach ($settings as $prefix => &$group) { $prefixUpperCase = strtoupper($prefix); $oldGroupValues = $$prefixUpperCase ?? null; foreach ($group as $key => &$value) { //merge original config or db config to default setting, exclude code part if ($prefix != 'code') { if (isset($settingsFromDb[$prefix][$key])) { $this->doLog(sprintf( "$prefix.$key, db exists, change from: %s => %s", is_scalar($value) ? $value : json_encode($value), is_scalar($settingsFromDb[$prefix][$key]) ? $settingsFromDb[$prefix][$key] : json_encode($settingsFromDb[$prefix][$key])) ); $value = $settingsFromDb[$prefix][$key]; } elseif (isset($oldGroupValues) && isset($oldGroupValues[$key])) { $this->doLog(sprintf( "$prefix.$key, original config file exists, change from: %s => %s", is_scalar($value) ? $value : json_encode($value), is_scalar($oldGroupValues[$key]) ? $oldGroupValues[$key] : json_encode($oldGroupValues[$key])) ); $value = $oldGroupValues[$key]; } } if ($prefix == 'basic' && Str::startsWith($value, 'localhost')) { $value = ''; } if (isset($requireDirs[$prefix]) && in_array($key, $requireDirs[$prefix])) { $dir = getFullDirectory($value); $tableRows[] = [ 'label' => "{$prefix}.{$key}", 'required' => 'exists && readable', 'current' => $dir, 'result' => $this->yesOrNo(is_dir($dir) && is_readable($dir)), ]; $symbolicLinks[] = $dir; } } } $fails = array_filter($tableRows, function ($value) {return $value['required'] == 'true' && $value['result'] == 'NO';}); $pass = empty($fails); return [ 'table_rows' => $tableRows, 'symbolic_links' => $symbolicLinks, 'settings' => $settings, 'pass' => $pass, ]; } public function nextStep() { $this->doneStep($this->currentStep); $this->gotoStep($this->currentStep + 1); } public function gotoStep($step) { nexus_redirect(getBaseUrl() . "?step=$step"); } public function maxStep() { return count($this->steps); } public function yesOrNo($condition) { if ($condition) { return 'YES'; } return 'NO'; } public function renderTable($header, $data) { $table = '