installDir = root_path() . 'modules' . DIRECTORY_SEPARATOR; $this->ebakDir = $this->installDir . 'ebak' . DIRECTORY_SEPARATOR; if (!is_dir($this->installDir)) { mkdir($this->installDir, 0755, true); } if (!is_dir($this->ebakDir)) { mkdir($this->ebakDir, 0755, true); } if ($uid) { $this->uid = $uid; $this->modulesDir = $this->installDir . $uid . DIRECTORY_SEPARATOR; } } public function getInstallState() { if (!is_dir($this->modulesDir)) { return self::UNINSTALLED; } $info = $this->getInfo(); if ($info && isset($info['state'])) { return $info['state']; } // 目录已存在,但非正常的模块 return dir_is_empty($this->modulesDir) ? self::UNINSTALLED : self::DIRECTORY_OCCUPIED; } public function download(string $token, int $orderId) { if (!$orderId) { throw new Exception('Order not found'); } // 下载 $sysVersion = Config::get('buildadmin.version'); $installed = Server::installedList($this->installDir); foreach ($installed as $item) { $installedUids[] = $item['uid']; } unset($installed); $zipFile = Server::download($this->uid, $this->installDir, [ 'sysVersion' => $sysVersion, 'ba-user-token' => $token, 'order_id' => $orderId, // 传递已安装模块,做互斥检测 'installed' => $installedUids ?? [], ]); // 删除旧版本代码 deldir($this->modulesDir); // 解压 Server::unzip($zipFile); // 删除下载的zip @unlink($zipFile); // 检查是否完整 $this->checkPackage(); // 设置为待安装状态 $this->setInfo([ 'state' => self::WAIT_INSTALL, ]); return $zipFile; } public function upload(string $file) { $file = path_transform(root_path() . 'public' . $file); if (!is_file($file)) { throw new Exception('Zip file not found'); } $copyTo = $this->installDir . 'uploadTemp' . date('YmdHis') . '.zip'; copy($file, $copyTo); // 解压 $copyToDir = Server::unzip($copyTo); $copyToDir .= DIRECTORY_SEPARATOR; // 删除zip @unlink($copyTo); // 读取ini $info = Server::getIni($copyToDir); if (!isset($info['uid']) || !$info['uid']) { deldir($copyToDir); throw new Exception('Basic configuration of the Module is incomplete'); } $this->uid = $info['uid']; $this->modulesDir = $this->installDir . $info['uid'] . DIRECTORY_SEPARATOR; if (is_dir($this->modulesDir)) { $info = $this->getInfo(); if ($info && isset($info['uid'])) { deldir($copyToDir); throw new Exception('Module already exists'); } if (!dir_is_empty($this->modulesDir)) { deldir($copyToDir); throw new Exception('The directory required by the module is occupied'); } } rename($copyToDir, $this->modulesDir); // 检查是否完整 $this->checkPackage(); // 设置为待安装状态 $this->setInfo([ 'state' => self::WAIT_INSTALL, ]); return $info; } public function update(string $token, int $orderId) { $state = $this->getInstallState(); if ($state != self::DISABLE) { throw new Exception('Please disable the module before updating'); } $this->download($token, $orderId); // 标记需要执行更新脚本,并在安装请求执行(当前请求未自动加载到新文件不方便执行) $info = $this->getInfo(); $info['update'] = 1; $this->setInfo([], $info); } /** * 安装模板或案例 * @param string $token 用户token * @param int $orderId 订单号 * @throws moduleException * @throws Exception */ public function install(string $token, int $orderId) { $state = $this->getInstallState(); if ($state == self::INSTALLED || $state == self::DIRECTORY_OCCUPIED || $state == self::DISABLE) { throw new Exception('Module already exists'); } if ($state == self::UNINSTALLED) { $this->download($token, $orderId); } // 导入sql Server::importSql($this->modulesDir); // 如果是更新,先执行更新脚本 $info = $this->getInfo(); if (isset($info['update']) && $info['update']) { Server::execEvent($this->uid, 'update'); unset($info['update']); $this->setInfo([], $info); } // 执行安装脚本 Server::execEvent($this->uid, 'install'); // 启用插件 $this->enable('install'); return $info; } public function uninstall() { $info = $this->getInfo(); if ($info['state'] != self::DISABLE) { throw new moduleException('Please disable the module first', 0, [ 'uid' => $this->uid, ]); } // 执行卸载脚本 Server::execEvent($this->uid, 'uninstall'); deldir($this->modulesDir); } /** * 启禁用模块 */ public function changeState(bool $state) { $info = $this->getInfo(); $canDisable = [ self::INSTALLED, self::CONFLICT_PENDING, self::DEPENDENT_WAIT_INSTALL, ]; if (!$state) { if (!in_array($info['state'], $canDisable)) { throw new moduleException('The current state of the module cannot be set to disabled', 0, [ 'uid' => $this->uid, 'state' => $info['state'], ]); } return $this->disable(); } if ($info['state'] != self::DISABLE) { throw new moduleException('The current state of the module cannot be set to enabled', 0, [ 'uid' => $this->uid, 'state' => $info['state'], ]); } $this->setInfo([ 'state' => self::WAIT_INSTALL, ]); return $info; } public function enable(string $trigger) { // 安装 WebBootstrap Server::installWebBootstrap($this->uid, $this->modulesDir); $this->conflictHandle($trigger); // 执行启用脚本 Server::execEvent($this->uid, 'enable'); $this->dependUpdateHandle(); } public function disable() { $update = request()->post("update/b", false); $confirmConflict = request()->post("confirmConflict/b", false); $dependConflictSolution = request()->post("dependConflictSolution/a", []); $info = $this->getInfo(); $zipFile = $this->ebakDir . $this->uid . '-install.zip'; $zipDir = false; if (is_file($zipFile)) { try { $zipDir = Server::unzip($zipFile); } catch (moduleException|Exception $e) { // skip } } $conflictFile = Server::getFileList($this->modulesDir, true); $dependConflict = $this->disableDependCheck(); if (($conflictFile || !self::emptyArray($dependConflict)) && !$confirmConflict) { $dependConflictTemp = []; foreach ($dependConflict as $env => $item) { $dev = !(stripos($env, 'dev') === false); foreach ($item as $depend => $v) { $dependConflictTemp[] = [ 'env' => $env, 'depend' => $depend, 'dependTitle' => $depend . ' ' . $v, 'solution' => 'delete', ]; } } throw new moduleException('Module file updated', -1, [ 'uid' => $this->uid, 'conflictFile' => $conflictFile, 'dependConflict' => $dependConflictTemp, ]); } // 执行禁用脚本 Server::execEvent($this->uid, 'disable', ['update' => $update]); // 是否需要备份依赖? $delNpmDepend = false; $delNuxtNpmDepend = false; $delComposerDepend = false; foreach ($dependConflictSolution as $env => $depends) { if (!$depends) continue; if ($env == 'require' || $env == 'require-dev') { $delComposerDepend = true; } elseif ($env == 'dependencies' || $env == 'devDependencies') { $delNpmDepend = true; } elseif ($env == 'nuxtDependencies' || $env == 'nuxtDevDependencies') { $delNuxtNpmDepend = true; } } // 备份 $dependWaitInstall = []; if ($delComposerDepend) { $conflictFile[] = 'composer.json'; $dependWaitInstall[] = [ 'pm' => false, 'command' => 'composer.update', 'type' => 'composer_dependent_wait_install', ]; } if ($delNpmDepend) { $conflictFile[] = 'web' . DIRECTORY_SEPARATOR . 'package.json'; $dependWaitInstall[] = [ 'pm' => true, 'command' => 'web-install', 'type' => 'npm_dependent_wait_install', ]; } if ($delNuxtNpmDepend) { $conflictFile[] = 'web-nuxt' . DIRECTORY_SEPARATOR . 'package.json'; $dependWaitInstall[] = [ 'pm' => true, 'command' => 'nuxt-install', 'type' => 'nuxt_npm_dependent_wait_install', ]; } if ($conflictFile) { $ebakZip = $this->ebakDir . $this->uid . '-disable-' . date('YmdHis') . '.zip'; Server::createZip($conflictFile, $ebakZip); } // 删除依赖 $serverDepend = new Depends(root_path() . 'composer.json', 'composer'); $webDep = new Depends(root_path() . 'web' . DIRECTORY_SEPARATOR . 'package.json'); $webNuxtDep = new Depends(root_path() . 'web-nuxt' . DIRECTORY_SEPARATOR . 'package.json'); foreach ($dependConflictSolution as $env => $depends) { if (!$depends) continue; $dev = !(stripos($env, 'dev') === false); if ($env == 'require' || $env == 'require-dev') { $serverDepend->removeDepends($depends, $dev); } elseif ($env == 'dependencies' || $env == 'devDependencies') { $webDep->removeDepends($depends, $dev); } elseif ($env == 'nuxtDependencies' || $env == 'nuxtDevDependencies') { $webNuxtDep->removeDepends($depends, $dev); } } // 删除模块文件 $protectedFiles = Server::getConfig($this->modulesDir, 'protectedFiles'); foreach ($protectedFiles as &$protectedFile) { $protectedFile = path_transform(root_path() . $protectedFile); } $moduleFile = Server::getFileList($this->modulesDir); foreach ($moduleFile as $item) { $file = path_transform(root_path() . $item); if (in_array($file, $protectedFiles)) { continue; } if (file_exists($file)) { unlink($file); } del_empty_dir(dirname($file)); } // 恢复备份文件 if ($zipDir) { foreach ( $iterator = new RecursiveIteratorIterator( new RecursiveDirectoryIterator($zipDir, FilesystemIterator::SKIP_DOTS), RecursiveIteratorIterator::SELF_FIRST ) as $item ) { $ebakFile = path_transform(root_path() . $iterator->getSubPathName()); if ($item->isDir()) { if (!is_dir($ebakFile)) { mkdir($ebakFile, 0755, true); } } else { if ($ebakFile != path_transform(root_path() . 'composer.json') && $ebakFile != path_transform(root_path() . 'web/package.json')) { copy($item, $ebakFile); } } } } // 删除解压后的备份文件 deldir($zipDir); // 卸载 WebBootstrap Server::uninstallWebBootstrap($this->uid, $this->modulesDir); $this->setInfo([ 'state' => self::DISABLE, ]); if ($update) { $token = request()->post("token/s", ''); $order = request()->post("order/d", 0); $this->update($token, $order); throw new moduleException('update', -3, [ 'uid' => $this->uid, ]); } if ($dependWaitInstall) { throw new moduleException('dependent wait install', -2, [ 'uid' => $this->uid, 'wait_install' => $dependWaitInstall, ]); } return $info; } /** * 处理依赖和文件冲突,并完成与前端的冲突处理交互 * @throws moduleException|Exception */ public function conflictHandle(string $trigger): bool { $info = $this->getInfo(); if ($info['state'] != self::WAIT_INSTALL && $info['state'] != self::CONFLICT_PENDING) { return false; } $fileConflict = Server::getFileList($this->modulesDir, true);// 文件冲突 $dependConflict = Server::dependConflictCheck($this->modulesDir);// 依赖冲突 $installFiles = Server::getFileList($this->modulesDir);// 待安装文件 $depends = Server::getDepend($this->modulesDir);// 待安装依赖 $coverFiles = [];// 要覆盖的文件-备份 $discardFiles = [];// 抛弃的文件-复制时不覆盖 $serverDep = new Depends(root_path() . 'composer.json', 'composer'); $webDep = new Depends(root_path() . 'web' . DIRECTORY_SEPARATOR . 'package.json'); $webNuxtDep = new Depends(root_path() . 'web-nuxt' . DIRECTORY_SEPARATOR . 'package.json'); if ($fileConflict || !self::emptyArray($dependConflict)) { $extend = request()->post('extend/a', []); if (!$extend) { // 发现冲突->手动处理->转换为方便前端使用的格式 $fileConflictTemp = []; foreach ($fileConflict as $key => $item) { $fileConflictTemp[$key] = [ 'newFile' => $this->uid . DIRECTORY_SEPARATOR . $item, 'oldFile' => $item, 'solution' => 'cover', ]; } $dependConflictTemp = []; foreach ($dependConflict as $env => $item) { $dev = !(stripos($env, 'dev') === false); foreach ($item as $depend => $v) { $oldDepend = ''; if (in_array($env, ['require', 'require-dev'])) { $oldDepend = $depend . ' ' . $serverDep->hasDepend($depend, $dev); } elseif (in_array($env, ['dependencies', 'devDependencies'])) { $oldDepend = $depend . ' ' . $webDep->hasDepend($depend, $dev); } elseif (in_array($env, ['nuxtDependencies', 'nuxtDevDependencies'])) { $oldDepend = $depend . ' ' . $webNuxtDep->hasDepend($depend, $dev); } $dependConflictTemp[] = [ 'env' => $env, 'newDepend' => $depend . ' ' . $v, 'oldDepend' => $oldDepend, 'depend' => $depend, 'solution' => 'cover', ]; } } $this->setInfo([ 'state' => self::CONFLICT_PENDING, ]); throw new moduleException('Module file conflicts', -1, [ 'fileConflict' => $fileConflictTemp, 'dependConflict' => $dependConflictTemp, 'uid' => $this->uid, 'state' => self::CONFLICT_PENDING, ]); } // 处理冲突 if ($fileConflict && isset($extend['fileConflict'])) { foreach ($installFiles as $ikey => $installFile) { if (isset($extend['fileConflict'][$installFile])) { if ($extend['fileConflict'][$installFile] == 'discard') { $discardFiles[] = $installFile; unset($installFiles[$ikey]); } else { $coverFiles[] = $installFile; } } } } if (!self::emptyArray($dependConflict) && isset($extend['dependConflict'])) { foreach ($depends as $fKey => $fItem) { foreach ($fItem as $cKey => $cItem) { if (isset($extend['dependConflict'][$fKey][$cKey])) { if ($extend['dependConflict'][$fKey][$cKey] == 'discard') { unset($depends[$fKey][$cKey]); } } } } } } // 如果有依赖更新,增加要备份的文件 if ($depends) { foreach ($depends as $key => $item) { if (!$item) { continue; } if ($key == 'require' || $key == 'require-dev') { $coverFiles[] = 'composer.json'; continue; } if ($key == 'dependencies' || $key == 'devDependencies') { $coverFiles[] = 'web' . DIRECTORY_SEPARATOR . 'package.json'; } } } // 备份将被覆盖的文件 if ($coverFiles) { $ebakZip = $trigger == 'install' ? $this->ebakDir . $this->uid . '-install.zip' : $this->ebakDir . $this->uid . '-cover-' . date('YmdHis') . '.zip'; Server::createZip($coverFiles, $ebakZip); } if ($depends) { $npm = false; $composer = false; $nuxtNpm = false; foreach ($depends as $key => $item) { if (!$item) { continue; } if ($key == 'require') { $composer = true; $serverDep->addDepends($item, false, true); } elseif ($key == 'require-dev') { $composer = true; $serverDep->addDepends($item, true, true); } elseif ($key == 'dependencies') { $npm = true; $webDep->addDepends($item, false, true); } elseif ($key == 'devDependencies') { $npm = true; $webDep->addDepends($item, true, true); } elseif ($key == 'nuxtDependencies') { $nuxtNpm = true; $webNuxtDep->addDepends($item, false, true); } elseif ($key == 'nuxtDevDependencies') { $nuxtNpm = true; $webNuxtDep->addDepends($item, true, true); } } if ($npm) { $info['npm_dependent_wait_install'] = 1; $info['state'] = self::DEPENDENT_WAIT_INSTALL; } if ($composer) { $info['composer_dependent_wait_install'] = 1; $info['state'] = self::DEPENDENT_WAIT_INSTALL; } if ($nuxtNpm) { $info['nuxt_npm_dependent_wait_install'] = 1; $info['state'] = self::DEPENDENT_WAIT_INSTALL; } if ($info['state'] != self::DEPENDENT_WAIT_INSTALL) { // 无冲突 $this->setInfo([ 'state' => self::INSTALLED, ]); } else { $this->setInfo([], $info); } } else { // 无冲突 $this->setInfo([ 'state' => self::INSTALLED, ]); } // 复制文件 $overwriteDir = Server::getOverwriteDir(); foreach ($overwriteDir as $dirItem) { $baseDir = $this->modulesDir . $dirItem; $destDir = root_path() . $dirItem; if (!is_dir($baseDir)) { continue; } foreach ( $iterator = new RecursiveIteratorIterator( new RecursiveDirectoryIterator($baseDir, FilesystemIterator::SKIP_DOTS), RecursiveIteratorIterator::SELF_FIRST ) as $item ) { $destDirItem = $destDir . DIRECTORY_SEPARATOR . $iterator->getSubPathName(); if ($item->isDir()) { $this->createDirectory($destDirItem); } else { if (!in_array(str_replace(root_path(), '', $destDirItem), $discardFiles)) { $this->createDirectory(dirname($destDirItem)); copy($item, $destDirItem); } } } } return true; } public function createDirectory(string $dirName) { if (!is_dir($dirName)) { mkdir($dirName, 0755, true); } } public function dependUpdateHandle() { $info = $this->getInfo(); if ($info['state'] == self::DEPENDENT_WAIT_INSTALL) { $waitInstall = []; if (isset($info['composer_dependent_wait_install'])) { $waitInstall[] = 'composer_dependent_wait_install'; } if (isset($info['npm_dependent_wait_install'])) { $waitInstall[] = 'npm_dependent_wait_install'; } if (isset($info['nuxt_npm_dependent_wait_install'])) { $waitInstall[] = 'nuxt_npm_dependent_wait_install'; } if ($waitInstall) { throw new moduleException('dependent wait install', -2, [ 'uid' => $this->uid, 'state' => self::DEPENDENT_WAIT_INSTALL, 'wait_install' => $waitInstall, ]); } else { $this->setInfo([ 'state' => self::INSTALLED, ]); } } } public function dependentInstallComplete(string $type) { $info = $this->getInfo(); if ($info['state'] == self::DEPENDENT_WAIT_INSTALL) { if ($type == 'npm') { unset($info['npm_dependent_wait_install']); } if ($type == 'nuxt_npm') { unset($info['nuxt_npm_dependent_wait_install']); } if ($type == 'composer') { unset($info['composer_dependent_wait_install']); } if ($type == 'all') { unset($info['npm_dependent_wait_install'], $info['composer_dependent_wait_install'], $info['nuxt_npm_dependent_wait_install']); } if (!isset($info['npm_dependent_wait_install']) && !isset($info['composer_dependent_wait_install']) && !isset($info['nuxt_npm_dependent_wait_install'])) { $info['state'] = self::INSTALLED; } $this->setInfo([], $info); } } public function disableDependCheck() { // 读取模块所有依赖 $depend = Server::getDepend($this->modulesDir); if (!$depend) { return []; } // 读取所有依赖中,系统上已经安装的依赖 $serverDep = new Depends(root_path() . 'composer.json', 'composer'); $webDep = new Depends(root_path() . 'web' . DIRECTORY_SEPARATOR . 'package.json'); $webNuxtDep = new Depends(root_path() . 'web-nuxt' . DIRECTORY_SEPARATOR . 'package.json'); foreach ($depend as $key => $depends) { $dev = !(stripos($key, 'dev') === false); if ($key == 'require' || $key == 'require-dev') { foreach ($depends as $dependKey => $dependItem) { if (!$serverDep->hasDepend($dependKey, $dev)) { unset($depends[$dependKey]); } } $depend[$key] = $depends; } elseif ($key == 'dependencies' || $key == 'devDependencies') { foreach ($depends as $dependKey => $dependItem) { if (!$webDep->hasDepend($dependKey, $dev)) { unset($depends[$dependKey]); } } $depend[$key] = $depends; } elseif ($key == 'nuxtDependencies' || $key == 'nuxtDevDependencies') { foreach ($depends as $dependKey => $dependItem) { if (!$webNuxtDep->hasDepend($dependKey, $dev)) { unset($depends[$dependKey]); } } $depend[$key] = $depends; } } return $depend; } public function disableDependConflictCheck(string $ebakDir): array { if (!$ebakDir) { return []; } $dependFile = [ path_transform($ebakDir . DIRECTORY_SEPARATOR . 'composer.json') => [ 'path' => path_transform(root_path() . 'composer.json'), 'type' => 'composer', ], path_transform($ebakDir . DIRECTORY_SEPARATOR . 'web/package.json') => [ 'path' => path_transform(root_path() . 'web/package.json'), 'type' => 'npm', ], ]; $conflict = []; foreach ($dependFile as $key => $item) { if (is_file($key) && (filesize($key) != filesize($item['path']) || md5_file($key) != md5_file($item['path']))) { $conflict[] = $item['type']; } } return $conflict; } public function checkPackage(): bool { if (!is_dir($this->modulesDir)) { throw new Exception('Module package file does not exist'); } $info = $this->getInfo(); $infoKeys = ['uid', 'title', 'intro', 'author', 'version', 'state']; foreach ($infoKeys as $value) { if (!array_key_exists($value, $info)) { deldir($this->modulesDir); throw new Exception('Basic configuration of the Module is incomplete'); } } return true; } public function getInfo() { return Server::getIni($this->modulesDir); } public function setInfo(array $kv = [], array $arr = []): bool { if ($kv) { $info = $this->getInfo(); foreach ($kv as $k => $v) { $info[$k] = $v; } return Server::setIni($this->modulesDir, $info); } elseif ($arr) { return Server::setIni($this->modulesDir, $arr); } throw new Exception('Parameter error'); } /** * 检查多维数组是否为空 */ public static function emptyArray($arr) { $empty = true; foreach ($arr as $item) { if (is_array($item)) { $empty = self::emptyArray($item); if (!$empty) return false; } elseif ($item) { return false; } } return $empty; } }