currentStep = min(intval($_REQUEST['step'] ?? 1) ?: 1, count($this->steps) + 1); } public function currentStep() { return $this->currentStep; } 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 listExistsTable() { dbconn(false, false); $sql = 'show tables'; $res = sql_query($sql); $data = []; while ($row = mysql_fetch_row($res)) { $data[] = $row[0]; } return $data; } public function listShouldAlterTableTableRows() { $tables = $this->listExistsTable(); $data = []; foreach ($tables as $table) { $sql = "desc $table"; $res = sql_query($sql); while ($row = mysql_fetch_assoc($res)) { if ($row['Type'] == 'datetime' && $row['Default'] == '0000-00-00 00:00:00') { $data[$table][] = $row['Field']; $data[] = [ 'label' => "$table." . $row['Field'], 'required' => 'default null', 'current' => '0000-00-00 00:00:00', 'result' => 'NO', ]; } } } return $data; } public function listRequirementTableRows() { $gdInfo = gd_info(); $tableRows = [ [ 'label' => 'PHP version', 'required' => '>= ' . $this->minimumPhpVersion, 'current' => PHP_VERSION, 'result' => $this->yesOrNo(version_compare(PHP_VERSION, $this->minimumPhpVersion, '>=')), ], [ 'label' => 'PHP extension redis', 'required' => 'optional', 'current' => extension_loaded('redis'), 'result' => $this->yesOrNo(extension_loaded('redis')), ], [ 'label' => 'PHP extension mysqli', 'required' => 'enabled', 'current' => extension_loaded('mysqli'), 'result' => $this->yesOrNo(extension_loaded('mysqli')), ], [ 'label' => 'PHP extension mbstring', 'required' => 'enabled', 'current' => extension_loaded('mbstring'), 'result' => $this->yesOrNo(extension_loaded('mbstring')), ], [ 'label' => 'PHP extension gd', 'required' => 'enabled', 'current' => extension_loaded('gd'), 'result' => $this->yesOrNo(extension_loaded('gd')), ], [ 'label' => 'PHP extension gd JPEG Support', 'required' => 'true', 'current' => $gdInfo['JPEG Support'], 'result' => $this->yesOrNo($gdInfo['JPEG Support']), ], [ 'label' => 'PHP extension gd PNG Support', 'required' => 'true', 'current' => $gdInfo['PNG Support'], 'result' => $this->yesOrNo($gdInfo['PNG Support']), ], [ 'label' => 'PHP extension gd GIF Read Support', 'required' => 'true', 'current' => $gdInfo['GIF Read Support'], 'result' => $this->yesOrNo($gdInfo['GIF Read Support']), ], ]; $fails = array_filter($tableRows, function ($value) {return $value['required'] == 'true' && $value['result'] == 'NO';}); $pass = empty($fails); return [ 'table_rows' => $tableRows, 'pass' => $pass, ]; } public function listSettingTableRows() { $defaultSettingsFile = ROOT_PATH . '_doc/install/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', 'torrent_dir'], 'attachment' => ['savedirectory', ], ]; $symbolicLinks = []; require $originalConfigFile; $settings = require $defaultSettingsFile; foreach ($settings as $prefix => &$group) { $prefixUpperCase = strtoupper($prefix); $oldGroupValues = $$prefixUpperCase ?? null; foreach ($group as $key => &$value) { //merge original config to default setting if (isset($oldGroupValues) && isset($oldGroupValues[$key])) { $value = $oldGroupValues[$key];; } 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->gotoStep($this->currentStep + 1); } public function gotoStep($step) { header('Location: ' . getBaseUrl() . "?step=$step"); exit(0); } public function maxStep() { return count($this->steps); } public function yesOrNo($condition) { if ($condition) { return 'YES'; } return 'NO'; } public function renderTable($header, $data) { $table = '