https://www.fastadmin.net/store.html */ class Addon extends Backend { protected $model = null; public function _initialize() { parent::_initialize(); if (!$this->auth->isSuperAdmin() && in_array($this->request->action(), ['install', 'uninstall', 'local', 'upgrade'])) { $this->error(__('Access is allowed only to the super management group')); } } /** * 查看 */ public function index() { $addons = get_addon_list(); foreach ($addons as $k => &$v) { $config = get_addon_config($v['name']); $v['config'] = $config ? 1 : 0; $v['url'] = str_replace($this->request->server('SCRIPT_NAME'), '', $v['url']); } $this->assignconfig(['addons' => $addons, 'api_url' => config('fastadmin.api_url'), 'faversion' => config('fastadmin.version')]); return $this->view->fetch(); } /** * 配置 */ public function config($name = null) { $name = $name ? $name : $this->request->get("name"); if (!$name) { $this->error(__('Parameter %s can not be empty', 'name')); } if (!preg_match("/^[a-zA-Z0-9]+$/", $name)) { $this->error(__('Addon name incorrect')); } if (!is_dir(ADDON_PATH . $name)) { $this->error(__('Directory not found')); } $info = get_addon_info($name); $config = get_addon_fullconfig($name); if (!$info) { $this->error(__('No Results were found')); } if ($this->request->isPost()) { $params = $this->request->post("row/a", [], 'trim'); if ($params) { foreach ($config as $k => &$v) { if (isset($params[$v['name']])) { if ($v['type'] == 'array') { $params[$v['name']] = is_array($params[$v['name']]) ? $params[$v['name']] : (array)json_decode($params[$v['name']], true); $value = $params[$v['name']]; } else { $value = is_array($params[$v['name']]) ? implode(',', $params[$v['name']]) : $params[$v['name']]; } $v['value'] = $value; } } try { //更新配置文件 set_addon_fullconfig($name, $config); Service::refresh(); $this->success(); } catch (Exception $e) { $this->error(__($e->getMessage())); } } $this->error(__('Parameter %s can not be empty', '')); } $tips = []; foreach ($config as $index => &$item) { if ($item['name'] == '__tips__') { $tips = $item; unset($config[$index]); } } $this->view->assign("addon", ['info' => $info, 'config' => $config, 'tips' => $tips]); $configFile = ADDON_PATH . $name . DS . 'config.html'; $viewFile = is_file($configFile) ? $configFile : ''; return $this->view->fetch($viewFile); } /** * 安装 */ public function install() { $name = $this->request->post("name"); $force = (int)$this->request->post("force"); if (!$name) { $this->error(__('Parameter %s can not be empty', 'name')); } if (!preg_match("/^[a-zA-Z0-9]+$/", $name)) { $this->error(__('Addon name incorrect')); } try { $uid = $this->request->post("uid"); $token = $this->request->post("token"); $version = $this->request->post("version"); $faversion = $this->request->post("faversion"); $extend = [ 'uid' => $uid, 'token' => $token, 'version' => $version, 'faversion' => $faversion ]; Service::install($name, $force, $extend); $info = get_addon_info($name); $info['config'] = get_addon_config($name) ? 1 : 0; $info['state'] = 1; $this->success(__('Install successful'), null, ['addon' => $info]); } catch (AddonException $e) { $this->result($e->getData(), $e->getCode(), __($e->getMessage())); } catch (Exception $e) { $this->error(__($e->getMessage()), $e->getCode()); } } /** * 卸载 */ public function uninstall() { $name = $this->request->post("name"); $force = (int)$this->request->post("force"); if (!$name) { $this->error(__('Parameter %s can not be empty', 'name')); } if (!preg_match("/^[a-zA-Z0-9]+$/", $name)) { $this->error(__('Addon name incorrect')); } try { Service::uninstall($name, $force); $this->success(__('Uninstall successful')); } catch (AddonException $e) { $this->result($e->getData(), $e->getCode(), __($e->getMessage())); } catch (Exception $e) { $this->error(__($e->getMessage())); } } /** * 禁用启用 */ public function state() { $name = $this->request->post("name"); $action = $this->request->post("action"); $force = (int)$this->request->post("force"); if (!$name) { $this->error(__('Parameter %s can not be empty', 'name')); } if (!preg_match("/^[a-zA-Z0-9]+$/", $name)) { $this->error(__('Addon name incorrect')); } try { $action = $action == 'enable' ? $action : 'disable'; //调用启用、禁用的方法 Service::$action($name, $force); Cache::rm('__menu__'); $this->success(__('Operate successful')); } catch (AddonException $e) { $this->result($e->getData(), $e->getCode(), __($e->getMessage())); } catch (Exception $e) { $this->error(__($e->getMessage())); } } /** * 本地上传 */ public function local() { Config::set('default_return_type', 'json'); $file = $this->request->file('file'); $addonTmpDir = RUNTIME_PATH . 'addons' . DS; if (!is_dir($addonTmpDir)) { @mkdir($addonTmpDir, 0755, true); } $info = $file->rule('uniqid')->validate(['size' => 10240000, 'ext' => 'zip'])->move($addonTmpDir); if ($info) { $tmpName = substr($info->getFilename(), 0, stripos($info->getFilename(), '.')); $tmpAddonDir = ADDON_PATH . $tmpName . DS; $tmpFile = $addonTmpDir . $info->getSaveName(); try { Service::unzip($tmpName); unset($info); @unlink($tmpFile); $infoFile = $tmpAddonDir . 'info.ini'; if (!is_file($infoFile)) { throw new Exception(__('Addon info file was not found')); } $config = Config::parse($infoFile, '', $tmpName); $name = isset($config['name']) ? $config['name'] : ''; if (!$name) { throw new Exception(__('Addon info file data incorrect')); } if (!preg_match("/^[a-zA-Z0-9]+$/", $name)) { throw new Exception(__('Addon name incorrect')); } $newAddonDir = ADDON_PATH . $name . DS; if (is_dir($newAddonDir)) { throw new Exception(__('Addon already exists')); } //重命名插件文件夹 rename($tmpAddonDir, $newAddonDir); try { //默认禁用该插件 $info = get_addon_info($name); if ($info['state']) { $info['state'] = 0; set_addon_info($name, $info); } //执行插件的安装方法 $class = get_addon_class($name); if (class_exists($class)) { $addon = new $class(); $addon->install(); } //导入SQL Service::importsql($name); $info['config'] = get_addon_config($name) ? 1 : 0; $this->success(__('Offline installed tips'), null, ['addon' => $info]); } catch (Exception $e) { @rmdirs($newAddonDir); throw new Exception(__($e->getMessage())); } } catch (Exception $e) { unset($info); @unlink($tmpFile); @rmdirs($tmpAddonDir); $this->error(__($e->getMessage())); } } else { // 上传失败获取错误信息 $this->error(__($file->getError())); } } /** * 更新插件 */ public function upgrade() { $name = $this->request->post("name"); $addonTmpDir = RUNTIME_PATH . 'addons' . DS; if (!$name) { $this->error(__('Parameter %s can not be empty', 'name')); } if (!preg_match("/^[a-zA-Z0-9]+$/", $name)) { $this->error(__('Addon name incorrect')); } if (!is_dir($addonTmpDir)) { @mkdir($addonTmpDir, 0755, true); } try { $uid = $this->request->post("uid"); $token = $this->request->post("token"); $version = $this->request->post("version"); $faversion = $this->request->post("faversion"); $extend = [ 'uid' => $uid, 'token' => $token, 'version' => $version, 'faversion' => $faversion ]; //调用更新的方法 Service::upgrade($name, $extend); Cache::rm('__menu__'); $this->success(__('Operate successful')); } catch (AddonException $e) { $this->result($e->getData(), $e->getCode(), __($e->getMessage())); } catch (Exception $e) { $this->error(__($e->getMessage())); } } /** * 已装插件 */ public function downloaded() { $offset = (int)$this->request->get("offset"); $limit = (int)$this->request->get("limit"); $filter = $this->request->get("filter"); $search = $this->request->get("search"); $search = htmlspecialchars(strip_tags($search)); $onlineaddons = Cache::get("onlineaddons"); if (!is_array($onlineaddons)) { $onlineaddons = []; $result = Http::sendRequest(config('fastadmin.api_url') . '/addon/index'); if ($result['ret']) { $json = (array)json_decode($result['msg'], true); $rows = isset($json['rows']) ? $json['rows'] : []; foreach ($rows as $index => $row) { $onlineaddons[$row['name']] = $row; } } Cache::set("onlineaddons", $onlineaddons, 600); } $filter = (array)json_decode($filter, true); $addons = get_addon_list(); $list = []; foreach ($addons as $k => $v) { if ($search && stripos($v['name'], $search) === false && stripos($v['intro'], $search) === false) { continue; } if (isset($onlineaddons[$v['name']])) { $v = array_merge($v, $onlineaddons[$v['name']]); } else { $v['category_id'] = 0; $v['flag'] = ''; $v['banner'] = ''; $v['image'] = ''; $v['donateimage'] = ''; $v['demourl'] = ''; $v['price'] = __('None'); $v['screenshots'] = []; $v['releaselist'] = []; } $v['url'] = addon_url($v['name']); $v['url'] = str_replace($this->request->server('SCRIPT_NAME'), '', $v['url']); $v['createtime'] = filemtime(ADDON_PATH . $v['name']); if ($filter && isset($filter['category_id']) && is_numeric($filter['category_id']) && $filter['category_id'] != $v['category_id']) { continue; } $list[] = $v; } $total = count($list); if ($limit) { $list = array_slice($list, $offset, $limit); } $result = array("total" => $total, "rows" => $list); $callback = $this->request->get('callback') ? "jsonp" : "json"; return $callback($result); } }