<?php // +---------------------------------------------------------------------- // | Niucloud-admin 企业快速开发的saas管理平台 // +---------------------------------------------------------------------- // | 官方网址:https://www.niucloud-admin.com // +---------------------------------------------------------------------- // | niucloud团队 版权所有 开源版本可自由商用 // +---------------------------------------------------------------------- // | Author: Niucloud Team // +---------------------------------------------------------------------- namespace app\service\admin\upgrade; use app\dict\addon\AddonDict; use app\model\addon\Addon; use app\service\admin\install\InstallSystemService; use app\service\core\addon\CoreAddonCloudService; use app\service\core\addon\CoreAddonInstallService; use app\service\core\addon\CoreAddonService; use app\service\core\addon\CoreDependService; use app\service\core\addon\WapTrait; use app\service\core\menu\CoreMenuService; use app\service\core\niucloud\CoreModuleService; use app\service\core\schedule\CoreScheduleInstallService; use core\base\BaseAdminService; use core\exception\CommonException; use core\util\niucloud\BaseNiucloudClient; use think\facade\Cache; use think\facade\Db; /** * 框架及插件升级 * @package app\service\core\upgrade */ class UpgradeService extends BaseAdminService { use WapTrait; use ExecuteSqlTrait; protected $upgrade_dir; protected $root_path; protected $cache_key = 'upgrade'; protected $upgrade_task = null; protected $addon = ''; private $steps = [ 'requestUpgrade' => ['step' => 'requestUpgrade', 'title' => '请求升级'], 'downloadFile' => ['step' => 'downloadFile', 'title' => '下载更新文件'], 'backupCode' => ['step' => 'backupCode', 'title' => '备份源码'], 'backupSql' => ['step' => 'backupSql', 'title' => '备份数据库'], 'coverCode' => ['step' => 'coverCode', 'title' => '合并更新文件'], 'handleUniapp' => ['step' => 'handleUniapp', 'title' => '处理uniapp'], 'refreshMenu' => ['step' => 'refreshMenu', 'title' => '刷新菜单'], 'installSchedule' => ['step' => 'installSchedule', 'title' => '安装计划任务'], 'upgradeComplete' => ['step' => 'upgradeComplete', 'title' => '升级完成'] ]; public function __construct() { parent::__construct(); $this->root_path = dirname(root_path()) . DIRECTORY_SEPARATOR; $this->upgrade_dir = $this->root_path . 'upgrade' . DIRECTORY_SEPARATOR; $this->upgrade_task = Cache::get($this->cache_key); } /** * 升级前环境检测 * @param string $addon * @return void */ public function upgradePreCheck(string $addon = '') { $niucloud_dir = $this->root_path . 'niucloud' . DIRECTORY_SEPARATOR; $admin_dir = $this->root_path . 'admin' . DIRECTORY_SEPARATOR; $web_dir = $this->root_path . 'web' . DIRECTORY_SEPARATOR; $wap_dir = $this->root_path . 'uni-app' . DIRECTORY_SEPARATOR; try { if (!is_dir($admin_dir)) throw new CommonException('ADMIN_DIR_NOT_EXIST'); if (!is_dir($web_dir)) throw new CommonException('WEB_DIR_NOT_EXIST'); if (!is_dir($wap_dir)) throw new CommonException('UNIAPP_DIR_NOT_EXIST'); } catch (\Exception $e) { if (strpos($e->getMessage(), 'open basedir') !== false) { throw new CommonException('OPEN_BASEDIR_ERROR'); } throw new CommonException($e->getMessage()); } $data = [ // 目录检测 'dir' => [ // 要求可读权限 'is_readable' => [], // 要求可写权限 'is_write' => [] ] ]; $data['dir']['is_readable'][] = ['dir' => str_replace(project_path(), '', $niucloud_dir), 'status' => is_readable($niucloud_dir)]; $data['dir']['is_readable'][] = ['dir' => str_replace(project_path(), '', $admin_dir), 'status' => is_readable($admin_dir)]; $data['dir']['is_readable'][] = ['dir' => str_replace(project_path(), '', $web_dir), 'status' => is_readable($web_dir)]; $data['dir']['is_readable'][] = ['dir' => str_replace(project_path(), '', $wap_dir), 'status' => is_readable($wap_dir)]; $data['dir']['is_write'][] = ['dir' => str_replace(project_path(), '', $niucloud_dir), 'status' => is_write($niucloud_dir) ]; $data['dir']['is_write'][] = ['dir' => str_replace(project_path(), '', $admin_dir), 'status' => is_write($admin_dir) ]; $data['dir']['is_write'][] = ['dir' => str_replace(project_path(), '', $web_dir), 'status' => is_write($web_dir) ]; $data['dir']['is_write'][] = ['dir' => str_replace(project_path(), '', $wap_dir), 'status' => is_write($wap_dir) ]; $check_res = array_merge( array_column($data['dir']['is_readable'], 'status'), array_column($data['dir']['is_write'], 'status') ); // 是否通过校验 $data['is_pass'] = !in_array(false, $check_res); return $data; } /** * 升级 * @param $addon * @return array */ public function upgrade(string $addon = '') { if ($this->upgrade_task) throw new CommonException('UPGRADE_TASK_EXIST'); $upgrade = [ 'product_key' => BaseNiucloudClient::PRODUCT, 'framework_version' => config('version.version') ]; if (!$addon) { $upgrade['app_key'] = AddonDict::FRAMEWORK_KEY; $upgrade['version'] = config('version.version'); } else { $upgrade['app_key'] = $addon; $upgrade['version'] = (new Addon())->where([ ['key', '=', $addon] ])->value('version'); } $response = (new CoreAddonCloudService())->upgradeAddon($upgrade); if (isset($response['code']) && $response['code'] == 0) throw new CommonException($response['msg']); try { $key = uniqid(); $upgrade_dir = $this->upgrade_dir . $key . DIRECTORY_SEPARATOR; if (!is_dir($upgrade_dir)) { dir_mkdir($upgrade_dir); } $upgrade_tsak = [ 'key' => $key, 'upgrade' => $upgrade, 'step' => 'requestUpgrade', 'executed' => ['requestUpgrade'], 'log' => [ $this->steps['requestUpgrade']['title'] ], 'params' => ['token' => $response['token'] ], 'upgrade_content' => $this->getUpgradeContent($addon) ]; Cache::set($this->cache_key, $upgrade_tsak); return $upgrade_tsak; } catch (\Exception $e) { if (strpos($e->getMessage(), 'open_basedir') !== false) { throw new CommonException('OPEN_BASEDIR_ERROR'); } throw new CommonException($e->getMessage()); } } /** * 执行升级 * @return true */ public function execute() { if (!$this->upgrade_task) return true; $steps = array_keys($this->steps); $index = array_search($this->upgrade_task['step'], $steps); $step = $steps[ $index + 1 ] ?? ''; $params = $this->upgrade_task['params'] ?? []; if ($step) { try { $res = $this->$step(...$params); if (is_array($res)) { $this->upgrade_task['params'] = $res; } else { $this->upgrade_task['step'] = $step; $this->upgrade_task['params'] = []; $this->upgrade_task['executed'][] = $step; $this->upgrade_task['log'][] = $this->steps[$step]['title']; } Cache::set($this->cache_key, $this->upgrade_task); } catch (\Exception $e) { $this->upgrade_task['step'] = $step; $this->upgrade_task['error'] = $e->getMessage(); $this->upgradeErrorHandle(); } return true; } else { return true; } } /** * 下载升级文件 * @param string $token * @param string $dir * @param int $index * @param $step * @return true|null */ public function downloadFile(string $token, string $dir = '', int $index = -1, $step = 0, $length = 0) { if (!$dir) { $dir = $this->upgrade_dir .$this->upgrade_task['key'] . DIRECTORY_SEPARATOR . 'download' . DIRECTORY_SEPARATOR; dir_mkdir($dir); } $res = (new CoreAddonCloudService())->downloadUpgradeFile($token, $dir, $index, $step, $length); return $res; } /** * 备份源码 * @return true */ public function backupCode() { (new BackupService())->backupCode(); return true; } /** * 备份数据库 * @return true */ public function backupSql() { (new BackupService())->backupSql(); return true; } /** * 覆盖更新升级的代码 * @return void */ public function coverCode($index = 0) { $this->upgrade_task['is_cover'] = 1; $addon = $this->upgrade_task['upgrade']['app_key']; $version_list = array_reverse($this->upgrade_task['upgrade_content']['version_list']); $code_dir = $this->upgrade_dir .$this->upgrade_task['key'] . DIRECTORY_SEPARATOR . 'download' . DIRECTORY_SEPARATOR . 'code' . DIRECTORY_SEPARATOR; $version_item = $version_list[$index]; $version_no = $version_item['version_no']; $to_dir = $addon == AddonDict::FRAMEWORK_KEY ? $this->root_path : $this->root_path . 'niucloud' . DIRECTORY_SEPARATOR . 'addon' . DIRECTORY_SEPARATOR . $addon; // 获取文件变更记录 if (file_exists($code_dir . $version_no . '.txt')) { $change = array_filter(explode("\n", file_get_contents($code_dir . $version_no . '.txt'))); foreach ($change as &$item) { list($operation, $md5, $file) = $item = explode(' ', $item); if ($operation == '-') { @unlink($to_dir . $file); } } // 合并依赖 $this->installDepend($code_dir . $version_no, array_column($change, 2)); } // 覆盖文件 if (is_dir($code_dir . $version_no)) { dir_copy($code_dir . $version_no, $to_dir); if ($addon != AddonDict::FRAMEWORK_KEY) { (new CoreAddonInstallService($addon))->installDir(); } } $upgrade_file_dir = 'v' . str_replace('.', '', $version_no); if ($addon == AddonDict::FRAMEWORK_KEY) { $class_path = "\\app\\upgrade\\{$upgrade_file_dir}\\Upgrade"; $sql_file = root_path() . 'app' . DIRECTORY_SEPARATOR . 'upgrade' . DIRECTORY_SEPARATOR . $upgrade_file_dir . DIRECTORY_SEPARATOR . 'upgrade.sql'; } else { $class_path = "\\addon\\{$addon}\\app\\upgrade\\{$upgrade_file_dir}\\Upgrade"; $sql_file = root_path() . 'addon' . DIRECTORY_SEPARATOR . $addon . DIRECTORY_SEPARATOR . 'app' . DIRECTORY_SEPARATOR . 'upgrade' . DIRECTORY_SEPARATOR . $upgrade_file_dir . DIRECTORY_SEPARATOR . 'upgrade.sql'; } // 执行升级sql if (file_exists($sql_file)) { $this->executeSql($sql_file); } // 执行升级方法 if (class_exists($class_path)) { (new $class_path())->handle(); } $index ++; if ($index < count($version_list)) { return compact('index'); } else { return true; } } /** * 合并依赖 * @param string $version_no * @return void */ public function installDepend(string $dir, array $change_files) { $addon = $this->upgrade_task['upgrade']['app_key']; $depend_service = new CoreDependService(); if ($addon == AddonDict::FRAMEWORK_KEY) { $composer = '/niucloud/composer.json'; $admin_package = '/admin/package.json'; $web_package = '/web/package.json'; $uniapp_package = '/uni-app/package.json'; } else { $composer = "/niucloud/addon/{$addon}/package/composer.json"; $admin_package = "/niucloud/addon/{$addon}/package/admin-package.json"; $web_package = "/niucloud/addon/{$addon}/package/web-package.json"; $uniapp_package = "/niucloud/addon/{$addon}/package/uni-app-package.json"; } if (in_array($composer, $change_files)) { $original = $depend_service->getComposerContent(); $new = $depend_service->jsonFileToArray($dir . $composer); foreach ($new as $name => $value) { $original[$name] = isset($original[$name]) && is_array($original[$name]) ? array_merge($original[$name], $new[$name]) : $new[$name]; } $depend_service->writeArrayToJsonFile($original, $dir . $composer); } if (in_array($admin_package, $change_files)) { $original = $depend_service->getNpmContent('admin'); $new = $depend_service->jsonFileToArray($dir . $admin_package); foreach ($new as $name => $value) { $original[$name] = isset($original[$name]) && is_array($original[$name]) ? array_merge($original[$name], $new[$name]) : $new[$name]; } $depend_service->writeArrayToJsonFile($original, $dir . $admin_package); } if (in_array($web_package, $change_files)) { $original = $depend_service->getNpmContent('web'); $new = $depend_service->jsonFileToArray($dir . $web_package); foreach ($new as $name => $value) { $original[$name] = isset($original[$name]) && is_array($original[$name]) ? array_merge($original[$name], $new[$name]) : $new[$name]; } $depend_service->writeArrayToJsonFile($original, $dir . $web_package); } if (in_array($uniapp_package, $change_files)) { $original = $depend_service->getNpmContent('uni-app'); $new = $depend_service->jsonFileToArray($dir . $uniapp_package); foreach ($new as $name => $value) { $original[$name] = isset($original[$name]) && is_array($original[$name]) ? array_merge($original[$name], $new[$name]) : $new[$name]; } $depend_service->writeArrayToJsonFile($original, $dir . $uniapp_package); } } /** * 处理手机端 * @param string $verson_no * @return true */ public function handleUniapp() { $code_dir = $this->upgrade_dir .$this->upgrade_task['key'] . DIRECTORY_SEPARATOR . 'download' . DIRECTORY_SEPARATOR . 'code' . DIRECTORY_SEPARATOR; dir_copy($code_dir . 'uni-app', $this->root_path . 'uni-app'); $addon_list = (new CoreAddonService())->getInstallAddonList(); if (!empty($addon_list)) { foreach ($addon_list as $addon => $item) { $this->addon = $addon; // 编译 diy-group 自定义组件代码文件 $this->compileDiyComponentsCode($this->root_path . 'uni-app' . DIRECTORY_SEPARATOR . 'src' . DIRECTORY_SEPARATOR, $addon); // 编译 fixed-group 固定模板组件代码文件 $this->compileFixedComponentsCode($this->root_path . 'uni-app' . DIRECTORY_SEPARATOR . 'src' . DIRECTORY_SEPARATOR, $addon); // 编译 pages.json 页面路由代码文件 $this->installPageCode($this->root_path . 'uni-app' . DIRECTORY_SEPARATOR . 'src' . DIRECTORY_SEPARATOR); // 编译 加载插件标题语言包 $this->compileLocale($this->root_path . 'uni-app' . DIRECTORY_SEPARATOR . 'src' . DIRECTORY_SEPARATOR, $addon); } } return true; } /** * 执行升级sql * @param string $sql_file * @return true */ private function executeSql(string $sql_file) { $sql_content = file_get_contents($sql_file); if (!empty($sql_content)) { $prefix = config('database.connections.mysql.prefix'); $sql_data = array_filter($this->getSqlQuery($sql_content)); if (!empty($sql_data)) { foreach ($sql_data as $sql) { $sql = $prefix ? $this->handleSqlPrefix($sql, $prefix) : $sql; Db::query($sql); } } } return true; } /** * 刷新菜单 * @return void */ public function refreshMenu() { if ($this->upgrade_task['upgrade']['app_key'] == AddonDict::FRAMEWORK_KEY) { (new InstallSystemService())->installMenu(); } else { (new CoreMenuService())->refreshAddonMenu($this->upgrade_task['upgrade']['app_key']); } return true; } /** * 安装计划任务 * @return true */ public function installSchedule() { if ($this->upgrade_task['upgrade']['app_key'] == AddonDict::FRAMEWORK_KEY) { (new CoreScheduleInstallService())->installSystemSchedule(); } else { (new CoreScheduleInstallService())->installAddonSchedule($this->upgrade_task['upgrade']['app_key']); } return true; } /** * 更新完成 * @return void */ public function upgradeComplete() { $addon = $this->upgrade_task['upgrade']['app_key']; if ($addon != AddonDict::FRAMEWORK_KEY) { $core_addon_service = new CoreAddonService(); $install_data = $core_addon_service->getAddonConfig($addon); $install_data['icon'] = 'addon/' . $addon . '/icon.png'; $core_addon_service->set($install_data); } $this->clearUpgradeTask(5); return true; } /** * 升级出错之后的处理 * @return true|void */ public function upgradeErrorHandle() { try { if (isset($this->upgrade_task['is_cover'])) { $restore_service = (new RestoreService()); $restore_service->restoreCode(); $restore_service->restoreSql(); } $this->clearUpgradeTask(5); return true; } catch (\Exception $e) { return true; } } /** * 获取升级内容 * @param string $addon * @return array|\core\util\niucloud\Response|object|\Psr\Http\Message\ResponseInterface * @throws \GuzzleHttp\Exception\GuzzleException */ public function getUpgradeContent(string $addon = '') { $upgrade = [ 'product_key' => BaseNiucloudClient::PRODUCT ]; if (!$addon) { $upgrade['app_key'] = AddonDict::FRAMEWORK_KEY; $upgrade['version'] = config('version.version'); } else { $upgrade['app_key'] = $addon; $upgrade['version'] = (new Addon())->where([ ['key', '=', $addon] ])->value('version'); } return (new CoreModuleService())->getUpgradeContent($upgrade)['data'] ?? []; } /** * 获取正在进行的升级任务 * @return mixed|null */ public function getUpgradeTask() { return $this->upgrade_task; } /** * 清除升级任务 * @return true */ public function clearUpgradeTask(int $delayed = 0) { if ($delayed) { Cache::set($this->cache_key, $this->upgrade_task, $delayed); } else { Cache::set($this->cache_key, null); } return true; } /** * 获取插件定义的package目录 * @param string $addon * @return string */ public function geAddonPackagePath(string $addon) { return root_path() . 'addon' .DIRECTORY_SEPARATOR . $addon . DIRECTORY_SEPARATOR . 'package' . DIRECTORY_SEPARATOR; } }