commit d7a086bf98ce61dd1471e383af1e65af37af50b1 Author: mkm <727897186@qq.com> Date: Mon Oct 9 13:42:49 2023 +0800 Initial commit diff --git a/.example.env b/.example.env new file mode 100755 index 0000000..aa607c9 --- /dev/null +++ b/.example.env @@ -0,0 +1 @@ +APP_DEBUG=true DB_CONNECTION=mysql DB_HOST=127.0.0.1 DB_PORT=3306 DB_DATABASE=likeadmin DB_USERNAME=root DB_PASSWORD=root SESSION_DRIVER=file REDIS_CONNECTION=default REDIS_HOST=127.0.0.1 REDIS_PASSWORD=null REDIS_PORT=6379 REDIS_DB=24 YLY_PARTNER=25991 YLY_API_KEY=d955cc2296d69b4094c6465aad360dc6b19a8c77 YLY_REQUEST_URL=http://open.10ss.net:8888 UNIQUE_IDENTIFICATION = "11d3" DEMO_ENV = "" \ No newline at end of file diff --git a/.gitignore b/.gitignore new file mode 100644 index 0000000..0307853 --- /dev/null +++ b/.gitignore @@ -0,0 +1,7 @@ +/.idea +/.vscode +/vendor +*.log +.env +/tests/tmp +/tests/.phpunit.result.cache diff --git a/LICENSE b/LICENSE new file mode 100644 index 0000000..2c66292 --- /dev/null +++ b/LICENSE @@ -0,0 +1,21 @@ +MIT License + +Copyright (c) 2021 walkor and contributors (see https://github.com/walkor/webman/contributors) + +Permission is hereby granted, free of charge, to any person obtaining a copy +of this software and associated documentation files (the "Software"), to deal +in the Software without restriction, including without limitation the rights +to use, copy, modify, merge, publish, distribute, sublicense, and/or sell +copies of the Software, and to permit persons to whom the Software is +furnished to do so, subject to the following conditions: + +The above copyright notice and this permission notice shall be included in all +copies or substantial portions of the Software. + +THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND, EXPRESS OR +IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF MERCHANTABILITY, +FITNESS FOR A PARTICULAR PURPOSE AND NONINFRINGEMENT. IN NO EVENT SHALL THE +AUTHORS OR COPYRIGHT HOLDERS BE LIABLE FOR ANY CLAIM, DAMAGES OR OTHER +LIABILITY, WHETHER IN AN ACTION OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, +OUT OF OR IN CONNECTION WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE +SOFTWARE. diff --git a/README.md b/README.md new file mode 100644 index 0000000..34c8bd4 --- /dev/null +++ b/README.md @@ -0,0 +1,20 @@ +# webman + +High performance HTTP Service Framework for PHP based on [Workerman](https://github.com/walkor/workerman). + +# Manual (文档) + +https://www.workerman.net/doc/webman + +# Home page (主页) +https://www.workerman.net/webman + + +# Benchmarks (压测) + +https://www.techempower.com/benchmarks/#section=test&runid=9716e3cd-9e53-433c-b6c5-d2c48c9593c1&hw=ph&test=db&l=zg24n3-1r&a=2 +![image](https://user-images.githubusercontent.com/6073368/96447814-120fc980-1245-11eb-938d-6ea408716c72.png) + +## LICENSE + +MIT diff --git a/app/BaseController.php b/app/BaseController.php new file mode 100755 index 0000000..b3fed7a --- /dev/null +++ b/app/BaseController.php @@ -0,0 +1,87 @@ +request = request(); + + // 控制器初始化 + $this->initialize(); + } + + // 初始化 + protected function initialize() + {} + + /** + * 验证数据 + * @access protected + * @param array $data 数据 + * @param string|array $validate 验证器名或者验证规则数组 + * @param array $message 提示信息 + * @param bool $batch 是否批量验证 + * @return array|string|true + * @throws ValidateException + */ + protected function validate(array $data, $validate, array $message = [], bool $batch = false) + { + // if (is_array($validate)) { + // $v = new Validate(); + // $v->rule($validate); + // } else { + // if (strpos($validate, '.')) { + // // 支持场景 + // [$validate, $scene] = explode('.', $validate); + // } + // $class = false !== strpos($validate, '\\') ? $validate : $this->app->parseClass('validate', $validate); + // $v = new $class(); + // if (!empty($scene)) { + // $v->scene($scene); + // } + // } + + // $v->message($message); + + // // 是否批量验证 + // if ($batch || $this->batchValidate) { + // $v->batch(true); + // } + + // return $v->failException(true)->check($data); + } + + +} diff --git a/app/Request.php b/app/Request.php new file mode 100644 index 0000000..91de0d0 --- /dev/null +++ b/app/Request.php @@ -0,0 +1,11 @@ +request->adminInfo) && $this->request->adminInfo) { + $this->adminInfo = $this->request->adminInfo; + $this->adminId = $this->request->adminInfo['admin_id']; + } + } +} \ No newline at end of file diff --git a/app/admin/controller/ConfigController.php b/app/admin/controller/ConfigController.php new file mode 100755 index 0000000..2cc101b --- /dev/null +++ b/app/admin/controller/ConfigController.php @@ -0,0 +1,59 @@ +data($data); + } + + + /** + * @notes 根据类型获取字典数据 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/9/27 19:10 + */ + public function dict() + { + $type = $this->request->get('type', ''); + $data = ConfigLogic::getDictByType($type); + return $this->data($data); + } + + + +} \ No newline at end of file diff --git a/app/admin/controller/DownloadController.php b/app/admin/controller/DownloadController.php new file mode 100644 index 0000000..8d5cdc4 --- /dev/null +++ b/app/admin/controller/DownloadController.php @@ -0,0 +1,38 @@ +get('file'); + + //通过文件缓存的key获取文件储存的路径 + $exportCache = new ExportCache(); + $fileInfo = $exportCache->getFile($fileKey); + + if (empty($fileInfo)) { + return JsonService::fail('下载文件不存在'); + } + + //下载前删除缓存 + Cache::delete($fileKey); + + return response()->download($fileInfo['src'] . $fileInfo['name'],$fileInfo['name']); + } +} \ No newline at end of file diff --git a/app/admin/controller/FileController.php b/app/admin/controller/FileController.php new file mode 100644 index 0000000..9f3f4eb --- /dev/null +++ b/app/admin/controller/FileController.php @@ -0,0 +1,113 @@ +dataLists(new FileLists()); + } + + + /** + * @notes 文件移动成功 + * @author 乔峰 + * @date 2021/12/29 14:30 + */ + public function move() + { + $params = (new FileValidate())->post()->goCheck('move'); + FileLogic::move($params); + return $this->success('移动成功', [], 1, 1); + } + + + /** + * @notes 重命名文件 + * @author 乔峰 + * @date 2021/12/29 14:31 + */ + public function rename() + { + $params = (new FileValidate())->post()->goCheck('rename'); + FileLogic::rename($params); + return $this->success('重命名成功', [], 1, 1); + } + + + /** + * @notes 删除文件 + * @author 乔峰 + * @date 2021/12/29 14:31 + */ + public function delete() + { + $params = (new FileValidate())->post()->goCheck('delete'); + FileLogic::delete($params); + return $this->success('删除成功', [], 1, 1); + } + + + /** + * @notes 分类列表 + * @author 乔峰 + * @date 2021/12/29 14:31 + */ + public function listCate() + { + return $this->dataLists(new FileCateLists()); + } + + + /** + * @notes 添加文件分类 + * @author 乔峰 + * @date 2021/12/29 14:31 + */ + public function addCate() + { + $params = (new FileValidate())->post()->goCheck('addCate'); + FileLogic::addCate($params); + return $this->success('添加成功', [], 1, 1); + } + + + /** + * @notes 编辑文件分类 + * @author 乔峰 + * @date 2021/12/29 14:31 + */ + public function editCate() + { + $params = (new FileValidate())->post()->goCheck('editCate'); + FileLogic::editCate($params); + return $this->success('编辑成功', [], 1, 1); + } + + + /** + * @notes 删除文件分类 + * @author 乔峰 + * @date 2021/12/29 14:32 + */ + public function delCate() + { + $params = (new FileValidate())->post()->goCheck('id'); + FileLogic::delCate($params); + return $this->success('删除成功', [], 1, 1); + } +} \ No newline at end of file diff --git a/app/admin/controller/LoginController.php b/app/admin/controller/LoginController.php new file mode 100755 index 0000000..01b37fd --- /dev/null +++ b/app/admin/controller/LoginController.php @@ -0,0 +1,58 @@ +post()->goCheck(); + return $this->data((new LoginLogic())->login($params)); + } + + /** + * @notes 退出登录 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2021/7/8 00:36 + */ + public function logout() + { + //退出登录情况特殊,只有成功的情况,也不需要token验证 + (new LoginLogic())->logout($this->adminInfo); + return $this->success(); + } +} \ No newline at end of file diff --git a/app/admin/controller/UploadController.php b/app/admin/controller/UploadController.php new file mode 100644 index 0000000..00d5341 --- /dev/null +++ b/app/admin/controller/UploadController.php @@ -0,0 +1,44 @@ +request->post('cid', 0); + $result = UploadService::image($cid); + return $this->success('上传成功', $result); + } catch (Exception $e) { + return $this->fail($e->getMessage()); + } + } + + /** + * @notes 上传视频 + * @author 乔峰 + * @date 2021/12/29 16:27 + */ + public function video() + { + try { + $cid = $this->request->post('cid', 0); + $result = UploadService::video($cid); + return $this->success('上传成功', $result); + } catch (Exception $e) { + return $this->fail($e->getMessage()); + } + } +} \ No newline at end of file diff --git a/app/admin/controller/WorkbenchController.php b/app/admin/controller/WorkbenchController.php new file mode 100755 index 0000000..893ce22 --- /dev/null +++ b/app/admin/controller/WorkbenchController.php @@ -0,0 +1,37 @@ +data($result); + } +} \ No newline at end of file diff --git a/app/admin/controller/auth/AdminController.php b/app/admin/controller/auth/AdminController.php new file mode 100755 index 0000000..f416836 --- /dev/null +++ b/app/admin/controller/auth/AdminController.php @@ -0,0 +1,127 @@ +dataLists(new AdminLists()); + } + + + /** + * @notes 添加管理员 + * @author 乔峰 + * @date 2021/12/29 10:21 + */ + public function add() + { + $params = (new AdminValidate())->post()->goCheck('add'); + $result = AdminLogic::add($params); + if (true === $result) { + return $this->success('操作成功', [], 1, 1); + } + return $this->fail(AdminLogic::getError()); + } + + + /** + * @notes 编辑管理员 + * @author 乔峰 + * @date 2021/12/29 11:03 + */ + public function edit() + { + $params = (new AdminValidate())->post()->goCheck('edit'); + $result = AdminLogic::edit($params); + if (true === $result) { + return $this->success('操作成功', [], 1, 1); + } + return $this->fail(AdminLogic::getError()); + } + + + /** + * @notes 删除管理员 + * @author 乔峰 + * @date 2021/12/29 11:03 + */ + public function delete() + { + $params = (new AdminValidate())->post()->goCheck('delete'); + $result = AdminLogic::delete($params); + if (true === $result) { + return $this->success('操作成功', [], 1, 1); + } + return $this->fail(AdminLogic::getError()); + } + + + /** + * @notes 查看管理员详情 + * @author 乔峰 + * @date 2021/12/29 11:07 + */ + public function detail() + { + $params = (new AdminValidate())->goCheck('detail'); + $result = AdminLogic::detail($params); + return $this->data($result); + } + + + /** + * @notes 获取当前管理员信息 + * @author 乔峰 + * @date 2021/12/31 10:53 + */ + public function mySelf() + { + $result = AdminLogic::detail(['id' => $this->adminId], 'auth'); + return $this->data($result); + } + + + /** + * @notes 编辑超级管理员信息 + * @author 乔峰 + * @date 2022/4/8 17:54 + */ + public function editSelf() + { + $params = (new editSelfValidate())->post()->goCheck('', ['admin_id' => $this->adminId]); + $result = AdminLogic::editSelf($params); + return $this->success('操作成功', [], 1, 1); + } + +} \ No newline at end of file diff --git a/app/admin/controller/auth/MenuController.php b/app/admin/controller/auth/MenuController.php new file mode 100755 index 0000000..a923a3f --- /dev/null +++ b/app/admin/controller/auth/MenuController.php @@ -0,0 +1,134 @@ +adminId); + return $this->data($result); + } + + + /** + * @notes 获取菜单列表 + * @author 乔峰 + * @date 2022/6/29 17:23 + */ + public function lists() + { + return $this->dataLists(new MenuLists()); + } + + + /** + * @notes 菜单详情 + * @author 乔峰 + * @date 2022/6/30 10:07 + */ + public function detail() + { + $params = (new MenuValidate())->goCheck('detail'); + return $this->data(MenuLogic::detail($params)); + } + + + /** + * @notes 添加菜单 + * @author 乔峰 + * @date 2022/6/30 10:07 + */ + public function add() + { + $params = (new MenuValidate())->post()->goCheck('add'); + MenuLogic::add($params); + return $this->success('操作成功', [], 1, 1); + } + + + /** + * @notes 编辑菜单 + * @author 乔峰 + * @date 2022/6/30 10:07 + */ + public function edit() + { + $params = (new MenuValidate())->post()->goCheck('edit'); + MenuLogic::edit($params); + return $this->success('操作成功', [], 1, 1); + } + + + /** + * @notes 删除菜单 + * @author 乔峰 + * @date 2022/6/30 10:07 + */ + public function delete() + { + $params = (new MenuValidate())->post()->goCheck('delete'); + MenuLogic::delete($params); + return $this->success('操作成功', [], 1, 1); + } + + + /** + * @notes 更新状态 + * @author 乔峰 + * @date 2022/7/6 17:04 + */ + public function updateStatus() + { + $params = (new MenuValidate())->post()->goCheck('status'); + MenuLogic::updateStatus($params); + return $this->success('操作成功', [], 1, 1); + } + + + /** + * @notes 获取菜单数据 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/10/13 11:03 + */ + public function all() + { + $result = MenuLogic::getAllData(); + return $this->data($result); + } + + +} \ No newline at end of file diff --git a/app/admin/controller/auth/RoleController.php b/app/admin/controller/auth/RoleController.php new file mode 100755 index 0000000..55e4868 --- /dev/null +++ b/app/admin/controller/auth/RoleController.php @@ -0,0 +1,118 @@ +dataLists(new RoleLists()); + } + + + /** + * @notes 添加权限 + * @author 乔峰 + * @date 2021/12/29 11:49 + */ + public function add() + { + $params = (new RoleValidate())->post()->goCheck('add'); + $res = RoleLogic::add($params); + if (true === $res) { + return $this->success('添加成功', [], 1, 1); + } + return $this->fail(RoleLogic::getError()); + } + + + /** + * @notes 编辑角色 + * @author 乔峰 + * @date 2021/12/29 14:18 + */ + public function edit() + { + $params = (new RoleValidate())->post()->goCheck('edit'); + $res = RoleLogic::edit($params); + if (true === $res) { + return $this->success('编辑成功', [], 1, 1); + } + return $this->fail(RoleLogic::getError()); + } + + + /** + * @notes 删除角色 + * @author 乔峰 + * @date 2021/12/29 14:18 + */ + public function delete() + { + $params = (new RoleValidate())->post()->goCheck('del'); + RoleLogic::delete($params['id']); + return $this->success('删除成功', [], 1, 1); + } + + + /** + * @notes 查看角色详情 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2021/12/29 14:18 + */ + public function detail() + { + $params = (new RoleValidate())->goCheck('detail'); + $detail = RoleLogic::detail($params['id']); + return $this->data($detail); + } + + + /** + * @notes 获取角色数据 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/10/13 10:39 + */ + public function all() + { + $result = RoleLogic::getAllData(); + return $this->data($result); + } + +} \ No newline at end of file diff --git a/app/admin/controller/channel/AppSettingController.php b/app/admin/controller/channel/AppSettingController.php new file mode 100755 index 0000000..032038c --- /dev/null +++ b/app/admin/controller/channel/AppSettingController.php @@ -0,0 +1,51 @@ +data($result); + } + + + /** + * @notes App设置 + * @author 乔峰 + * @date 2022/3/29 10:25 + */ + public function setConfig() + { + $params = $this->request->post(); + AppSettingLogic::setConfig($params); + return $this->success('操作成功', [], 1, 1); + } +} \ No newline at end of file diff --git a/app/admin/controller/channel/MnpSettingsController.php b/app/admin/controller/channel/MnpSettingsController.php new file mode 100755 index 0000000..603ab6d --- /dev/null +++ b/app/admin/controller/channel/MnpSettingsController.php @@ -0,0 +1,50 @@ +getConfig(); + return $this->data($result); + } + + /** + * @notes 设置小程序配置 + * @author ljj + * @date 2022/2/16 9:51 上午 + */ + public function setConfig() + { + $params = (new MnpSettingsValidate())->post()->goCheck(); + (new MnpSettingsLogic())->setConfig($params); + return $this->success('操作成功', [], 1, 1); + } +} \ No newline at end of file diff --git a/app/admin/controller/channel/OfficialAccountMenuController.php b/app/admin/controller/channel/OfficialAccountMenuController.php new file mode 100755 index 0000000..075c543 --- /dev/null +++ b/app/admin/controller/channel/OfficialAccountMenuController.php @@ -0,0 +1,71 @@ +request->post(); + $result = OfficialAccountMenuLogic::save($params); + if(false === $result) { + return $this->fail(OfficialAccountMenuLogic::getError()); + } + return $this->success('保存成功',[],1,1); + } + + + /** + * @notes 保存发布菜单 + * @author 乔峰 + * @date 2022/3/29 10:42 + */ + public function saveAndPublish() + { + $params = $this->request->post(); + $result = OfficialAccountMenuLogic::saveAndPublish($params); + if($result) { + return $this->success('保存并发布成功',[],1,1); + } + return $this->fail(OfficialAccountMenuLogic::getError()); + } + + + + /** + * @notes 查看菜单详情 + * @author 乔峰 + * @date 2022/3/29 10:42 + */ + public function detail() + { + $result = OfficialAccountMenuLogic::detail(); + return $this->data($result); + } +} \ No newline at end of file diff --git a/app/admin/controller/channel/OfficialAccountReplyController.php b/app/admin/controller/channel/OfficialAccountReplyController.php new file mode 100755 index 0000000..a94775f --- /dev/null +++ b/app/admin/controller/channel/OfficialAccountReplyController.php @@ -0,0 +1,138 @@ +dataLists(new OfficialAccountReplyLists()); + } + + + /** + * @notes 添加回复(关注/关键词/默认) + * @author 乔峰 + * @date 2022/3/29 10:58 + */ + public function add() + { + $params = (new OfficialAccountReplyValidate())->post()->goCheck('add'); + $result = OfficialAccountReplyLogic::add($params); + if ($result) { + return $this->success('操作成功', [], 1, 1); + } + return $this->fail(OfficialAccountReplyLogic::getError()); + } + + + /** + * @notes 查看回复详情 + * @author 乔峰 + * @date 2022/3/29 10:58 + */ + public function detail() + { + $params = (new OfficialAccountReplyValidate())->goCheck('detail'); + $result = OfficialAccountReplyLogic::detail($params); + return $this->data($result); + } + + + /** + * @notes 编辑回复(关注/关键词/默认) + * @author 乔峰 + * @date 2022/3/29 10:58 + */ + public function edit() + { + $params = (new OfficialAccountReplyValidate())->post()->goCheck('edit'); + $result = OfficialAccountReplyLogic::edit($params); + if ($result) { + return $this->success('操作成功', [], 1, 1); + } + return $this->fail(OfficialAccountReplyLogic::getError()); + } + + + /** + * @notes 删除回复(关注/关键词/默认) + * @author 乔峰 + * @date 2022/3/29 10:59 + */ + public function delete() + { + $params = (new OfficialAccountReplyValidate())->post()->goCheck('delete'); + OfficialAccountReplyLogic::delete($params); + return $this->success('操作成功', [], 1, 1); + } + + + /** + * @notes 更新排序 + * @author 乔峰 + * @date 2022/3/29 10:59 + */ + public function sort() + { + $params = (new OfficialAccountReplyValidate())->post()->goCheck('sort'); + OfficialAccountReplyLogic::sort($params); + return $this->success('操作成功', [], 1, 1); + } + + + /** + * @notes 更新状态 + * @author 乔峰 + * @date 2022/3/29 10:59 + */ + public function status() + { + $params = (new OfficialAccountReplyValidate())->post()->goCheck('status'); + OfficialAccountReplyLogic::status($params); + return $this->success('操作成功', [], 1, 1); + } + + + /** + * @notes 微信公众号回调 + * @throws \ReflectionException + * @author 乔峰 + * @date 2022/3/29 10:59 + */ + public function index() + { + OfficialAccountReplyLogic::index(); + } +} \ No newline at end of file diff --git a/app/admin/controller/channel/OfficialAccountSettingController.php b/app/admin/controller/channel/OfficialAccountSettingController.php new file mode 100755 index 0000000..4f7f50e --- /dev/null +++ b/app/admin/controller/channel/OfficialAccountSettingController.php @@ -0,0 +1,50 @@ +getConfig(); + return $this->data($result); + } + + /** + * @notes 设置公众号配置 + * @author ljj + * @date 2022/2/16 10:09 上午 + */ + public function setConfig() + { + $params = (new OfficialAccountSettingValidate())->post()->goCheck(); + (new OfficialAccountSettingLogic())->setConfig($params); + return $this->success('操作成功',[],1,1); + } +} \ No newline at end of file diff --git a/app/admin/controller/channel/OpenSettingController.php b/app/admin/controller/channel/OpenSettingController.php new file mode 100755 index 0000000..e34af0d --- /dev/null +++ b/app/admin/controller/channel/OpenSettingController.php @@ -0,0 +1,52 @@ +data($result); + } + + + /** + * @notes 微信开放平台设置 + * @author 乔峰 + * @date 2022/3/29 11:03 + */ + public function setConfig() + { + $params = (new OpenSettingValidate())->post()->goCheck(); + OpenSettingLogic::setConfig($params); + return $this->success('操作成功', [], 1, 1); + } +} \ No newline at end of file diff --git a/app/admin/controller/channel/WebPageSettingController.php b/app/admin/controller/channel/WebPageSettingController.php new file mode 100755 index 0000000..b4f7162 --- /dev/null +++ b/app/admin/controller/channel/WebPageSettingController.php @@ -0,0 +1,52 @@ +data($result); + } + + + /** + * @notes H5设置 + * @author 乔峰 + * @date 2022/3/29 10:36 + */ + public function setConfig() + { + $params = (new WebPageSettingValidate())->post()->goCheck(); + WebPageSettingLogic::setConfig($params); + return $this->success('操作成功', [], 1, 1); + } +} \ No newline at end of file diff --git a/app/admin/controller/crontab/CrontabController.php b/app/admin/controller/crontab/CrontabController.php new file mode 100755 index 0000000..cf20eb4 --- /dev/null +++ b/app/admin/controller/crontab/CrontabController.php @@ -0,0 +1,128 @@ +dataLists(new CrontabLists()); + } + + + /** + * @notes 添加定时任务 + * @author 乔峰 + * @date 2022/3/29 14:27 + */ + public function add() + { + $params = (new CrontabValidate())->post()->goCheck('add'); + $result = CrontabLogic::add($params); + if($result) { + return $this->success('添加成功', [], 1, 1); + } + return $this->fail(CrontabLogic::getError()); + } + + + /** + * @notes 查看定时任务详情 + * @author 乔峰 + * @date 2022/3/29 14:27 + */ + public function detail() + { + $params = (new CrontabValidate())->goCheck('detail'); + $result = CrontabLogic::detail($params); + return $this->data($result); + } + + + /** + * @notes 编辑定时任务 + * @author 乔峰 + * @date 2022/3/29 14:27 + */ + public function edit() + { + $params = (new CrontabValidate())->post()->goCheck('edit'); + $result = CrontabLogic::edit($params); + if($result) { + return $this->success('编辑成功', [], 1, 1); + } + return $this->fail(CrontabLogic::getError()); + } + + + /** + * @notes 删除定时任务 + * @author 乔峰 + * @date 2022/3/29 14:27 + */ + public function delete() + { + $params = (new CrontabValidate())->post()->goCheck('delete'); + $result = CrontabLogic::delete($params); + if($result) { + return $this->success('删除成功', [], 1, 1); + } + return $this->fail('删除失败'); + } + + + /** + * @notes 操作定时任务 + * @author 乔峰 + * @date 2022/3/29 14:28 + */ + public function operate() + { + $params = (new CrontabValidate())->post()->goCheck('operate'); + $result = CrontabLogic::operate($params); + if($result) { + return $this->success('操作成功', [], 1, 1); + } + return $this->fail(CrontabLogic::getError()); + } + + + /** + * @notes 获取规则执行时间 + * @author 乔峰 + * @date 2022/3/29 14:28 + */ + public function expression() + { + $params = (new CrontabValidate())->goCheck('expression'); + $result = CrontabLogic::expression($params); + return $this->data($result); + } +} \ No newline at end of file diff --git a/app/admin/controller/dept/DeptController.php b/app/admin/controller/dept/DeptController.php new file mode 100755 index 0000000..6d9c727 --- /dev/null +++ b/app/admin/controller/dept/DeptController.php @@ -0,0 +1,127 @@ +request->get(); + $result = DeptLogic::lists($params); + return $this->success('',$result); + } + + + /** + * @notes 上级部门 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/5/26 18:36 + */ + public function leaderDept() + { + $result = DeptLogic::leaderDept(); + return $this->success('',$result); + } + + + /** + * @notes 添加部门 + * @author 乔峰 + * @date 2022/5/25 18:40 + */ + public function add() + { + $params = (new DeptValidate())->post()->goCheck('add'); + DeptLogic::add($params); + return $this->success('添加成功', [], 1, 1); + } + + + /** + * @notes 编辑部门 + * @author 乔峰 + * @date 2022/5/25 18:41 + */ + public function edit() + { + $params = (new DeptValidate())->post()->goCheck('edit'); + $result = DeptLogic::edit($params); + if (true === $result) { + return $this->success('编辑成功', [], 1, 1); + } + return $this->fail(DeptLogic::getError()); + } + + + /** + * @notes 删除部门 + * @author 乔峰 + * @date 2022/5/25 18:41 + */ + public function delete() + { + $params = (new DeptValidate())->post()->goCheck('delete'); + DeptLogic::delete($params); + return $this->success('删除成功', [], 1, 1); + } + + + /** + * @notes 获取部门详情 + * @author 乔峰 + * @date 2022/5/25 18:41 + */ + public function detail() + { + $params = (new DeptValidate())->goCheck('detail'); + $result = DeptLogic::detail($params); + return $this->data($result); + } + + + /** + * @notes 获取部门数据 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/10/13 10:28 + */ + public function all() + { + $result = DeptLogic::getAllData(); + return $this->data($result); + } + + +} \ No newline at end of file diff --git a/app/admin/controller/dept/JobsController.php b/app/admin/controller/dept/JobsController.php new file mode 100755 index 0000000..36ed3f2 --- /dev/null +++ b/app/admin/controller/dept/JobsController.php @@ -0,0 +1,112 @@ +dataLists(new JobsLists()); + } + + + /** + * @notes 添加岗位 + * @author 乔峰 + * @date 2022/5/25 18:40 + */ + public function add() + { + $params = (new JobsValidate())->post()->goCheck('add'); + JobsLogic::add($params); + return $this->success('添加成功', [], 1, 1); + } + + + /** + * @notes 编辑岗位 + * @author 乔峰 + * @date 2022/5/25 18:41 + */ + public function edit() + { + $params = (new JobsValidate())->post()->goCheck('edit'); + $result = JobsLogic::edit($params); + if (true === $result) { + return $this->success('编辑成功', [], 1, 1); + } + return $this->fail(JobsLogic::getError()); + } + + + /** + * @notes 删除岗位 + * @author 乔峰 + * @date 2022/5/25 18:41 + */ + public function delete() + { + $params = (new JobsValidate())->post()->goCheck('delete'); + JobsLogic::delete($params); + return $this->success('删除成功', [], 1, 1); + } + + + /** + * @notes 获取岗位详情 + * @author 乔峰 + * @date 2022/5/25 18:41 + */ + public function detail() + { + $params = (new JobsValidate())->goCheck('detail'); + $result = JobsLogic::detail($params); + return $this->data($result); + } + + + /** + * @notes 获取岗位数据 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/10/13 10:31 + */ + public function all() + { + $result = JobsLogic::getAllData(); + return $this->data($result); + } + + +} \ No newline at end of file diff --git a/app/admin/controller/notice/NoticeController.php b/app/admin/controller/notice/NoticeController.php new file mode 100755 index 0000000..8393ff1 --- /dev/null +++ b/app/admin/controller/notice/NoticeController.php @@ -0,0 +1,67 @@ +dataLists(new NoticeSettingLists()); + } + + + /** + * @notes 查看通知设置详情 + * @author 乔峰 + * @date 2022/3/29 11:18 + */ + public function detail() + { + $params = (new NoticeValidate())->goCheck('detail'); + $result = NoticeLogic::detail($params); + return $this->data($result); + } + + + /** + * @notes 通知设置 + * @author 乔峰 + * @date 2022/3/29 11:18 + */ + public function set() + { + $params = $this->request->post(); + $result = NoticeLogic::set($params); + if ($result) { + return $this->success('设置成功'); + } + return $this->fail(NoticeLogic::getError()); + } +} \ No newline at end of file diff --git a/app/admin/controller/notice/SmsConfigController.php b/app/admin/controller/notice/SmsConfigController.php new file mode 100755 index 0000000..bd74ec9 --- /dev/null +++ b/app/admin/controller/notice/SmsConfigController.php @@ -0,0 +1,66 @@ +data($result); + } + + + /** + * @notes 短信配置 + * @author 乔峰 + * @date 2022/3/29 11:36 + */ + public function setConfig() + { + $params = (new SmsConfigValidate())->post()->goCheck('setConfig'); + SmsConfigLogic::setConfig($params); + return $this->success('操作成功',[],1,1); + } + + + /** + * @notes 查看短信配置详情 + * @author 乔峰 + * @date 2022/3/29 11:36 + */ + public function detail() + { + $params = (new SmsConfigValidate())->goCheck('detail'); + $result = SmsConfigLogic::detail($params); + return $this->data($result); + } + +} \ No newline at end of file diff --git a/app/admin/controller/setting/CustomerServiceController.php b/app/admin/controller/setting/CustomerServiceController.php new file mode 100755 index 0000000..caaf427 --- /dev/null +++ b/app/admin/controller/setting/CustomerServiceController.php @@ -0,0 +1,49 @@ +data($result); + } + + /** + * @notes 设置客服设置 + * @author ljj + * @date 2022/2/15 12:11 下午 + */ + public function setConfig() + { + $params = $this->request->post(); + CustomerServiceLogic::setConfig($params); + return $this->success('设置成功', [], 1, 1); + } +} \ No newline at end of file diff --git a/app/admin/controller/setting/HotSearchController.php b/app/admin/controller/setting/HotSearchController.php new file mode 100755 index 0000000..0b3b620 --- /dev/null +++ b/app/admin/controller/setting/HotSearchController.php @@ -0,0 +1,54 @@ +data($result); + } + + + /** + * @notes 设置热门搜索 + * @author 乔峰 + * @date 2022/9/5 19:00 + */ + public function setConfig() + { + $params = $this->request->post(); + $result = HotSearchLogic::setConfig($params); + if (false === $result) { + return $this->fail(HotSearchLogic::getError() ?: '系统错误'); + } + return $this->success('设置成功', [], 1, 1); + } +} \ No newline at end of file diff --git a/app/admin/controller/setting/StorageController.php b/app/admin/controller/setting/StorageController.php new file mode 100644 index 0000000..a613fd6 --- /dev/null +++ b/app/admin/controller/setting/StorageController.php @@ -0,0 +1,63 @@ +success('获取成功', StorageLogic::lists()); + } + + + /** + * @notes 存储配置信息 + * @author 乔峰 + * @date 2022/4/20 16:19 + */ + public function detail() + { + $param = (new StorageValidate())->get()->goCheck('detail'); + return $this->success('获取成功', StorageLogic::detail($param)); + } + + + /** + * @notes 设置存储参数 + * @author 乔峰 + * @date 2022/4/20 16:19 + */ + public function setup() + { + $params = (new StorageValidate())->post()->goCheck('setup'); + $result = StorageLogic::setup($params); + if (true === $result) { + return $this->success('配置成功', [], 1, 1); + } + return $this->success($result, [], 1, 1); + } + + + /** + * @notes 切换存储引擎 + * @author 乔峰 + * @date 2022/4/20 16:19 + */ + public function change() + { + $params = (new StorageValidate())->post()->goCheck('change'); + StorageLogic::change($params); + return $this->success('切换成功', [], 1, 1); + } +} \ No newline at end of file diff --git a/app/admin/controller/setting/TransactionSettingsController.php b/app/admin/controller/setting/TransactionSettingsController.php new file mode 100755 index 0000000..18ab825 --- /dev/null +++ b/app/admin/controller/setting/TransactionSettingsController.php @@ -0,0 +1,51 @@ +data($result); + } + + /** + * @notes 设置交易设置 + * @author ljj + * @date 2022/2/15 11:50 上午 + */ + public function setConfig() + { + $params = (new TransactionSettingsValidate())->post()->goCheck('setConfig'); + TransactionSettingsLogic::setConfig($params); + return $this->success('操作成功',[],1,1); + } +} \ No newline at end of file diff --git a/app/admin/controller/setting/dict/DictDataController.php b/app/admin/controller/setting/dict/DictDataController.php new file mode 100755 index 0000000..d8f590f --- /dev/null +++ b/app/admin/controller/setting/dict/DictDataController.php @@ -0,0 +1,94 @@ +dataLists(new DictDataLists()); + } + + + /** + * @notes 添加字典数据 + * @author 乔峰 + * @date 2022/6/20 17:13 + */ + public function add() + { + $params = (new DictDataValidate())->post()->goCheck('add'); + DictDataLogic::save($params); + return $this->success('添加成功', [], 1, 1); + } + + + /** + * @notes 编辑字典数据 + * @author 乔峰 + * @date 2022/6/20 17:13 + */ + public function edit() + { + $params = (new DictDataValidate())->post()->goCheck('edit'); + DictDataLogic::save($params); + return $this->success('编辑成功', [], 1, 1); + } + + + /** + * @notes 删除字典数据 + * @author 乔峰 + * @date 2022/6/20 17:13 + */ + public function delete() + { + $params = (new DictDataValidate())->post()->goCheck('id'); + DictDataLogic::delete($params); + return $this->success('删除成功', [], 1, 1); + } + + + /** + * @notes 获取字典详情 + * @author 乔峰 + * @date 2022/6/20 17:14 + */ + public function detail() + { + $params = (new DictDataValidate())->goCheck('id'); + $result = DictDataLogic::detail($params); + return $this->data($result); + } + + +} \ No newline at end of file diff --git a/app/admin/controller/setting/dict/DictTypeController.php b/app/admin/controller/setting/dict/DictTypeController.php new file mode 100755 index 0000000..48b7059 --- /dev/null +++ b/app/admin/controller/setting/dict/DictTypeController.php @@ -0,0 +1,110 @@ +dataLists(new DictTypeLists()); + } + + + /** + * @notes 添加字典类型 + * @author 乔峰 + * @date 2022/6/20 16:24 + */ + public function add() + { + $params = (new DictTypeValidate())->post()->goCheck('add'); + DictTypeLogic::add($params); + return $this->success('添加成功', [], 1, 1); + } + + + /** + * @notes 编辑字典类型 + * @author 乔峰 + * @date 2022/6/20 16:25 + */ + public function edit() + { + $params = (new DictTypeValidate())->post()->goCheck('edit'); + DictTypeLogic::edit($params); + return $this->success('编辑成功', [], 1, 1); + } + + + /** + * @notes 删除字典类型 + * @author 乔峰 + * @date 2022/6/20 16:25 + */ + public function delete() + { + $params = (new DictTypeValidate())->post()->goCheck('delete'); + DictTypeLogic::delete($params); + return $this->success('删除成功', [], 1, 1); + } + + + /** + * @notes 获取字典详情 + * @author 乔峰 + * @date 2022/6/20 16:25 + */ + public function detail() + { + $params = (new DictTypeValidate())->goCheck('detail'); + $result = DictTypeLogic::detail($params); + return $this->data($result); + } + + + /** + * @notes 获取字典类型数据 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/10/13 10:46 + */ + public function all() + { + $result = DictTypeLogic::getAllData(); + return $this->data($result); + } + + +} \ No newline at end of file diff --git a/app/admin/controller/setting/system/CacheController.php b/app/admin/controller/setting/system/CacheController.php new file mode 100755 index 0000000..e13ea29 --- /dev/null +++ b/app/admin/controller/setting/system/CacheController.php @@ -0,0 +1,38 @@ +success('清除成功', [], 1, 1); + } +} \ No newline at end of file diff --git a/app/admin/controller/setting/system/LogController.php b/app/admin/controller/setting/system/LogController.php new file mode 100755 index 0000000..01de22b --- /dev/null +++ b/app/admin/controller/setting/system/LogController.php @@ -0,0 +1,37 @@ +dataLists(new LogLists()); + } +} \ No newline at end of file diff --git a/app/admin/controller/setting/system/SystemController.php b/app/admin/controller/setting/system/SystemController.php new file mode 100755 index 0000000..296480b --- /dev/null +++ b/app/admin/controller/setting/system/SystemController.php @@ -0,0 +1,41 @@ +data($result); + } + + +} \ No newline at end of file diff --git a/app/admin/controller/setting/user/UserController.php b/app/admin/controller/setting/user/UserController.php new file mode 100755 index 0000000..f515aa0 --- /dev/null +++ b/app/admin/controller/setting/user/UserController.php @@ -0,0 +1,80 @@ +getConfig(); + return $this->data($result); + } + + + /** + * @notes 设置用户设置 + * @author 乔峰 + * @date 2022/3/29 10:08 + */ + public function setConfig() + { + $params = (new UserConfigValidate())->post()->goCheck('user'); + (new UserLogic())->setConfig($params); + return $this->success('操作成功', [], 1, 1); + } + + + /** + * @notes 获取注册配置 + * @author 乔峰 + * @date 2022/3/29 10:08 + */ + public function getRegisterConfig() + { + $result = (new UserLogic())->getRegisterConfig(); + return $this->data($result); + } + + + /** + * @notes 设置注册配置 + * @author 乔峰 + * @date 2022/3/29 10:08 + */ + public function setRegisterConfig() + { + $params = (new UserConfigValidate())->post()->goCheck('register'); + (new UserLogic())->setRegisterConfig($params); + return $this->success('操作成功', [], 1, 1); + } + +} \ No newline at end of file diff --git a/app/admin/controller/setting/web/WebSettingController.php b/app/admin/controller/setting/web/WebSettingController.php new file mode 100755 index 0000000..f099251 --- /dev/null +++ b/app/admin/controller/setting/web/WebSettingController.php @@ -0,0 +1,106 @@ +data($result); + } + + + /** + * @notes 设置网站信息 + * @author 乔峰 + * @date 2021/12/28 15:45 + */ + public function setWebsite() + { + $params = (new WebSettingValidate())->post()->goCheck('website'); + WebSettingLogic::setWebsiteInfo($params); + return $this->success('设置成功', [], 1, 1); + } + + + + /** + * @notes 获取备案信息 + * @author 乔峰 + * @date 2021/12/28 16:10 + */ + public function getCopyright() + { + $result = WebSettingLogic::getCopyright(); + return $this->data($result); + } + + + /** + * @notes 设置备案信息 + * @author 乔峰 + * @date 2021/12/28 16:10 + */ + public function setCopyright() + { + $params = $this->request->post(); + $result = WebSettingLogic::setCopyright($params); + if (false === $result) { + return $this->fail(WebSettingLogic::getError() ?: '操作失败'); + } + return $this->success('设置成功', [], 1, 1); + } + + + /** + * @notes 设置政策协议 + * @author ljj + * @date 2022/2/15 11:00 上午 + */ + public function setAgreement() + { + $params = $this->request->post(); + WebSettingLogic::setAgreement($params); + return $this->success('设置成功', [], 1, 1); + } + + + /** + * @notes 获取政策协议 + * @author ljj + * @date 2022/2/15 11:16 上午 + */ + public function getAgreement() + { + $result = WebSettingLogic::getAgreement(); + return $this->data($result); + } +} \ No newline at end of file diff --git a/app/admin/controller/tools/GeneratorController.php b/app/admin/controller/tools/GeneratorController.php new file mode 100644 index 0000000..f46d54e --- /dev/null +++ b/app/admin/controller/tools/GeneratorController.php @@ -0,0 +1,208 @@ +dataLists(new DataTableLists()); + } + + + /** + * @notes 获取已选择的数据表 + * @return \think\response\Json + * @author 段誉 + * @date 2022/6/14 10:57 + */ + public function generateTable() + { + return $this->dataLists(new GenerateTableLists()); + } + + + /** + * @notes 选择数据表 + * @return \think\response\Json + * @author 段誉 + * @date 2022/6/15 10:09 + */ + public function selectTable() + { + $params = (new GenerateTableValidate())->post()->goCheck('select'); + $result = GeneratorLogic::selectTable($params, $this->adminId); + if (true === $result) { + return $this->success('操作成功', [], 1, 1); + } + return $this->fail(GeneratorLogic::getError()); + } + + + /** + * @notes 生成代码 + * @return \think\response\Json + * @author 段誉 + * @date 2022/6/23 19:08 + */ + public function generate() + { + $params = (new GenerateTableValidate())->post()->goCheck('id'); + + $result = GeneratorLogic::generate($params); + if (false === $result) { + return $this->fail(GeneratorLogic::getError()); + } + + return $this->success('操作成功', $result, 1, 1); + } + + + /** + * @notes 下载文件 + * @return \think\response\File|\think\response\Json + * @author 段誉 + * @date 2022/6/24 9:51 + */ + public function download() + { + $params = (new GenerateTableValidate())->goCheck('download'); + $result = GeneratorLogic::download($params['file']); + if (false === $result) { + return $this->fail(GeneratorLogic::getError() ?: '下载失败'); + } + return download($result, 'likeadmin-curd.zip'); + } + + + /** + * @notes 预览代码 + * @return \think\response\Json + * @author 段誉 + * @date 2022/6/23 19:07 + */ + public function preview() + { + $params = (new GenerateTableValidate())->post()->goCheck('id'); + $result = GeneratorLogic::preview($params); + if (false === $result) { + return $this->fail(GeneratorLogic::getError()); + } + return $this->data($result); + } + + + /** + * @notes 同步字段 + * @return \think\response\Json + * @author 段誉 + * @date 2022/6/17 15:22 + */ + public function syncColumn() + { + $params = (new GenerateTableValidate())->post()->goCheck('id'); + $result = GeneratorLogic::syncColumn($params); + if (true === $result) { + return $this->success('操作成功', [], 1, 1); + } + return $this->fail(GeneratorLogic::getError()); + } + + + /** + * @notes 编辑表信息 + * @return \think\response\Json + * @author 段誉 + * @date 2022/6/20 10:44 + */ + public function edit() + { + $params = (new EditTableValidate())->post()->goCheck(); + $result = GeneratorLogic::editTable($params); + if (true === $result) { + return $this->success('操作成功', [], 1, 1); + } + return $this->fail(GeneratorLogic::getError()); + } + + + /** + * @notes 获取已选择的数据表详情 + * @return \think\response\Json + * @author 段誉 + * @date 2022/6/15 19:00 + */ + public function detail() + { + $params = (new GenerateTableValidate())->goCheck('id'); + $result = GeneratorLogic::getTableDetail($params); + return $this->success('', $result); + } + + + /** + * @notes 删除已选择的数据表信息 + * @return \think\response\Json + * @author 段誉 + * @date 2022/6/15 19:00 + */ + public function delete() + { + $params = (new GenerateTableValidate())->post()->goCheck('id'); + $result = GeneratorLogic::deleteTable($params); + if (true === $result) { + return $this->success('操作成功', [], 1, 1); + } + return $this->fail(GeneratorLogic::getError()); + } + + + /** + * @notes 获取模型 + * @return \think\response\Json + * @author 段誉 + * @date 2022/12/14 11:07 + */ + public function getModels() + { + $result = GeneratorLogic::getAllModels(); + return $this->success('', $result, 1, 1); + } + +} + diff --git a/app/admin/controller/user/UserController.php b/app/admin/controller/user/UserController.php new file mode 100644 index 0000000..38c1989 --- /dev/null +++ b/app/admin/controller/user/UserController.php @@ -0,0 +1,49 @@ +dataLists(new UserLists()); + } + + + /** + * @notes 获取用户详情 + * @author 乔峰 + * @date 2022/9/22 16:34 + */ + public function detail() + { + $params = (new UserValidate())->goCheck('detail'); + $detail = UserLogic::detail($params['id']); + return $this->success('', $detail); + } + + + /** + * @notes 编辑用户信息 + * @author 乔峰 + * @date 2022/9/22 16:34 + */ + public function edit() + { + $params = (new UserValidate())->post()->goCheck('setInfo'); + UserLogic::setUserInfo($params); + return $this->success('操作成功', [], 1, 1); + } +} \ No newline at end of file diff --git a/app/admin/lists/BaseAdminDataLists.php b/app/admin/lists/BaseAdminDataLists.php new file mode 100755 index 0000000..b21537b --- /dev/null +++ b/app/admin/lists/BaseAdminDataLists.php @@ -0,0 +1,39 @@ +adminInfo = $this->request->adminInfo; + $this->adminId = $this->request->adminId; + } + + +} \ No newline at end of file diff --git a/app/admin/lists/article/ArticleLists.php b/app/admin/lists/article/ArticleLists.php new file mode 100644 index 0000000..b80a3dd --- /dev/null +++ b/app/admin/lists/article/ArticleLists.php @@ -0,0 +1,83 @@ + ['title'], + '=' => ['cid', 'is_show'] + ]; + } + + /** + * @notes 设置支持排序字段 + * @return array + * @author heshihu + * @date 2022/2/9 15:11 + */ + public function setSortFields(): array + { + return ['create_time' => 'create_time', 'id' => 'id']; + } + + /** + * @notes 设置默认排序 + * @return array + * @author heshihu + * @date 2022/2/9 15:08 + */ + public function setDefaultOrder(): array + { + return ['sort' => 'desc', 'id' => 'desc']; + } + + /** + * @notes 获取管理列表 + * @return array + * @author heshihu + * @date 2022/2/21 17:11 + */ + public function lists(): array + { + $ArticleLists = Article::where($this->searchWhere) + ->append(['cate_name', 'click']) + ->limit($this->limitOffset, $this->limitLength) + ->order($this->sortOrder) + ->select() + ->toArray(); + + return $ArticleLists; + } + + /** + * @notes 获取数量 + * @return int + * @author heshihu + * @date 2022/2/9 15:12 + */ + public function count(): int + { + return Article::where($this->searchWhere)->count(); + } + + public function extend() + { + return []; + } +} \ No newline at end of file diff --git a/app/admin/lists/auth/AdminLists.php b/app/admin/lists/auth/AdminLists.php new file mode 100755 index 0000000..6d6fb9b --- /dev/null +++ b/app/admin/lists/auth/AdminLists.php @@ -0,0 +1,206 @@ + '账号', + 'name' => '名称', + 'role_name' => '角色', + 'dept_name' => '部门', + 'create_time' => '创建时间', + 'login_time' => '最近登录时间', + 'login_ip' => '最近登录IP', + 'disable_desc' => '状态', + ]; + } + + + /** + * @notes 设置导出文件名 + * @return string + * @author 乔峰 + * @date 2021/12/29 10:08 + */ + public function setFileName(): string + { + return '管理员列表'; + } + + + /** + * @notes 设置搜索条件 + * @return \string[][] + * @author 乔峰 + * @date 2021/12/29 10:07 + */ + public function setSearch(): array + { + return [ + '%like%' => ['name', 'account'], + ]; + } + + + /** + * @notes 设置支持排序字段 + * @return string[] + * @author 乔峰 + * @date 2021/12/29 10:07 + * @remark 格式: ['前端传过来的字段名' => '数据库中的字段名']; + */ + public function setSortFields(): array + { + return ['create_time' => 'create_time', 'id' => 'id']; + } + + + /** + * @notes 设置默认排序 + * @return string[] + * @author 乔峰 + * @date 2021/12/29 10:06 + */ + public function setDefaultOrder(): array + { + return ['id' => 'desc']; + } + + /** + * @notes 查询条件 + * @return array + * @author 乔峰 + * @date 2022/11/29 11:33 + */ + public function queryWhere() + { + $where = []; + if (isset($this->params['role_id']) && $this->params['role_id'] != '') { + $adminIds = AdminRole::where('role_id', $this->params['role_id'])->column('admin_id'); + if (!empty($adminIds)) { + $where[] = ['id', 'in', $adminIds]; + } + } + return $where; + } + + + /** + * @notes 获取管理列表 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2021/12/29 10:05 + */ + public function lists(): array + { + $field = [ + 'id', 'name', 'account', 'create_time', 'disable', 'root', + 'login_time', 'login_ip', 'multipoint_login', 'avatar' + ]; + + $adminLists = Admin::field($field) + ->where($this->searchWhere) + ->where($this->queryWhere()) + ->limit($this->limitOffset, $this->limitLength) + ->order($this->sortOrder) + ->append(['role_id', 'dept_id', 'jobs_id', 'disable_desc']) + ->select() + ->toArray(); + + // 角色数组('角色id'=>'角色名称') + $roleLists = SystemRole::column('name', 'id'); + // 部门列表 + $deptLists = Dept::column('name', 'id'); + // 岗位列表 + $jobsLists = Jobs::column('name', 'id'); + + //管理员列表增加角色名称 + foreach ($adminLists as $k => $v) { + $roleName = ''; + if ($v['root'] == 1) { + $roleName = '系统管理员'; + } else { + foreach ($v['role_id'] as $roleId) { + $roleName .= $roleLists[$roleId] ?? ''; + $roleName .= '/'; + } + } + + $deptName = ''; + foreach ($v['dept_id'] as $deptId) { + $deptName .= $deptLists[$deptId] ?? ''; + $deptName .= '/'; + } + + $jobsName = ''; + foreach ($v['jobs_id'] as $jobsId) { + $jobsName .= $jobsLists[$jobsId] ?? ''; + $jobsName .= '/'; + } + + $adminLists[$k]['role_name'] = trim($roleName, '/'); + $adminLists[$k]['dept_name'] = trim($deptName, '/'); + $adminLists[$k]['jobs_name'] = trim($jobsName, '/'); + } + + return $adminLists; + } + + /** + * @notes 获取数量 + * @return int + * @author 乔峰 + * @date 2021/7/13 00:52 + */ + public function count(): int + { + return Admin::where($this->searchWhere) + ->where($this->queryWhere()) + ->count(); + } + + public function extend() + { + return []; + } +} \ No newline at end of file diff --git a/app/admin/lists/auth/MenuLists.php b/app/admin/lists/auth/MenuLists.php new file mode 100755 index 0000000..9f84bc4 --- /dev/null +++ b/app/admin/lists/auth/MenuLists.php @@ -0,0 +1,58 @@ + 'desc', 'id' => 'asc']) + ->select() + ->toArray(); + return linear_to_tree($lists, 'children'); + } + + + /** + * @notes 获取菜单数量 + * @return int + * @author 乔峰 + * @date 2022/6/29 16:41 + */ + public function count(): int + { + return SystemMenu::count(); + } + +} \ No newline at end of file diff --git a/app/admin/lists/auth/RoleLists.php b/app/admin/lists/auth/RoleLists.php new file mode 100755 index 0000000..c63a820 --- /dev/null +++ b/app/admin/lists/auth/RoleLists.php @@ -0,0 +1,93 @@ + '角色名称', + 'desc' => '备注', + 'create_time' => '创建时间' + ]; + } + + /** + * @notes 导出表名 + * @return string + * @author Tab + * @date 2021/9/22 18:52 + */ + public function setFileName(): string + { + return '角色表'; + } + + /** + * @notes 角色列表 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author cjhao + * @date 2021/8/25 18:00 + */ + public function lists(): array + { + $lists = SystemRole::with(['role_menu_index']) + ->field('id,name,desc,sort,create_time') + ->limit($this->limitOffset, $this->limitLength) + ->order(['sort' => 'desc', 'id' => 'desc']) + ->select() + ->toArray(); + + foreach ($lists as $key => $role) { + //使用角色的人数 + $lists[$key]['num'] = AdminRole::where('role_id', $role['id'])->count(); + $menuId = array_column($role['role_menu_index'], 'menu_id'); + $lists[$key]['menu_id'] = $menuId; + unset($lists[$key]['role_menu_index']); + } + + return $lists; + } + + /** + * @notes 总记录数 + * @return int + * @author Tab + * @date 2021/7/13 11:26 + */ + public function count(): int + { + return SystemRole::count(); + } +} \ No newline at end of file diff --git a/app/admin/lists/channel/OfficialAccountReplyLists.php b/app/admin/lists/channel/OfficialAccountReplyLists.php new file mode 100755 index 0000000..89e0028 --- /dev/null +++ b/app/admin/lists/channel/OfficialAccountReplyLists.php @@ -0,0 +1,80 @@ + ['reply_type'] + ]; + } + + + /** + * @notes 回复列表 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/3/30 15:02 + */ + public function lists(): array + { + $field = 'id,name,keyword,matching_type,content,content_type,status,sort'; + $field .= ',matching_type as matching_type_desc,content_type as content_type_desc,status as status_desc'; + + $lists = OfficialAccountReply::field($field) + ->where($this->searchWhere) + ->order(['sort' => 'desc', 'id' => 'desc']) + ->limit($this->limitOffset, $this->limitLength) + ->select() + ->toArray(); + + return $lists; + } + + + /** + * @notes 回复记录数 + * @return int + * @author 乔峰 + * @date 2022/3/30 15:02 + */ + public function count(): int + { + $count = OfficialAccountReply::where($this->searchWhere)->count(); + + return $count; + } +} \ No newline at end of file diff --git a/app/admin/lists/crontab/CrontabLists.php b/app/admin/lists/crontab/CrontabLists.php new file mode 100755 index 0000000..10d1eb4 --- /dev/null +++ b/app/admin/lists/crontab/CrontabLists.php @@ -0,0 +1,61 @@ +limit($this->limitOffset, $this->limitLength) + ->order('id', 'desc') + ->select() + ->toArray(); + + return $lists; + } + + + /** + * @notes 定时任务数量 + * @return int + * @author 乔峰 + * @date 2022/3/29 14:38 + */ + public function count(): int + { + return Crontab::count(); + } +} \ No newline at end of file diff --git a/app/admin/lists/dept/JobsLists.php b/app/admin/lists/dept/JobsLists.php new file mode 100755 index 0000000..d6b619c --- /dev/null +++ b/app/admin/lists/dept/JobsLists.php @@ -0,0 +1,105 @@ + ['name'], + '=' => ['code', 'status'] + ]; + } + + + /** + * @notes 获取管理列表 + * @return array + * @author heshihu + * @date 2022/2/21 17:11 + */ + public function lists(): array + { + $lists = Jobs::where($this->searchWhere) + ->append(['status_desc']) + ->limit($this->limitOffset, $this->limitLength) + ->order(['sort' => 'desc', 'id' => 'desc']) + ->select() + ->toArray(); + + return $lists; + } + + + /** + * @notes 获取数量 + * @return int + * @author 乔峰 + * @date 2022/5/26 9:48 + */ + public function count(): int + { + return Jobs::where($this->searchWhere)->count(); + } + + + /** + * @notes 导出文件名 + * @return string + * @author 乔峰 + * @date 2022/11/24 16:17 + */ + public function setFileName(): string + { + return '岗位列表'; + } + + + /** + * @notes 导出字段 + * @return string[] + * @author 乔峰 + * @date 2022/11/24 16:17 + */ + public function setExcelFields(): array + { + return [ + 'code' => '岗位编码', + 'name' => '岗位名称', + 'remark' => '备注', + 'status_desc' => '状态', + 'create_time' => '添加时间', + ]; + } + +} \ No newline at end of file diff --git a/app/admin/lists/file/FileCateLists.php b/app/admin/lists/file/FileCateLists.php new file mode 100644 index 0000000..4b08cd4 --- /dev/null +++ b/app/admin/lists/file/FileCateLists.php @@ -0,0 +1,57 @@ + ['type'] + ]; + } + + + /** + * @notes 获取文件分类列表 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2021/12/29 14:24 + */ + public function lists(): array + { + $lists = (new FileCate())->field(['id,pid,type,name']) + ->where($this->searchWhere) + ->select()->toArray(); + + return linear_to_tree($lists, 'children'); + } + + + /** + * @notes 获取文件分类数量 + * @return int + * @author 乔峰 + * @date 2021/12/29 14:24 + */ + public function count(): int + { + return (new FileCate())->where($this->searchWhere)->count(); + } +} \ No newline at end of file diff --git a/app/admin/lists/file/FileLists.php b/app/admin/lists/file/FileLists.php new file mode 100644 index 0000000..76bd3bb --- /dev/null +++ b/app/admin/lists/file/FileLists.php @@ -0,0 +1,70 @@ + ['type', 'cid'], + '%like%' => ['name'] + ]; + } + + + /** + * @notes 获取文件列表 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2021/12/29 14:27 + */ + public function lists(): array + { + $lists = (new File())->field(['id,cid,type,name,uri,create_time']) + ->order('id', 'desc') + ->where($this->searchWhere) + ->where('source', FileEnum::SOURCE_ADMIN) + ->limit($this->limitOffset, $this->limitLength) + ->select() + ->toArray(); + + foreach ($lists as &$item) { + $item['url'] = $item['uri']; + $item['uri'] = FileService::getFileUrl($item['uri']); + } + + return $lists; + } + + + /** + * @notes 获取文件数量 + * @return int + * @author 乔峰 + * @date 2021/12/29 14:29 + */ + public function count(): int + { + return (new File())->where($this->searchWhere) + ->where('source', FileEnum::SOURCE_ADMIN) + ->count(); + } +} \ No newline at end of file diff --git a/app/admin/lists/notice/NoticeSettingLists.php b/app/admin/lists/notice/NoticeSettingLists.php new file mode 100755 index 0000000..63bee9d --- /dev/null +++ b/app/admin/lists/notice/NoticeSettingLists.php @@ -0,0 +1,71 @@ + ['recipient', 'type'] + ]; + } + + /** + * @notes 通知设置列表 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author ljj + * @date 2022/2/16 3:18 下午 + */ + public function lists(): array + { + $lists = (new NoticeSetting())->field('id,scene_name,sms_notice,type') + ->append(['sms_status_desc','type_desc']) + ->where($this->searchWhere) + ->select() + ->toArray(); + + return $lists; + } + + /** + * @notes 通知设置数量 + * @return int + * @author ljj + * @date 2022/2/16 3:18 下午 + */ + public function count(): int + { + return (new NoticeSetting())->where($this->searchWhere)->count(); + } +} \ No newline at end of file diff --git a/app/admin/lists/setting/dict/DictDataLists.php b/app/admin/lists/setting/dict/DictDataLists.php new file mode 100755 index 0000000..4d8c6a7 --- /dev/null +++ b/app/admin/lists/setting/dict/DictDataLists.php @@ -0,0 +1,76 @@ + ['name', 'type_value'], + '=' => ['status', 'type_id'] + ]; + } + + + /** + * @notes 获取列表 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/6/20 16:35 + */ + public function lists(): array + { + return DictData::where($this->searchWhere) + ->append(['status_desc']) + ->limit($this->limitOffset, $this->limitLength) + ->order(['sort' => 'desc', 'id' => 'desc']) + ->select() + ->toArray(); + } + + + /** + * @notes 获取数量 + * @return int + * @author 乔峰 + * @date 2022/6/20 16:35 + */ + public function count(): int + { + return DictData::where($this->searchWhere)->count(); + } + +} \ No newline at end of file diff --git a/app/admin/lists/setting/dict/DictTypeLists.php b/app/admin/lists/setting/dict/DictTypeLists.php new file mode 100755 index 0000000..bf07c95 --- /dev/null +++ b/app/admin/lists/setting/dict/DictTypeLists.php @@ -0,0 +1,76 @@ + ['name', 'type'], + '=' => ['status'] + ]; + } + + + /** + * @notes 获取列表 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/6/20 15:54 + */ + public function lists(): array + { + return DictType::where($this->searchWhere) + ->limit($this->limitOffset, $this->limitLength) + ->append(['status_desc']) + ->order(['id' => 'desc']) + ->select() + ->toArray(); + } + + + /** + * @notes 获取数量 + * @return int + * @author 乔峰 + * @date 2022/6/20 15:54 + */ + public function count(): int + { + return DictType::where($this->searchWhere)->count(); + } + +} \ No newline at end of file diff --git a/app/admin/lists/setting/system/LogLists.php b/app/admin/lists/setting/system/LogLists.php new file mode 100755 index 0000000..94d5994 --- /dev/null +++ b/app/admin/lists/setting/system/LogLists.php @@ -0,0 +1,108 @@ + ['admin_name','url','ip','type'], + 'between_time' => 'create_time', + ]; + } + + /** + * @notes 查看系统日志列表 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author ljj + * @date 2021/8/3 4:21 下午 + */ + public function lists(): array + { + $lists = OperationLog::field('id,action,admin_name,admin_id,url,type,params,ip,create_time') + ->where($this->searchWhere) + ->limit($this->limitOffset, $this->limitLength) + ->order('id','desc') + ->select() + ->toArray(); + + return $lists; + } + + /** + * @notes 查看系统日志总数 + * @return int + * @author ljj + * @date 2021/8/3 4:23 下午 + */ + public function count(): int + { + return OperationLog::where($this->searchWhere)->count(); + } + + /** + * @notes 设置导出字段 + * @return string[] + * @author ljj + * @date 2021/8/3 4:48 下午 + */ + public function setExcelFields(): array + { + return [ + // '数据库字段名(支持别名) => 'Excel表字段名' + 'id' => '记录ID', + 'action' => '操作', + 'admin_name' => '管理员', + 'admin_id' => '管理员ID', + 'url' => '访问链接', + 'type' => '访问方式', + 'params' => '访问参数', + 'ip' => '来源IP', + 'create_time' => '日志时间', + ]; + } + + /** + * @notes 设置默认表名 + * @return string + * @author ljj + * @date 2021/8/3 4:48 下午 + */ + public function setFileName(): string + { + return '系统日志'; + } +} \ No newline at end of file diff --git a/app/admin/lists/tools/DataTableLists.php b/app/admin/lists/tools/DataTableLists.php new file mode 100644 index 0000000..406ff66 --- /dev/null +++ b/app/admin/lists/tools/DataTableLists.php @@ -0,0 +1,74 @@ +params['name'])) { + $sql .= "AND name LIKE '%" . $this->params['name'] . "%'"; + } + if (!empty($this->params['comment'])) { + $sql .= "AND comment LIKE '%" . $this->params['comment'] . "%'"; + } + return Db::query($sql); + } + + + /** + * @notes 处理列表 + * @return array + * @author 段誉 + * @date 2022/6/13 18:54 + */ + public function lists(): array + { + $lists = array_map("array_change_key_case", $this->queryResult()); + $offset = max(0, ($this->pageNo - 1) * $this->pageSize); + $lists = array_slice($lists, $offset, $this->pageSize, true); + return array_values($lists); + } + + + /** + * @notes 获取数量 + * @return int + * @author 段誉 + * @date 2022/6/13 18:54 + */ + public function count(): int + { + return count($this->queryResult()); + } + +} \ No newline at end of file diff --git a/app/admin/lists/tools/GenerateTableLists.php b/app/admin/lists/tools/GenerateTableLists.php new file mode 100644 index 0000000..d7fd97b --- /dev/null +++ b/app/admin/lists/tools/GenerateTableLists.php @@ -0,0 +1,75 @@ + ['table_name', 'table_comment'] + ]; + } + + + /** + * @notes 查询列表 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 段誉 + * @date 2022/6/14 10:55 + */ + public function lists(): array + { + return GenerateTable::where($this->searchWhere) + ->order(['id' => 'desc']) + ->append(['template_type_desc']) + ->limit($this->limitOffset, $this->limitLength) + ->select() + ->toArray(); + } + + + /** + * @notes 获取数量 + * @return int + * @author 段誉 + * @date 2022/6/14 10:55 + */ + public function count(): int + { + return GenerateTable::count(); + } + +} \ No newline at end of file diff --git a/app/admin/lists/user/UserLists.php b/app/admin/lists/user/UserLists.php new file mode 100644 index 0000000..688ef6f --- /dev/null +++ b/app/admin/lists/user/UserLists.php @@ -0,0 +1,94 @@ +params), $allowSearch); + } + + + /** + * @notes 获取用户列表 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/9/22 15:50 + */ + public function lists(): array + { + $field = "id,sn,nickname,sex,avatar,account,mobile,channel,create_time"; + $lists = User::withSearch($this->setSearch(), $this->params) + ->limit($this->limitOffset, $this->limitLength) + ->field($field) + ->order('id desc') + ->select()->toArray(); + + foreach ($lists as &$item) { + $item['channel'] = UserTerminalEnum::getTermInalDesc($item['channel']); + } + + return $lists; + } + + + /** + * @notes 获取数量 + * @return int + * @author 乔峰 + * @date 2022/9/22 15:51 + */ + public function count(): int + { + return User::withSearch($this->setSearch(), $this->params)->count(); + } + + + /** + * @notes 导出文件名 + * @return string + * @author 乔峰 + * @date 2022/11/24 16:17 + */ + public function setFileName(): string + { + return '用户列表'; + } + + + /** + * @notes 导出字段 + * @return string[] + * @author 乔峰 + * @date 2022/11/24 16:17 + */ + public function setExcelFields(): array + { + return [ + 'sn' => '用户编号', + 'nickname' => '用户昵称', + 'account' => '账号', + 'mobile' => '手机号码', + 'channel' => '注册来源', + 'create_time' => '注册时间', + ]; + } +} \ No newline at end of file diff --git a/app/admin/logic/ConfigLogic.php b/app/admin/logic/ConfigLogic.php new file mode 100755 index 0000000..d9ef9f8 --- /dev/null +++ b/app/admin/logic/ConfigLogic.php @@ -0,0 +1,92 @@ + FileService::getFileUrl(), + + // 网站名称 + 'web_name' => ConfigService::get('website', 'name'), + // 网站图标 + 'web_favicon' => FileService::getFileUrl(ConfigService::get('website', 'web_favicon')), + // 网站logo + 'web_logo' => FileService::getFileUrl(ConfigService::get('website', 'web_logo')), + // 登录页 + 'login_image' => FileService::getFileUrl(ConfigService::get('website', 'login_image')), + + // 版权信息 + 'copyright_config' => ConfigService::get('copyright', 'config', []), + ]; + return $config; + } + + + /** + * @notes 根据类型获取字典类型 + * @param $type + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/9/27 19:09 + */ + public static function getDictByType($type) + { + if (!is_string($type)) { + return []; + } + + $type = explode(',', $type); + $lists = DictData::whereIn('type_value', $type)->select()->toArray(); + + if (empty($lists)) { + return []; + } + + $result = []; + foreach ($type as $item) { + foreach ($lists as $dict) { + if ($dict['type_value'] == $item) { + $result[$item][] = $dict; + } + } + } + return $result; + } + + + +} \ No newline at end of file diff --git a/app/admin/logic/FileLogic.php b/app/admin/logic/FileLogic.php new file mode 100755 index 0000000..6ae53ae --- /dev/null +++ b/app/admin/logic/FileLogic.php @@ -0,0 +1,119 @@ +whereIn('id', $params['ids']) + ->update([ + 'cid' => $params['cid'], + 'update_time' => time() + ]); + } + + /** + * @notes 重命名文件 + * @param $params + * @author 乔峰 + * @date 2021/7/29 17:16 + */ + public static function rename($params) + { + (new File())->where('id', $params['id']) + ->update([ + 'name' => $params['name'], + 'update_time' => time() + ]); + } + + /** + * @notes 批量删除文件 + * @param $params + * @author 乔峰 + * @date 2021/7/28 15:41 + */ + public static function delete($params) + { + $result = File::whereIn('id', $params['ids'])->select(); + $StorageDriver = new StorageDriver([ + 'default' => ConfigService::get('storage', 'default', 'local'), + 'engine' => ConfigService::get('storage') ?? ['local'=>[]], + ]); + foreach ($result as $item) { + $StorageDriver->delete($item['uri']); + } + File::destroy($params['ids']); + } + + /** + * @notes 添加文件分类 + * @param $params + * @author 乔峰 + * @date 2021/7/28 11:32 + */ + public static function addCate($params) + { + FileCate::create([ + 'type' => $params['type'], + 'pid' => $params['pid'], + 'name' => $params['name'] + ]); + } + + /** + * @notes 编辑文件分类 + * @param $params + * @author 乔峰 + * @date 2021/7/28 14:03 + */ + public static function editCate($params) + { + FileCate::update([ + 'name' => $params['name'], + 'update_time' => time() + ], ['id' => $params['id']]); + } + + /** + * @notes 删除文件分类 + * @param $params + * @author 乔峰 + * @date 2021/7/28 14:21 + */ + public static function delCate($params) + { + FileCate::destroy($params['id']); + } +} \ No newline at end of file diff --git a/app/admin/logic/LoginLogic.php b/app/admin/logic/LoginLogic.php new file mode 100755 index 0000000..6320957 --- /dev/null +++ b/app/admin/logic/LoginLogic.php @@ -0,0 +1,83 @@ +find(); + + //用户表登录信息更新 + $admin->login_time = $time; + $admin->login_ip = request()->getLocalIp(); + $admin->save(); + //设置token + $adminInfo = AdminTokenService::setToken($admin->id, $params['terminal'], $admin->multipoint_login); + + //返回登录信息 + $avatar = $admin->avatar ? $admin->avatar : Config::get('project.default_image.admin_avatar'); + $avatar = FileService::getFileUrl($avatar); + return [ + 'name' => $adminInfo['name'], + 'avatar' => $avatar, + 'role_name' => $adminInfo['role_name'], + 'token' => $adminInfo['token'], + ]; + } + + + /** + * @notes 退出登录 + * @param $adminInfo + * @return bool + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2021/7/5 14:34 + */ + public function logout($adminInfo) + { + //token不存在,不注销 + if (!isset($adminInfo['token'])) { + return false; + } + //设置token过期 + return AdminTokenService::expireToken($adminInfo['token']); + } +} \ No newline at end of file diff --git a/app/admin/logic/WorkbenchLogic.php b/app/admin/logic/WorkbenchLogic.php new file mode 100755 index 0000000..297ad05 --- /dev/null +++ b/app/admin/logic/WorkbenchLogic.php @@ -0,0 +1,198 @@ + self::versionInfo(), + // 今日数据 + 'today' => self::today(), + // 常用功能 + 'menu' => self::menu(), + // 近15日访客数 + 'visitor' => self::visitor(), + // 服务支持 + 'support' => self::support() + ]; + } + + + /** + * @notes 常用功能 + * @return array[] + * @author 乔峰 + * @date 2021/12/29 16:40 + */ + public static function menu(): array + { + return [ + [ + 'name' => '管理员', + 'image' => FileService::getFileUrl(config('project.default_image.menu_admin')), + 'url' => '/permission/admin' + ], + [ + 'name' => '角色管理', + 'image' => FileService::getFileUrl(config('project.default_image.menu_role')), + 'url' => '/permission/role' + ], + [ + 'name' => '部门管理', + 'image' => FileService::getFileUrl(config('project.default_image.menu_dept')), + 'url' => '/organization/department' + ], + [ + 'name' => '素材中心', + 'image' => FileService::getFileUrl(config('project.default_image.menu_file')), + 'url' => '/material/index' + ], + [ + 'name' => '菜单权限', + 'image' => FileService::getFileUrl(config('project.default_image.menu_auth')), + 'url' => '/permission/menu' + ], + [ + 'name' => '网站信息', + 'image' => FileService::getFileUrl(config('project.default_image.menu_web')), + 'url' => '/setting/website/information' + ], + ]; + } + + + /** + * @notes 版本信息 + * @return array + * @author 乔峰 + * @date 2021/12/29 16:08 + */ + public static function versionInfo(): array + { + return [ + 'version' => config('project.version'), + 'website' => config('project.website.url'), + 'name' => ConfigService::get('website', 'name'), + 'based' => 'vue3.x、ElementUI、MySQL', + 'channel' => [ + 'website' => 'https://gitee.com/MuZJun/gather-admin.git', + 'gitee' => 'https://gitee.com/MuZJun/gather-vue.git', + ] + ]; + } + + + /** + * @notes 今日数据 + * @return int[] + * @author 乔峰 + * @date 2021/12/29 16:15 + */ + public static function today(): array + { + return [ + 'time' => date('Y-m-d H:i:s'), + // 今日销售额 + 'today_sales' => 100, + // 总销售额 + 'total_sales' => 1000, + + // 今日访问量 + 'today_visitor' => 10, + // 总访问量 + 'total_visitor' => 100, + + // 今日新增用户量 + 'today_new_user' => 30, + // 总用户量 + 'total_new_user' => 3000, + + // 订单量 (笔) + 'order_num' => 12, + // 总订单量 + 'order_sum' => 255 + ]; + } + + + /** + * @notes 访问数 + * @return array + * @author 乔峰 + * @date 2021/12/29 16:57 + */ + public static function visitor(): array + { + $num = []; + $date = []; + for ($i = 0; $i < 15; $i++) { + $where_start = strtotime("- " . $i . "day"); + $date[] = date('Y/m/d', $where_start); + $num[$i] = rand(0, 100); + } + + return [ + 'date' => $date, + 'list' => [ + ['name' => '访客数', 'data' => $num] + ] + ]; + } + + + /** + * @notes 服务支持 + * @return array[] + * @author 乔峰 + * @date 2022/7/18 11:18 + */ + public static function support() + { + return [ + [ + 'image' => FileService::getFileUrl(config('project.default_image.qq_group')), + 'title' => '官方公众号', + 'desc' => '关注官方公众号', + ], + [ + 'image' => FileService::getFileUrl(config('project.default_image.customer_service')), + 'title' => '添加企业客服微信', + 'desc' => '想了解更多请添加客服', + ] + ]; + } + +} \ No newline at end of file diff --git a/app/admin/logic/article/ArticleCateLogic.php b/app/admin/logic/article/ArticleCateLogic.php new file mode 100755 index 0000000..e66e423 --- /dev/null +++ b/app/admin/logic/article/ArticleCateLogic.php @@ -0,0 +1,127 @@ + $params['name'], + 'is_show' => $params['is_show'], + 'sort' => $params['sort'] ?? 0 + ]); + } + + + /** + * @notes 编辑资讯分类 + * @param array $params + * @return bool + * @author heshihu + * @date 2022/2/21 17:50 + */ + public static function edit(array $params) : bool + { + try { + ArticleCate::update([ + 'id' => $params['id'], + 'name' => $params['name'], + 'is_show' => $params['is_show'], + 'sort' => $params['sort'] ?? 0 + ]); + return true; + } catch (\Exception $e) { + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 删除资讯分类 + * @param array $params + * @author heshihu + * @date 2022/2/21 17:52 + */ + public static function delete(array $params) + { + ArticleCate::destroy($params['id']); + } + + /** + * @notes 查看资讯分类详情 + * @param $params + * @return array + * @author heshihu + * @date 2022/2/21 17:54 + */ + public static function detail($params) : array + { + return ArticleCate::findOrEmpty($params['id'])->toArray(); + } + + /** + * @notes 更改资讯分类状态 + * @param array $params + * @return bool + * @author heshihu + * @date 2022/2/21 18:04 + */ + public static function updateStatus(array $params) + { + ArticleCate::update([ + 'id' => $params['id'], + 'is_show' => $params['is_show'] + ]); + return true; + } + + + /** + * @notes 文章分类数据 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/10/13 10:53 + */ + public static function getAllData() + { + return ArticleCate::where(['is_show' => YesNoEnum::YES]) + ->order(['sort' => 'desc', 'id' => 'desc']) + ->select() + ->toArray(); + } + +} \ No newline at end of file diff --git a/app/admin/logic/article/ArticleLogic.php b/app/admin/logic/article/ArticleLogic.php new file mode 100755 index 0000000..28331f9 --- /dev/null +++ b/app/admin/logic/article/ArticleLogic.php @@ -0,0 +1,121 @@ + $params['title'], + 'desc' => $params['desc'] ?? '', + 'author' => $params['author'] ?? '', //作者 + 'sort' => $params['sort'] ?? 0, // 排序 + 'abstract' => $params['abstract'], // 文章摘要 + 'click_virtual' => $params['click_virtual'] ?? 0, + 'image' => $params['image'] ? FileService::setFileUrl($params['image']) : '', + 'cid' => $params['cid'], + 'is_show' => $params['is_show'], + 'content' => $params['content'] ?? '', + ]); + } + + + /** + * @notes 编辑资讯 + * @param array $params + * @return bool + * @author heshihu + * @date 2022/2/22 10:12 + */ + public static function edit(array $params) : bool + { + try { + Article::update([ + 'id' => $params['id'], + 'title' => $params['title'], + 'desc' => $params['desc'] ?? '', // 简介 + 'author' => $params['author'] ?? '', //作者 + 'sort' => $params['sort'] ?? 0, // 排序 + 'abstract' => $params['abstract'], // 文章摘要 + 'click_virtual' => $params['click_virtual'] ?? 0, + 'image' => $params['image'] ? FileService::setFileUrl($params['image']) : '', + 'cid' => $params['cid'], + 'is_show' => $params['is_show'], + 'content' => $params['content'] ?? '', + ]); + return true; + } catch (\Exception $e) { + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 删除资讯 + * @param array $params + * @author heshihu + * @date 2022/2/22 10:17 + */ + public static function delete(array $params) + { + Article::destroy($params['id']); + } + + /** + * @notes 查看资讯详情 + * @param $params + * @return array + * @author heshihu + * @date 2022/2/22 10:15 + */ + public static function detail($params) : array + { + return Article::findOrEmpty($params['id'])->toArray(); + } + + /** + * @notes 更改资讯状态 + * @param array $params + * @return bool + * @author heshihu + * @date 2022/2/22 10:18 + */ + public static function updateStatus(array $params) + { + Article::update([ + 'id' => $params['id'], + 'is_show' => $params['is_show'] + ]); + return true; + } +} \ No newline at end of file diff --git a/app/admin/logic/auth/AdminLogic.php b/app/admin/logic/auth/AdminLogic.php new file mode 100755 index 0000000..3e85c82 --- /dev/null +++ b/app/admin/logic/auth/AdminLogic.php @@ -0,0 +1,334 @@ + $params['name'], + 'account' => $params['account'], + 'avatar' => $avatar, + 'password' => $password, + 'create_time' => time(), + 'disable' => $params['disable'], + 'multipoint_login' => $params['multipoint_login'], + ]); + + // 角色 + self::insertRole($admin['id'], $params['role_id'] ?? []); + // 部门 + self::insertDept($admin['id'], $params['dept_id'] ?? []); + // 岗位 + self::insertJobs($admin['id'], $params['jobs_id'] ?? []); + + Db::commit(); + return true; + } catch (\Exception $e) { + Db::rollback(); + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 编辑管理员 + * @param array $params + * @return bool + * @author 乔峰 + * @date 2021/12/29 10:43 + */ + public static function edit(array $params): bool + { + Db::startTrans(); + try { + // 基础信息 + $data = [ + 'id' => $params['id'], + 'name' => $params['name'], + 'account' => $params['account'], + 'disable' => $params['disable'], + 'multipoint_login' => $params['multipoint_login'] + ]; + + // 头像 + $data['avatar'] = !empty($params['avatar']) ? FileService::setFileUrl($params['avatar']) : ''; + + // 密码 + if (!empty($params['password'])) { + $data['password'] = password_hash($params['password'],PASSWORD_DEFAULT); + } + + // 禁用或更换角色后.设置token过期 + $roleId = AdminRole::where('admin_id', $params['id'])->column('role_id'); + $editRole = false; + if (!empty(array_diff_assoc($roleId, $params['role_id']))) { + $editRole = true; + } + + if ($params['disable'] == 1 || $editRole) { + $tokenArr = AdminSession::where('admin_id', $params['id'])->select()->toArray(); + foreach ($tokenArr as $token) { + self::expireToken($token['token']); + } + } + + Admin::update($data); + (new AdminAuthCache($params['id']))->clearAuthCache(); + + // 删除旧的关联信息 + AdminRole::delByUserId($params['id']); + AdminDept::delByUserId($params['id']); + AdminJobs::delByUserId($params['id']); + // 角色 + self::insertRole($params['id'], $params['role_id']); + // 部门 + self::insertDept($params['id'], $params['dept_id'] ?? []); + // 岗位 + self::insertJobs($params['id'], $params['jobs_id'] ?? []); + + Db::commit(); + return true; + } catch (\Exception $e) { + Db::rollback(); + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 删除管理员 + * @param array $params + * @return bool + * @author 乔峰 + * @date 2021/12/29 10:45 + */ + public static function delete(array $params): bool + { + Db::startTrans(); + try { + $admin = Admin::findOrEmpty($params['id']); + if ($admin->root == YesNoEnum::YES) { + throw new \Exception("超级管理员不允许被删除"); + } + Admin::destroy($params['id']); + + //设置token过期 + $tokenArr = AdminSession::where('admin_id', $params['id'])->select()->toArray(); + foreach ($tokenArr as $token) { + self::expireToken($token['token']); + } + (new AdminAuthCache($params['id']))->clearAuthCache(); + + // 删除旧的关联信息 + AdminRole::delByUserId($params['id']); + AdminDept::delByUserId($params['id']); + AdminJobs::delByUserId($params['id']); + + Db::commit(); + return true; + } catch (\Exception $e) { + Db::rollback(); + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 过期token + * @param $token + * @return bool + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2021/12/29 10:46 + */ + public static function expireToken($token): bool + { + $adminSession = AdminSession::where('token', '=', $token) + ->with('admin') + ->find(); + + if (empty($adminSession)) { + return false; + } + + $time = time(); + $adminSession->expire_time = $time; + $adminSession->update_time = $time; + $adminSession->save(); + + return (new AdminTokenCache())->deleteAdminInfo($token); + } + + + /** + * @notes 查看管理员详情 + * @param $params + * @return array + * @author 乔峰 + * @date 2021/12/29 11:07 + */ + public static function detail($params, $action = 'detail'): array + { + $admin = Admin::field([ + 'id', 'account', 'name', 'disable', 'root', + 'multipoint_login', 'avatar', + ])->findOrEmpty($params['id'])->toArray(); + + if ($action == 'detail') { + return $admin; + } + + $result['user'] = $admin; + // 当前管理员角色拥有的菜单 + $result['menu'] = MenuLogic::getMenuByAdminId($params['id']); + // 当前管理员橘色拥有的按钮权限 + $result['permissions'] = AuthLogic::getBtnAuthByRoleId($admin); + return $result; + } + + + /** + * @notes 编辑超级管理员 + * @param $params + * @return Admin + * @author 乔峰 + * @date 2022/4/8 17:54 + */ + public static function editSelf($params) + { + $data = [ + 'id' => $params['admin_id'], + 'name' => $params['name'], + 'avatar' => FileService::setFileUrl($params['avatar']), + ]; + + if (!empty($params['password'])) { + $data['password'] = password_hash($params['password'],PASSWORD_DEFAULT); + } + + return Admin::update($data); + } + + + /** + * @notes 新增角色 + * @param $adminId + * @param $roleIds + * @throws \Exception + * @author 乔峰 + * @date 2022/11/25 14:23 + */ + public static function insertRole($adminId, $roleIds) + { + if (!empty($roleIds)) { + // 角色 + $roleData = []; + foreach ($roleIds as $roleId) { + $roleData[] = [ + 'admin_id' => $adminId, + 'role_id' => $roleId, + ]; + } + (new AdminRole())->saveAll($roleData); + } + } + + + /** + * @notes 新增部门 + * @param $adminId + * @param $deptIds + * @throws \Exception + * @author 乔峰 + * @date 2022/11/25 14:22 + */ + public static function insertDept($adminId, $deptIds) + { + // 部门 + if (!empty($deptIds)) { + $deptData = []; + foreach ($deptIds as $deptId) { + $deptData[] = [ + 'admin_id' => $adminId, + 'dept_id' => $deptId + ]; + } + (new AdminDept())->saveAll($deptData); + } + } + + + /** + * @notes 新增岗位 + * @param $adminId + * @param $jobsIds + * @throws \Exception + * @author 乔峰 + * @date 2022/11/25 14:22 + */ + public static function insertJobs($adminId, $jobsIds) + { + // 岗位 + if (!empty($jobsIds)) { + $jobsData = []; + foreach ($jobsIds as $jobsId) { + $jobsData[] = [ + 'admin_id' => $adminId, + 'jobs_id' => $jobsId + ]; + } + (new AdminJobs())->saveAll($jobsData); + } + } + +} \ No newline at end of file diff --git a/app/admin/logic/auth/AuthLogic.php b/app/admin/logic/auth/AuthLogic.php new file mode 100755 index 0000000..0960012 --- /dev/null +++ b/app/admin/logic/auth/AuthLogic.php @@ -0,0 +1,105 @@ +where([ + ['is_disable', '=', 0], + ['perms', '<>', ''] + ]) + ->column('perms'); + } + + + /** + * @notes 获取当前管理员角色按钮权限 + * @param $roleId + * @return mixed + * @author 乔峰 + * @date 2022/7/1 16:10 + */ + public static function getBtnAuthByRoleId($admin) + { + if ($admin['root']) { + return ['*']; + } + + $menuId = SystemRoleMenu::whereIn('role_id', $admin['role_id']) + ->column('menu_id'); + + $where[] = ['is_disable', '=', 0]; + $where[] = ['perms', '<>', '']; + + $roleAuth = SystemMenu::distinct(true) + ->where('id', 'in', $menuId) + ->where($where) + ->column('perms'); + + $allAuth = SystemMenu::distinct(true) + ->where($where) + ->column('perms'); + + $hasAllAuth = array_diff($allAuth, $roleAuth); + if (empty($hasAllAuth)) { + return ['*']; + } + + return $roleAuth; + } + + + /** + * @notes 获取管理员角色关联的菜单id(菜单,权限) + * @param int $adminId + * @return array + * @author 乔峰 + * @date 2022/7/1 15:56 + */ + public static function getAuthByAdminId(int $adminId): array + { + $roleIds = AdminRole::where('admin_id', $adminId)->column('role_id'); + $menuId = SystemRoleMenu::whereIn('role_id', $roleIds)->column('menu_id'); + + return SystemMenu::distinct(true) + ->where([ + ['is_disable', '=', 0], + ['perms', '<>', ''], + ['id', 'in', array_unique($menuId)], + ]) + ->column('perms'); + } +} \ No newline at end of file diff --git a/app/admin/logic/auth/MenuLogic.php b/app/admin/logic/auth/MenuLogic.php new file mode 100755 index 0000000..6cb00e4 --- /dev/null +++ b/app/admin/logic/auth/MenuLogic.php @@ -0,0 +1,184 @@ +column('menu_id'); + $where[] = ['id', 'in', $roleMenu]; + } + + $menu = SystemMenu::where($where) + ->order(['sort' => 'desc', 'id' => 'asc']) + ->select(); + + return linear_to_tree($menu, 'children'); + } + + + /** + * @notes 添加菜单 + * @param array $params + * @return SystemMenu|\think\Model + * @author 乔峰 + * @date 2022/6/30 10:06 + */ + public static function add(array $params) + { + return SystemMenu::create([ + 'pid' => $params['pid'], + 'type' => $params['type'], + 'name' => $params['name'], + 'icon' => $params['icon'] ?? '', + 'sort' => $params['sort'], + 'perms' => $params['perms'] ?? '', + 'paths' => $params['paths'] ?? '', + 'component' => $params['component'] ?? '', + 'selected' => $params['selected'] ?? '', + 'params' => $params['params'] ?? '', + 'is_cache' => $params['is_cache'], + 'is_show' => $params['is_show'], + 'is_disable' => $params['is_disable'], + ]); + } + + + /** + * @notes 编辑菜单 + * @param array $params + * @return SystemMenu + * @author 乔峰 + * @date 2022/6/30 10:07 + */ + public static function edit(array $params) + { + return SystemMenu::update([ + 'id' => $params['id'], + 'pid' => $params['pid'], + 'type' => $params['type'], + 'name' => $params['name'], + 'icon' => $params['icon'] ?? '', + 'sort' => $params['sort'], + 'perms' => $params['perms'] ?? '', + 'paths' => $params['paths'] ?? '', + 'component' => $params['component'] ?? '', + 'selected' => $params['selected'] ?? '', + 'params' => $params['params'] ?? '', + 'is_cache' => $params['is_cache'], + 'is_show' => $params['is_show'], + 'is_disable' => $params['is_disable'], + ]); + } + + + /** + * @notes 详情 + * @param $params + * @return array + * @author 乔峰 + * @date 2022/6/30 9:54 + */ + public static function detail($params) + { + return SystemMenu::findOrEmpty($params['id'])->toArray(); + } + + + /** + * @notes 删除菜单 + * @param $params + * @author 乔峰 + * @date 2022/6/30 9:47 + */ + public static function delete($params) + { + // 删除菜单 + SystemMenu::destroy($params['id']); + // 删除角色-菜单表中 与该菜单关联的记录 + SystemRoleMenu::where(['menu_id' => $params['id']])->delete(); + } + + + /** + * @notes 更新状态 + * @param array $params + * @return SystemMenu + * @author 乔峰 + * @date 2022/7/6 17:02 + */ + public static function updateStatus(array $params) + { + return SystemMenu::update([ + 'id' => $params['id'], + 'is_disable' => $params['is_disable'] + ]); + } + + + /** + * @notes 全部数据 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/10/13 11:03 + */ + public static function getAllData() + { + $data = SystemMenu::where(['is_disable' => YesNoEnum::NO]) + ->field('id,pid,name') + ->order(['sort' => 'desc', 'id' => 'desc']) + ->select() + ->toArray(); + + return linear_to_tree($data, 'children'); + } + +} \ No newline at end of file diff --git a/app/admin/logic/auth/RoleLogic.php b/app/admin/logic/auth/RoleLogic.php new file mode 100755 index 0000000..8146899 --- /dev/null +++ b/app/admin/logic/auth/RoleLogic.php @@ -0,0 +1,170 @@ + $params['name'], + 'desc' => $params['desc'] ?? '', + 'sort' => $params['sort'] ?? 0, + ]); + + $data = []; + foreach ($menuId as $item) { + if (empty($item)) { + continue; + } + $data[] = [ + 'role_id' => $role['id'], + 'menu_id' => $item, + ]; + } + (new SystemRoleMenu)->insertAll($data); + + Db::commit(); + return true; + } catch (\Exception $e) { + Db::rollback(); + self::$error = $e->getMessage(); + return false; + } + } + + + /** + * @notes 编辑角色 + * @param array $params + * @return bool + * @author 乔峰 + * @date 2021/12/29 14:16 + */ + public static function edit(array $params): bool + { + Db::startTrans(); + try { + $menuId = !empty($params['menu_id']) ? $params['menu_id'] : []; + + SystemRole::update([ + 'id' => $params['id'], + 'name' => $params['name'], + 'desc' => $params['desc'] ?? '', + 'sort' => $params['sort'] ?? 0, + ]); + + if (!empty($menuId)) { + SystemRoleMenu::where(['role_id' => $params['id']])->delete(); + $data = []; + foreach ($menuId as $item) { + $data[] = [ + 'role_id' => $params['id'], + 'menu_id' => $item, + ]; + } + (new SystemRoleMenu)->insertAll($data); + } + + (new AdminAuthCache())->deleteTag(); + + Db::commit(); + return true; + } catch (\Exception $e) { + Db::rollback(); + self::$error = $e->getMessage(); + return false; + } + } + + /** + * @notes 删除角色 + * @param int $id + * @return bool + * @author 乔峰 + * @date 2021/12/29 14:16 + */ + public static function delete(int $id) + { + SystemRole::destroy(['id' => $id]); + (new AdminAuthCache())->deleteTag(); + return true; + } + + + /** + * @notes 角色详情 + * @param int $id + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2021/12/29 14:17 + */ + public static function detail(int $id): array + { + $detail = SystemRole::field('id,name,desc,sort')->find($id); + $authList = $detail->roleMenuIndex()->select()->toArray(); + $menuId = array_column($authList, 'menu_id'); + $detail['menu_id'] = $menuId; + return $detail->toArray(); + } + + + /** + * @notes 角色数据 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/10/13 10:39 + */ + public static function getAllData() + { + return SystemRole::order(['sort' => 'desc', 'id' => 'desc']) + ->select() + ->toArray(); + } + + +} \ No newline at end of file diff --git a/app/admin/logic/channel/AppSettingLogic.php b/app/admin/logic/channel/AppSettingLogic.php new file mode 100755 index 0000000..52e8641 --- /dev/null +++ b/app/admin/logic/channel/AppSettingLogic.php @@ -0,0 +1,56 @@ + ConfigService::get('app', 'ios_download_url', ''), + 'android_download_url' => ConfigService::get('app', 'android_download_url', ''), + 'download_title' => ConfigService::get('app', 'download_title', ''), + ]; + return $config; + } + + + /** + * @notes App设置 + * @param $params + * @author 乔峰 + * @date 2022/3/29 10:26 + */ + public static function setConfig($params) + { + ConfigService::set('app', 'ios_download_url', $params['ios_download_url'] ?? ''); + ConfigService::set('app', 'android_download_url', $params['android_download_url'] ?? ''); + ConfigService::set('app', 'download_title', $params['download_title'] ?? ''); + } +} \ No newline at end of file diff --git a/app/admin/logic/channel/MnpSettingsLogic.php b/app/admin/logic/channel/MnpSettingsLogic.php new file mode 100755 index 0000000..5d85df7 --- /dev/null +++ b/app/admin/logic/channel/MnpSettingsLogic.php @@ -0,0 +1,72 @@ + ConfigService::get('mnp_setting', 'name', ''), + 'original_id' => ConfigService::get('mnp_setting', 'original_id', ''), + 'qr_code' => $qrCode, + 'app_id' => ConfigService::get('mnp_setting', 'app_id', ''), + 'app_secret' => ConfigService::get('mnp_setting', 'app_secret', ''), + 'request_domain' => 'https://'.$domainName, + 'socket_domain' => 'wss://'.$domainName, + 'upload_file_domain' => 'https://'.$domainName, + 'download_file_domain' => 'https://'.$domainName, + 'udp_domain' => 'udp://'.$domainName, + 'business_domain' => $domainName, + ]; + + return $config; + } + + /** + * @notes 设置小程序配置 + * @param $params + * @author ljj + * @date 2022/2/16 9:51 上午 + */ + public function setConfig($params) + { + $qrCode = isset($params['qr_code']) ? FileService::setFileUrl($params['qr_code']) : ''; + + ConfigService::set('mnp_setting','name', $params['name'] ?? ''); + ConfigService::set('mnp_setting','original_id',$params['original_id'] ?? ''); + ConfigService::set('mnp_setting','qr_code',$qrCode); + ConfigService::set('mnp_setting','app_id',$params['app_id']); + ConfigService::set('mnp_setting','app_secret',$params['app_secret']); + } +} \ No newline at end of file diff --git a/app/admin/logic/channel/OfficialAccountMenuLogic.php b/app/admin/logic/channel/OfficialAccountMenuLogic.php new file mode 100755 index 0000000..b4e9d02 --- /dev/null +++ b/app/admin/logic/channel/OfficialAccountMenuLogic.php @@ -0,0 +1,228 @@ +getMessage()); + return false; + } + } + + + /** + * @notes 一级菜单校验 + * @param $menu + * @throws \Exception + * @author 乔峰 + * @date 2022/3/29 10:55 + */ + public static function checkMenu($menu) + { + if (empty($menu) || !is_array($menu)) { + throw new \Exception('请设置正确格式菜单'); + } + + if (count($menu) > 3) { + throw new \Exception('一级菜单超出限制(最多3个)'); + } + + foreach ($menu as $item) { + if (!is_array($item)) { + throw new \Exception('一级菜单项须为数组格式'); + } + + if (empty($item['name'])) { + throw new \Exception('请输入一级菜单名称'); + } + + if (false == $item['has_menu']) { + if (empty($item['type'])) { + throw new \Exception('一级菜单未选择菜单类型'); + } + if (!in_array($item['type'], OfficialAccountEnum::MENU_TYPE)) { + throw new \Exception('一级菜单类型错误'); + } + self::checkType($item); + } + + if (true == $item['has_menu'] && empty($item['sub_button'])) { + throw new \Exception('请配置子菜单'); + } + + self::checkType($item); + + if (!empty($item['sub_button'])) { + self::checkSubButton($item['sub_button']); + } + } + } + + + /** + * @notes 二级菜单校验 + * @param $subButtion + * @throws \Exception + * @author 乔峰 + * @date 2022/3/29 10:55 + */ + public static function checkSubButton($subButtion) + { + if (!is_array($subButtion)) { + throw new \Exception('二级菜单须为数组格式'); + } + + if (count($subButtion) > 5) { + throw new \Exception('二级菜单超出限制(最多5个)'); + } + + foreach ($subButtion as $subItem) { + if (!is_array($subItem)) { + throw new \Exception('二级菜单项须为数组'); + } + + if (empty($subItem['name'])) { + throw new \Exception('请输入二级菜单名称'); + } + + if (empty($subItem['type']) || !in_array($subItem['type'], OfficialAccountEnum::MENU_TYPE)) { + throw new \Exception('二级未选择菜单类型或菜单类型错误'); + } + + self::checkType($subItem); + } + } + + + /** + * @notes 菜单类型校验 + * @param $item + * @throws \Exception + * @author 乔峰 + * @date 2022/3/29 10:55 + */ + public static function checkType($item) + { + switch ($item['type']) { + // 关键字 + case 'click': + if (empty($item['key'])) { + throw new \Exception('请输入关键字'); + } + break; + // 跳转网页链接 + case 'view': + if (empty($item['url'])) { + throw new \Exception('请输入网页链接'); + } + break; + // 小程序 + case 'miniprogram': + if (empty($item['url'])) { + throw new \Exception('请输入网页链接'); + } + if (empty($item['appid'])) { + throw new \Exception('请输入appid'); + } + if (empty($item['pagepath'])) { + throw new \Exception('请输入小程序路径'); + } + break; + } + } + + /** + * @notes 保存发布菜单 + * @param $params + * @return bool + * @throws \GuzzleHttp\Exception\GuzzleException + * @author 乔峰 + * @date 2022/3/29 10:55 + */ + public static function saveAndPublish($params) + { + try { + self::checkMenu($params); + + $officialAccountSetting = (new OfficialAccountSettingLogic())->getConfig(); + if (empty($officialAccountSetting['app_id']) || empty($officialAccountSetting['app_secret'])) { + throw new \Exception('请先配置好微信公众号'); + } + + $app = Factory::officialAccount([ + 'app_id' => $officialAccountSetting['app_id'], + 'secret' => $officialAccountSetting['app_secret'], + 'response_type' => 'array', + ]); + + $result = $app->menu->create($params); + if ($result['errcode'] == 0) { + ConfigService::set('oa_setting', 'menu', $params); + return true; + } + + self::setError('保存发布菜单失败' . json_encode($result)); + return false; + + } catch (\Exception $e) { + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 查看菜单详情 + * @return array|int|mixed|string|null + * @author 乔峰 + * @date 2022/3/29 10:56 + */ + public static function detail() + { + $data = ConfigService::get('oa_setting', 'menu', []); + + if (!empty($data)) { + foreach ($data as &$item) { + $item['has_menu'] = !empty($item['has_menu']); + } + } + + return $data; + } +} \ No newline at end of file diff --git a/app/admin/logic/channel/OfficialAccountReplyLogic.php b/app/admin/logic/channel/OfficialAccountReplyLogic.php new file mode 100755 index 0000000..cc5d8ab --- /dev/null +++ b/app/admin/logic/channel/OfficialAccountReplyLogic.php @@ -0,0 +1,217 @@ + $params['reply_type']])->update(['status' => YesNoEnum::NO]); + } + OfficialAccountReply::create($params); + return true; + } catch (\Exception $e) { + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 查看回复详情 + * @param $params + * @return array + * @author 乔峰 + * @date 2022/3/29 11:00 + */ + public static function detail($params) + { + $field = 'id,name,keyword,reply_type,matching_type,content_type,content,status,sort'; + $field .= ',reply_type as reply_type_desc, matching_type as matching_type_desc, content_type as content_type_desc, status as status_desc'; + return OfficialAccountReply::field($field)->findOrEmpty($params['id'])->toArray(); + } + + + /** + * @notes 编辑回复(关注/关键词/默认) + * @param $params + * @return bool + * @author 乔峰 + * @date 2022/3/29 11:01 + */ + public static function edit($params) + { + try { + // 关键字回复排序值须大于0 + if ($params['reply_type'] == OfficialAccountEnum::REPLY_TYPE_KEYWORD && $params['sort'] <= 0) { + throw new \Exception('排序值须大于0'); + } + if ($params['reply_type'] != OfficialAccountEnum::REPLY_TYPE_KEYWORD && $params['status']) { + // 非关键词回复只能有一条记录处于启用状态,所以将该回复类型下的已有记录置为禁用状态 + OfficialAccountReply::where(['reply_type' => $params['reply_type']])->update(['status' => YesNoEnum::NO]); + } + OfficialAccountReply::update($params); + return true; + } catch (\Exception $e) { + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 删除回复(关注/关键词/默认) + * @param $params + * @author 乔峰 + * @date 2022/3/29 11:01 + */ + public static function delete($params) + { + OfficialAccountReply::destroy($params['id']); + } + + + /** + * @notes 更新排序 + * @param $params + * @author 乔峰 + * @date 2022/3/29 11:01 + */ + public static function sort($params) + { + $params['sort'] = $params['new_sort']; + OfficialAccountReply::update($params); + } + + + /** + * @notes 更新状态 + * @param $params + * @author 乔峰 + * @date 2022/3/29 11:01 + */ + public static function status($params) + { + $reply = OfficialAccountReply::findOrEmpty($params['id']); + $reply->status = !$reply->status; + $reply->save(); + } + + /** + * @notes 微信公众号回调 + * @throws \EasyWeChat\Kernel\Exceptions\BadRequestException + * @throws \EasyWeChat\Kernel\Exceptions\InvalidArgumentException + * @throws \EasyWeChat\Kernel\Exceptions\InvalidConfigException + * @throws \ReflectionException + * @author 乔峰 + * @date 2022/3/29 11:01 + */ + public static function index() + { + // 确认此次GET请求来自微信服务器,原样返回echostr参数内容,接入生效,成为开发者成功 + if (isset($_GET['echostr'])) { + echo $_GET['echostr']; + exit; + } + + $officialAccountSetting = (new OfficialAccountSettingLogic())->getConfig(); + $config = [ + 'app_id' => $officialAccountSetting['app_id'], + 'secret' => $officialAccountSetting['app_secret'], + 'response_type' => 'array', + ]; + $app = Factory::officialAccount($config); + + $app->server->push(function ($message) { + switch ($message['MsgType']) { // 消息类型 + case OfficialAccountEnum::MSG_TYPE_EVENT: // 事件 + switch ($message['Event']) { + case OfficialAccountEnum::EVENT_SUBSCRIBE: // 关注事件 + $reply_content = OfficialAccountReply::where(['reply_type' => OfficialAccountEnum::REPLY_TYPE_FOLLOW, 'status' => YesNoEnum::YES]) + ->value('content'); + + if (empty($reply_content)) { + // 未启用关注回复 或 关注回复内容为空 + $reply_content = OfficialAccountReply::where(['reply_type' => OfficialAccountEnum::REPLY_TYPE_DEFAULT, 'status' => YesNoEnum::YES]) + ->value('content'); + } + if ($reply_content) { + $text = new Text($reply_content); + return $text; + } + break; + } + break; + + case OfficialAccountEnum::MSG_TYPE_TEXT: // 文本 + $reply_list = OfficialAccountReply::where(['reply_type' => OfficialAccountEnum::REPLY_TYPE_KEYWORD, 'status' => YesNoEnum::YES]) + ->order('sort asc') + ->select(); + $reply_content = ''; + foreach ($reply_list as $reply) { + switch ($reply['matching_type']) { + case OfficialAccountEnum::MATCHING_TYPE_FULL: + $reply['keyword'] === $message['Content'] && $reply_content = $reply['content']; + break; + case OfficialAccountEnum::MATCHING_TYPE_FUZZY: + stripos($message['Content'], $reply['keyword']) !== false && $reply_content = $reply['content']; + break; + } + if ($reply_content) { + break; // 得到回复文本,中止循环 + } + } + //消息回复为空的话,找默认回复 + if (empty($reply_content)) { + $reply_content = OfficialAccountReply::where(['reply_type' => OfficialAccountEnum::REPLY_TYPE_DEFAULT, 'status' => YesNoEnum::YES]) + ->value('content'); + } + if ($reply_content) { + $text = new Text($reply_content); + return $text; + } + break; + } + }); + $response = $app->server->serve(); + $response->send(); + } +} \ No newline at end of file diff --git a/app/admin/logic/channel/OfficialAccountSettingLogic.php b/app/admin/logic/channel/OfficialAccountSettingLogic.php new file mode 100755 index 0000000..29d6837 --- /dev/null +++ b/app/admin/logic/channel/OfficialAccountSettingLogic.php @@ -0,0 +1,76 @@ + ConfigService::get('oa_setting', 'name', ''), + 'original_id' => ConfigService::get('oa_setting', 'original_id', ''), + 'qr_code' => $qrCode, + 'app_id' => ConfigService::get('oa_setting', 'app_id', ''), + 'app_secret' => ConfigService::get('oa_setting', 'app_secret', ''), + // url()方法返回Url实例,通过与空字符串连接触发该实例的__toString()方法以得到路由地址 + 'url' => url('admin/wechat.official_account_reply/index', [],'',true).'', + 'token' => ConfigService::get('oa_setting', 'token'), + 'encoding_aes_key' => ConfigService::get('oa_setting', 'encoding_aes_key', ''), + 'encryption_type' => ConfigService::get('oa_setting', 'encryption_type'), + 'business_domain' => $domainName, + 'js_secure_domain' => $domainName, + 'web_auth_domain' => $domainName, + ]; + return $config; + } + + /** + * @notes 设置公众号配置 + * @param $params + * @author ljj + * @date 2022/2/16 10:08 上午 + */ + public function setConfig($params) + { + $qrCode = isset($params['qr_code']) ? FileService::setFileUrl($params['qr_code']) : ''; + + ConfigService::set('oa_setting','name', $params['name'] ?? ''); + ConfigService::set('oa_setting','original_id', $params['original_id'] ?? ''); + ConfigService::set('oa_setting','qr_code', $qrCode); + ConfigService::set('oa_setting','app_id',$params['app_id']); + ConfigService::set('oa_setting','app_secret',$params['app_secret']); + ConfigService::set('oa_setting','token',$params['token'] ?? ''); + ConfigService::set('oa_setting','encoding_aes_key',$params['encoding_aes_key'] ?? ''); + ConfigService::set('oa_setting','encryption_type',$params['encryption_type']); + } +} \ No newline at end of file diff --git a/app/admin/logic/channel/OpenSettingLogic.php b/app/admin/logic/channel/OpenSettingLogic.php new file mode 100755 index 0000000..0fb394c --- /dev/null +++ b/app/admin/logic/channel/OpenSettingLogic.php @@ -0,0 +1,55 @@ + ConfigService::get('open_platform', 'app_id', ''), + 'app_secret' => ConfigService::get('open_platform', 'app_secret', ''), + ]; + + return $config; + } + + + /** + * @notes 微信开放平台设置 + * @param $params + * @author 乔峰 + * @date 2022/3/29 11:03 + */ + public static function setConfig($params) + { + ConfigService::set('open_platform', 'app_id', $params['app_id'] ?? ''); + ConfigService::set('open_platform', 'app_secret', $params['app_secret'] ?? ''); + } +} \ No newline at end of file diff --git a/app/admin/logic/channel/WebPageSettingLogic.php b/app/admin/logic/channel/WebPageSettingLogic.php new file mode 100755 index 0000000..b8243d2 --- /dev/null +++ b/app/admin/logic/channel/WebPageSettingLogic.php @@ -0,0 +1,59 @@ + ConfigService::get('web_page', 'status', 1), + // 关闭后渠道后访问页面 0-空页面 1-自定义链接 + 'page_status' => ConfigService::get('web_page', 'page_status', 0), + // 自定义链接 + 'page_url' => ConfigService::get('web_page', 'page_url', ''), + 'url' => request()->domain() . '/mobile' + ]; + return $config; + } + + + /** + * @notes H5设置 + * @param $params + * @author 乔峰 + * @date 2022/3/29 10:34 + */ + public static function setConfig($params) + { + ConfigService::set('web_page', 'status', $params['status']); + ConfigService::set('web_page', 'page_status', $params['page_status']); + ConfigService::set('web_page', 'page_url', $params['page_url']); + } +} \ No newline at end of file diff --git a/app/admin/logic/crontab/CrontabLogic.php b/app/admin/logic/crontab/CrontabLogic.php new file mode 100755 index 0000000..5a6518c --- /dev/null +++ b/app/admin/logic/crontab/CrontabLogic.php @@ -0,0 +1,169 @@ +getMessage()); + return false; + } + } + + + /** + * @notes 查看定时任务详情 + * @param $params + * @return array + * @author 乔峰 + * @date 2022/3/29 14:41 + */ + public static function detail($params) + { + $field = 'id,name,type,type as type_desc,command,params,status,status as status_desc,expression,remark'; + $crontab = Crontab::field($field)->findOrEmpty($params['id']); + if ($crontab->isEmpty()) { + return []; + } + return $crontab->toArray(); + } + + + /** + * @notes 编辑定时任务 + * @param $params + * @return bool + * @author 乔峰 + * @date 2022/3/29 14:42 + */ + public static function edit($params) + { + try { + $params['remark'] = $params['remark'] ?? ''; + $params['params'] = $params['params'] ?? ''; + + Crontab::update($params); + + return true; + } catch (\Exception $e) { + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 删除定时任务 + * @param $params + * @return bool + * @author 乔峰 + * @date 2022/3/29 14:42 + */ + public static function delete($params) + { + try { + Crontab::destroy($params['id']); + + return true; + } catch (\Exception $e) { + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 操作定时任务 + * @param $params + * @return bool + * @author 乔峰 + * @date 2022/3/29 14:42 + */ + public static function operate($params) + { + try { + $crontab = Crontab::findOrEmpty($params['id']); + if ($crontab->isEmpty()) { + throw new \Exception('定时任务不存在'); + } + switch ($params['operate']) { + case 'start'; + $crontab->status = CrontabEnum::START; + break; + case 'stop': + $crontab->status = CrontabEnum::STOP; + break; + } + $crontab->save(); + + return true; + } catch (\Exception $e) { + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 获取规则执行时间 + * @param $params + * @return array|string + * @author 乔峰 + * @date 2022/3/29 14:42 + */ + public static function expression($params) + { + try { + $cron = new CronExpression($params['expression']); + $result = $cron->getMultipleRunDates(5); + $result = json_decode(json_encode($result), true); + $lists = []; + foreach ($result as $k => $v) { + $lists[$k]['time'] = $k + 1; + $lists[$k]['date'] = str_replace('.000000', '', $v['date']); + } + $lists[] = ['time' => 'x', 'date' => '……']; + return $lists; + } catch (\Exception $e) { + return $e->getMessage(); + } + } +} \ No newline at end of file diff --git a/app/admin/logic/decorate/DecorateDataLogic.php b/app/admin/logic/decorate/DecorateDataLogic.php new file mode 100755 index 0000000..acb9841 --- /dev/null +++ b/app/admin/logic/decorate/DecorateDataLogic.php @@ -0,0 +1,56 @@ + 1]) + ->field($field) + ->order(['id' => 'desc']) + ->limit($limit) + ->append(['click']) + ->hidden(['click_virtual', 'click_actual']) + ->select()->toArray(); + } + + + + + +} \ No newline at end of file diff --git a/app/admin/logic/decorate/DecoratePageLogic.php b/app/admin/logic/decorate/DecoratePageLogic.php new file mode 100755 index 0000000..396e29b --- /dev/null +++ b/app/admin/logic/decorate/DecoratePageLogic.php @@ -0,0 +1,67 @@ +toArray(); + } + + + /** + * @notes 保存装修配置 + * @param $params + * @return bool + * @author 乔峰 + * @date 2022/9/15 9:37 + */ + public static function save($params) + { + $pageData = DecoratePage::where(['id' => $params['id']])->findOrEmpty(); + if ($pageData->isEmpty()) { + self::$error = '信息不存在'; + return false; + } + DecoratePage::update([ + 'id' => $params['id'], + 'type' => $params['type'], + 'data' => $params['data'], + ]); + return true; + } + + + +} \ No newline at end of file diff --git a/app/admin/logic/decorate/DecorateTabbarLogic.php b/app/admin/logic/decorate/DecorateTabbarLogic.php new file mode 100755 index 0000000..f1058ae --- /dev/null +++ b/app/admin/logic/decorate/DecorateTabbarLogic.php @@ -0,0 +1,80 @@ + $style, 'list' => $list]; + } + + + /** + * @notes 底部导航保存 + * @param $params + * @return bool + * @throws \Exception + * @author 乔峰 + * @date 2022/9/7 17:19 + */ + public static function save($params): bool + { + $model = new DecorateTabbar(); + // 删除旧配置数据 + $model->where('id', '>', 0)->delete(); + + // 保存数据 + $tabbars = $params['list'] ?? []; + $data = []; + foreach ($tabbars as $item) { + $data[] = [ + 'name' => $item['name'], + 'selected' => FileService::setFileUrl($item['selected']), + 'unselected' => FileService::setFileUrl($item['unselected']), + 'link' => $item['link'], + ]; + } + $model->saveAll($data); + + if (!empty($params['style'])) { + ConfigService::set('tabbar', 'style', $params['style']); + } + return true; + } + +} \ No newline at end of file diff --git a/app/admin/logic/dept/DeptLogic.php b/app/admin/logic/dept/DeptLogic.php new file mode 100755 index 0000000..d39e42b --- /dev/null +++ b/app/admin/logic/dept/DeptLogic.php @@ -0,0 +1,202 @@ +append(['status_desc']) + ->order(['sort' => 'desc', 'id' => 'desc']) + ->select() + ->toArray(); + + $pid = 0; + if (!empty($lists)) { + $pid = min(array_column($lists, 'pid')); + } + return self::getTree($lists, $pid); + } + + + /** + * @notes 列表树状结构 + * @param $array + * @param int $pid + * @param int $level + * @return array + * @author 乔峰 + * @date 2022/5/30 15:44 + */ + public static function getTree($array, $pid = 0, $level = 0) + { + $list = []; + foreach ($array as $key => $item) { + if ($item['pid'] == $pid) { + $item['level'] = $level; + $item['children'] = self::getTree($array, $item['id'], $level + 1); + $list[] = $item; + } + } + return $list; + } + + + /** + * @notes 上级部门 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/5/26 18:36 + */ + public static function leaderDept() + { + $lists = Dept::field(['id', 'name'])->where(['status' => 1]) + ->order(['sort' => 'desc', 'id' => 'desc']) + ->select() + ->toArray(); + return $lists; + } + + + /** + * @notes 添加部门 + * @param array $params + * @author 乔峰 + * @date 2022/5/25 18:20 + */ + public static function add(array $params) + { + Dept::create([ + 'pid' => $params['pid'], + 'name' => $params['name'], + 'leader' => $params['leader'] ?? '', + 'mobile' => $params['mobile'] ?? '', + 'status' => $params['status'], + 'sort' => $params['sort'] ?? 0 + ]); + } + + + /** + * @notes 编辑部门 + * @param array $params + * @return bool + * @author 乔峰 + * @date 2022/5/25 18:39 + */ + public static function edit(array $params): bool + { + try { + $pid = $params['pid']; + $oldDeptData = Dept::findOrEmpty($params['id']); + if ($oldDeptData['pid'] == 0) { + $pid = 0; + } + + Dept::update([ + 'id' => $params['id'], + 'pid' => $pid, + 'name' => $params['name'], + 'leader' => $params['leader'] ?? '', + 'mobile' => $params['mobile'] ?? '', + 'status' => $params['status'], + 'sort' => $params['sort'] ?? 0 + ]); + return true; + } catch (\Exception $e) { + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 删除部门 + * @param array $params + * @author 乔峰 + * @date 2022/5/25 18:40 + */ + public static function delete(array $params) + { + Dept::destroy($params['id']); + } + + + /** + * @notes 获取部门详情 + * @param $params + * @return array + * @author 乔峰 + * @date 2022/5/25 18:40 + */ + public static function detail($params): array + { + return Dept::findOrEmpty($params['id'])->toArray(); + } + + + /** + * @notes 部门数据 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/10/13 10:19 + */ + public static function getAllData() + { + $data = Dept::where(['status' => YesNoEnum::YES]) + ->order(['sort' => 'desc', 'id' => 'desc']) + ->select() + ->toArray(); + + $pid = min(array_column($data, 'pid')); + return self::getTree($data, $pid); + } + +} \ No newline at end of file diff --git a/app/admin/logic/dept/JobsLogic.php b/app/admin/logic/dept/JobsLogic.php new file mode 100755 index 0000000..f23c055 --- /dev/null +++ b/app/admin/logic/dept/JobsLogic.php @@ -0,0 +1,117 @@ + $params['name'], + 'code' => $params['code'], + 'sort' => $params['sort'] ?? 0, + 'status' => $params['status'], + 'remark' => $params['remark'] ?? '', + ]); + } + + + /** + * @notes 编辑岗位 + * @param array $params + * @return bool + * @author 乔峰 + * @date 2022/5/26 9:58 + */ + public static function edit(array $params) : bool + { + try { + Jobs::update([ + 'id' => $params['id'], + 'name' => $params['name'], + 'code' => $params['code'], + 'sort' => $params['sort'] ?? 0, + 'status' => $params['status'], + 'remark' => $params['remark'] ?? '', + ]); + return true; + } catch (\Exception $e) { + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 删除岗位 + * @param array $params + * @author 乔峰 + * @date 2022/5/26 9:59 + */ + public static function delete(array $params) + { + Jobs::destroy($params['id']); + } + + + /** + * @notes 获取岗位详情 + * @param $params + * @return array + * @author 乔峰 + * @date 2022/5/26 9:59 + */ + public static function detail($params) : array + { + return Jobs::findOrEmpty($params['id'])->toArray(); + } + + + /** + * @notes 岗位数据 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/10/13 10:30 + */ + public static function getAllData() + { + return Jobs::where(['status' => YesNoEnum::YES]) + ->order(['sort' => 'desc', 'id' => 'desc']) + ->select() + ->toArray(); + } + +} \ No newline at end of file diff --git a/app/admin/logic/notice/NoticeLogic.php b/app/admin/logic/notice/NoticeLogic.php new file mode 100755 index 0000000..a6307a6 --- /dev/null +++ b/app/admin/logic/notice/NoticeLogic.php @@ -0,0 +1,225 @@ +findOrEmpty($params['id'])->toArray(); + if (empty($noticeSetting)) { + return []; + } + if (empty($noticeSetting['system_notice'])) { + $noticeSetting['system_notice'] = [ + 'title' => '', + 'content' => '', + 'status' => 0, + ]; + } + $noticeSetting['system_notice']['tips'] = NoticeEnum::getOperationTips(NoticeEnum::SYSTEM, $noticeSetting['scene_id']); + if (empty($noticeSetting['sms_notice'])) { + $noticeSetting['sms_notice'] = [ + 'template_id' => '', + 'content' => '', + 'status' => 0, + ]; + } + $noticeSetting['sms_notice']['tips'] = NoticeEnum::getOperationTips(NoticeEnum::SMS, $noticeSetting['scene_id']); + if (empty($noticeSetting['oa_notice'])) { + $noticeSetting['oa_notice'] = [ + 'template_id' => '', + 'template_sn' => '', + 'name' => '', + 'first' => '', + 'remark' => '', + 'tpl' => [], + 'status' => 0, + ]; + } + $noticeSetting['oa_notice']['tips'] = NoticeEnum::getOperationTips(NoticeEnum::MNP, $noticeSetting['scene_id']); + if (empty($noticeSetting['mnp_notice'])) { + $noticeSetting['mnp_notice'] = [ + 'template_id' => '', + 'template_sn' => '', + 'name' => '', + 'tpl' => [], + 'status' => 0, + ]; + } + $noticeSetting['mnp_notice']['tips'] = NoticeEnum::getOperationTips(NoticeEnum::MNP, $noticeSetting['scene_id']); + $noticeSetting['system_notice']['is_show'] = in_array(NoticeEnum::SYSTEM, explode(',', $noticeSetting['support'])); + $noticeSetting['sms_notice']['is_show'] = in_array(NoticeEnum::SMS, explode(',', $noticeSetting['support'])); + $noticeSetting['oa_notice']['is_show'] = in_array(NoticeEnum::OA, explode(',', $noticeSetting['support'])); + $noticeSetting['mnp_notice']['is_show'] = in_array(NoticeEnum::MNP, explode(',', $noticeSetting['support'])); + $noticeSetting['default'] = ''; + $noticeSetting['type'] = NoticeEnum::getTypeDesc($noticeSetting['type']); + return $noticeSetting; + } + + + /** + * @notes 通知设置 + * @param $params + * @return bool + * @author 乔峰 + * @date 2022/3/29 11:34 + */ + public static function set($params) + { + try { + // 校验参数 + self::checkSet($params); + // 拼装更新数据 + $updateData = []; + foreach ($params['template'] as $item) { + $updateData[$item['type'] . '_notice'] = json_encode($item, JSON_UNESCAPED_UNICODE); + } + // 更新通知设置 + NoticeSetting::where('id', $params['id'])->update($updateData); + return true; + } catch (\Exception $e) { + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 校验参数 + * @param $params + * @throws \Exception + * @author 乔峰 + * @date 2022/3/29 11:35 + */ + public static function checkSet($params) + { + $noticeSetting = NoticeSetting::findOrEmpty($params['id'] ?? 0); + + if ($noticeSetting->isEmpty()) { + throw new \Exception('通知配置不存在'); + } + + if (!isset($params['template']) || !is_array($params['template']) || count($params['template']) == 0) { + throw new \Exception('模板配置不存在或格式错误'); + } + + // 通知类型 + $noticeType = ['system', 'sms', 'oa', 'mnp']; + + foreach ($params['template'] as $item) { + if (!is_array($item)) { + throw new \Exception('模板项格式错误'); + } + + if (!isset($item['type']) || !in_array($item['type'], $noticeType)) { + throw new \Exception('模板项缺少模板类型或模板类型有误'); + } + + switch ($item['type']) { + case "system"; + self::checkSystem($item); + break; + case "sms"; + self::checkSms($item); + break; + case "oa"; + self::checkOa($item); + break; + case "mnp"; + self::checkMnp($item); + break; + } + } + } + + + /** + * @notes 校验系统通知参数 + * @param $item + * @throws \Exception + * @author 乔峰 + * @date 2022/3/29 11:35 + */ + public static function checkSystem($item) + { + if (!isset($item['title']) || !isset($item['content']) || !isset($item['status'])) { + throw new \Exception('系统通知必填参数:title、content、status'); + } + } + + + /** + * @notes 校验短信通知必填参数 + * @param $item + * @throws \Exception + * @author 乔峰 + * @date 2022/3/29 11:35 + */ + public static function checkSms($item) + { + if (!isset($item['template_id']) || !isset($item['content']) || !isset($item['status'])) { + throw new \Exception('短信通知必填参数:template_id、content、status'); + } + } + + + /** + * @notes 校验微信模板消息参数 + * @param $item + * @throws \Exception + * @author 乔峰 + * @date 2022/3/29 11:35 + */ + public static function checkOa($item) + { + if (!isset($item['template_id']) || !isset($item['template_sn']) || !isset($item['name']) || !isset($item['first']) || !isset($item['remark']) || !isset($item['tpl']) || !isset($item['status'])) { + throw new \Exception('微信模板消息必填参数:template_id、template_sn、name、first、remark、tpl、status'); + } + } + + + /** + * @notes 校验微信小程序提醒必填参数 + * @param $item + * @throws \Exception + * @author 乔峰 + * @date 2022/3/29 11:35 + */ + public static function checkMnp($item) + { + if (!isset($item['template_id']) || !isset($item['template_sn']) || !isset($item['name']) || !isset($item['tpl']) || !isset($item['status'])) { + throw new \Exception('微信模板消息必填参数:template_id、template_sn、name、tpl、status'); + } + } +} \ No newline at end of file diff --git a/app/admin/logic/notice/SmsConfigLogic.php b/app/admin/logic/notice/SmsConfigLogic.php new file mode 100755 index 0000000..443b12c --- /dev/null +++ b/app/admin/logic/notice/SmsConfigLogic.php @@ -0,0 +1,127 @@ + 'ali', 'name' => '阿里云短信', 'status' => 1]), + ConfigService::get('sms', 'tencent', ['type' => 'tencent', 'name' => '腾讯云短信', 'status' => 0]), + ]; + return $config; + } + + + /** + * @notes 短信配置 + * @param $params + * @return bool|void + * @author 乔峰 + * @date 2022/3/29 11:37 + */ + public static function setConfig($params) + { + $type = $params['type']; + $params['name'] = self::getNameDesc(strtoupper($type)); + ConfigService::set('sms', $type, $params); + $default = ConfigService::get('sms', 'engine', false); + if ($params['status'] == 1 && $default === false) { + // 启用当前短信配置 并 设置当前短信配置为默认 + ConfigService::set('sms', 'engine', strtoupper($type)); + return true; + } + if ($params['status'] == 1 && $default != strtoupper($type)) { + // 找到默认短信配置 + $defaultConfig = ConfigService::get('sms', strtolower($default)); + // 状态置为禁用 并 更新 + $defaultConfig['status'] = 0; + ConfigService::set('sms', strtolower($default), $defaultConfig); + // 设置当前短信配置为默认 + ConfigService::set('sms', 'engine', strtoupper($type)); + return true; + } + } + + + /** + * @notes 查看短信配置详情 + * @param $params + * @return array|int|mixed|string|null + * @author 乔峰 + * @date 2022/3/29 11:37 + */ + public static function detail($params) + { + $default = []; + switch ($params['type']) { + case 'ali': + $default = [ + 'sign' => '', + 'app_key' => '', + 'secret_key' => '', + 'status' => 1, + 'name' => '阿里云短信', + ]; + break; + case 'tencent': + $default = [ + 'sign' => '', + 'app_id' => '', + 'secret_key' => '', + 'status' => 0, + 'secret_id' => '', + 'name' => '腾讯云短信', + ]; + break; + } + $result = ConfigService::get('sms', $params['type'], $default); + $result['status'] = intval($result['status'] ?? 0); + return $result; + } + + + /** + * @notes 获取短信平台名称 + * @param $value + * @return string + * @author 乔峰 + * @date 2022/3/29 11:37 + */ + public static function getNameDesc($value) + { + $desc = [ + 'ALI' => '阿里云短信', + 'TENCENT' => '腾讯云短信', + ]; + return $desc[$value] ?? ''; + } +} \ No newline at end of file diff --git a/app/admin/logic/setting/CustomerServiceLogic.php b/app/admin/logic/setting/CustomerServiceLogic.php new file mode 100755 index 0000000..2d8af50 --- /dev/null +++ b/app/admin/logic/setting/CustomerServiceLogic.php @@ -0,0 +1,65 @@ + $qrCode, + 'wechat' => ConfigService::get('customer_service', 'wechat', ''), + 'phone' => ConfigService::get('customer_service', 'phone', ''), + 'service_time' => ConfigService::get('customer_service', 'service_time', ''), + ]; + return $config; + } + + /** + * @notes 设置客服设置 + * @param $params + * @author ljj + * @date 2022/2/15 12:11 下午 + */ + public static function setConfig($params) + { + $allowField = ['qr_code','wechat','phone','service_time']; + foreach($params as $key => $value) { + if(in_array($key, $allowField)) { + if ($key == 'qr_code') { + $value = FileService::setFileUrl($value); + } + ConfigService::set('customer_service', $key, $value); + } + } + } +} \ No newline at end of file diff --git a/app/admin/logic/setting/HotSearchLogic.php b/app/admin/logic/setting/HotSearchLogic.php new file mode 100755 index 0000000..9a706d4 --- /dev/null +++ b/app/admin/logic/setting/HotSearchLogic.php @@ -0,0 +1,75 @@ + ConfigService::get('hot_search', 'status', 0), + // 热门搜索数据 + 'data' => HotSearch::field(['name', 'sort'])->order(['sort' => 'desc', 'id' =>'desc'])->select()->toArray(), + ]; + } + + + /** + * @notes 设置热门搜搜 + * @param $params + * @return bool + * @author 乔峰 + * @date 2022/9/5 18:58 + */ + public static function setConfig($params) + { + try { + if (!empty($params['data'])) { + $model = (new HotSearch()); + $model->where('id', '>', 0)->delete(); + $model->saveAll($params['data']); + } + + $status = empty($params['status']) ? 0 : $params['status']; + ConfigService::set('hot_search', 'status', $status); + + return true; + } catch (\Exception $e) { + self::$error = $e->getMessage(); + return false; + } + } + + +} \ No newline at end of file diff --git a/app/admin/logic/setting/StorageLogic.php b/app/admin/logic/setting/StorageLogic.php new file mode 100755 index 0000000..9bfab28 --- /dev/null +++ b/app/admin/logic/setting/StorageLogic.php @@ -0,0 +1,203 @@ + '本地存储', + 'path' => '存储在本地服务器', + 'engine' => 'local', + 'status' => $default == 'local' ? 1 : 0 + ], + [ + 'name' => '七牛云存储', + 'path' => '存储在七牛云,请前往七牛云开通存储服务', + 'engine' => 'qiniu', + 'status' => $default == 'qiniu' ? 1 : 0 + ], + [ + 'name' => '阿里云OSS', + 'path' => '存储在阿里云,请前往阿里云开通存储服务', + 'engine' => 'aliyun', + 'status' => $default == 'aliyun' ? 1 : 0 + ], + [ + 'name' => '腾讯云COS', + 'path' => '存储在腾讯云,请前往腾讯云开通存储服务', + 'engine' => 'qcloud', + 'status' => $default == 'qcloud' ? 1 : 0 + ] + ]; + return $data; + } + + + /** + * @notes 存储设置详情 + * @param $param + * @return mixed + * @author 乔峰 + * @date 2022/4/20 16:15 + */ + public static function detail($param) + { + + $default = ConfigService::get('storage', 'default', ''); + + // 本地存储 + $local = ['status' => $default == 'local' ? 1 : 0]; + // 七牛云存储 + $qiniu = ConfigService::get('storage', 'qiniu', [ + 'bucket' => '', + 'access_key' => '', + 'secret_key' => '', + 'domain' => '', + 'status' => $default == 'qiniu' ? 1 : 0 + ]); + + // 阿里云存储 + $aliyun = ConfigService::get('storage', 'aliyun', [ + 'bucket' => '', + 'access_key' => '', + 'secret_key' => '', + 'domain' => '', + 'status' => $default == 'aliyun' ? 1 : 0 + ]); + + // 腾讯云存储 + $qcloud = ConfigService::get('storage', 'qcloud', [ + 'bucket' => '', + 'region' => '', + 'access_key' => '', + 'secret_key' => '', + 'domain' => '', + 'status' => $default == 'qcloud' ? 1 : 0 + ]); + + $data = [ + 'local' => $local, + 'qiniu' => $qiniu, + 'aliyun' => $aliyun, + 'qcloud' => $qcloud + ]; + $result = $data[$param['engine']]; + if ($param['engine'] == $default) { + $result['status'] = 1; + } else { + $result['status'] = 0; + } + return $result; + } + + + /** + * @notes 设置存储参数 + * @param $params + * @return bool|string + * @author 乔峰 + * @date 2022/4/20 16:16 + */ + public static function setup($params) + { + if ($params['status'] == 1) { //状态为开启 + ConfigService::set('storage', 'default', $params['engine']); + } else { + ConfigService::set('storage', 'default', 'local'); + } + + switch ($params['engine']) { + case 'local': + ConfigService::set('storage', 'local', []); + break; + case 'qiniu': + ConfigService::set('storage', 'qiniu', [ + 'bucket' => $params['bucket'] ?? '', + 'access_key' => $params['access_key'] ?? '', + 'secret_key' => $params['secret_key'] ?? '', + 'domain' => $params['domain'] ?? '' + ]); + break; + case 'aliyun': + ConfigService::set('storage', 'aliyun', [ + 'bucket' => $params['bucket'] ?? '', + 'access_key' => $params['access_key'] ?? '', + 'secret_key' => $params['secret_key'] ?? '', + 'domain' => $params['domain'] ?? '' + ]); + break; + case 'qcloud': + ConfigService::set('storage', 'qcloud', [ + 'bucket' => $params['bucket'] ?? '', + 'region' => $params['region'] ?? '', + 'access_key' => $params['access_key'] ?? '', + 'secret_key' => $params['secret_key'] ?? '', + 'domain' => $params['domain'] ?? '', + ]); + break; + } + + Cache::delete('STORAGE_DEFAULT'); + Cache::delete('STORAGE_ENGINE'); + if ($params['engine'] == 'local' && $params['status'] == 0) { + return '默认开启本地存储'; + } else { + return true; + } + } + + + /** + * @notes 切换状态 + * @param $params + * @author 乔峰 + * @date 2022/4/20 16:17 + */ + public static function change($params) + { + $default = ConfigService::get('storage', 'default', ''); + if ($default == $params['engine']) { + ConfigService::set('storage', 'default', 'local'); + } else { + ConfigService::set('storage', 'default', $params['engine']); + } + Cache::delete('STORAGE_DEFAULT'); + Cache::delete('STORAGE_ENGINE'); + } +} \ No newline at end of file diff --git a/app/admin/logic/setting/TransactionSettingsLogic.php b/app/admin/logic/setting/TransactionSettingsLogic.php new file mode 100755 index 0000000..99d26f8 --- /dev/null +++ b/app/admin/logic/setting/TransactionSettingsLogic.php @@ -0,0 +1,64 @@ + ConfigService::get('transaction', 'cancel_unpaid_orders', 1), + 'cancel_unpaid_orders_times' => ConfigService::get('transaction', 'cancel_unpaid_orders_times', 30), + 'verification_orders' => ConfigService::get('transaction', 'verification_orders', 1), + 'verification_orders_times' => ConfigService::get('transaction', 'verification_orders_times', 24), + ]; + + return $config; + } + + /** + * @notes 设置交易设置 + * @param $params + * @author ljj + * @date 2022/2/15 11:49 上午 + */ + public static function setConfig($params) + { + ConfigService::set('transaction', 'cancel_unpaid_orders', $params['cancel_unpaid_orders']); + ConfigService::set('transaction', 'verification_orders', $params['verification_orders']); + + if (isset($params['cancel_unpaid_orders_times'])) { + ConfigService::set('transaction', 'cancel_unpaid_orders_times', $params['cancel_unpaid_orders_times']); + } + + if (isset($params['verification_orders_times'])) { + ConfigService::set('transaction', 'verification_orders_times', $params['verification_orders_times']); + } + } +} \ No newline at end of file diff --git a/app/admin/logic/setting/dict/DictDataLogic.php b/app/admin/logic/setting/dict/DictDataLogic.php new file mode 100755 index 0000000..0efda51 --- /dev/null +++ b/app/admin/logic/setting/dict/DictDataLogic.php @@ -0,0 +1,84 @@ + $params['name'], + 'value' => $params['value'], + 'sort' => $params['sort'] ?? 0, + 'status' => $params['status'], + 'remark' => $params['remark'] ?? '', + ]; + + if (!empty($params['id'])) { + return DictData::where(['id' => $params['id']])->update($data); + } else { + $dictType = DictType::findOrEmpty($params['type_id']); + $data['type_id'] = $params['type_id']; + $data['type_value'] = $dictType['type']; + return DictData::create($data); + } + } + + + /** + * @notes 删除字典数据 + * @param array $params + * @return bool + * @author 乔峰 + * @date 2022/6/20 17:01 + */ + public static function delete(array $params) + { + return DictData::destroy($params['id']); + } + + + /** + * @notes 获取字典数据详情 + * @param $params + * @return array + * @author 乔峰 + * @date 2022/6/20 17:01 + */ + public static function detail($params): array + { + return DictData::findOrEmpty($params['id'])->toArray(); + } + + +} \ No newline at end of file diff --git a/app/admin/logic/setting/dict/DictTypeLogic.php b/app/admin/logic/setting/dict/DictTypeLogic.php new file mode 100755 index 0000000..66fab0a --- /dev/null +++ b/app/admin/logic/setting/dict/DictTypeLogic.php @@ -0,0 +1,111 @@ + $params['name'], + 'type' => $params['type'], + 'status' => $params['status'], + 'remark' => $params['remark'] ?? '', + ]); + } + + + /** + * @notes 编辑字典类型 + * @param array $params + * @author 乔峰 + * @date 2022/6/20 16:10 + */ + public static function edit(array $params) + { + DictType::update([ + 'id' => $params['id'], + 'name' => $params['name'], + 'type' => $params['type'], + 'status' => $params['status'], + 'remark' => $params['remark'] ?? '', + ]); + + DictData::where(['type_id' => $params['id']]) + ->update(['type_value' => $params['type']]); + } + + + /** + * @notes 删除字典类型 + * @param array $params + * @author 乔峰 + * @date 2022/6/20 16:23 + */ + public static function delete(array $params) + { + DictType::destroy($params['id']); + } + + + /** + * @notes 获取字典详情 + * @param $params + * @return array + * @author 乔峰 + * @date 2022/6/20 16:23 + */ + public static function detail($params): array + { + return DictType::findOrEmpty($params['id'])->toArray(); + } + + + /** + * @notes 角色数据 + * @return array + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/10/13 10:44 + */ + public static function getAllData() + { + return DictType::where(['status' => YesNoEnum::YES]) + ->order(['id' => 'desc']) + ->select() + ->toArray(); + } +} \ No newline at end of file diff --git a/app/admin/logic/setting/system/CacheLogic.php b/app/admin/logic/setting/system/CacheLogic.php new file mode 100755 index 0000000..5335460 --- /dev/null +++ b/app/admin/logic/setting/system/CacheLogic.php @@ -0,0 +1,37 @@ +getRootPath().'runtime/file',true); + } +} \ No newline at end of file diff --git a/app/admin/logic/setting/system/SystemLogic.php b/app/admin/logic/setting/system/SystemLogic.php new file mode 100755 index 0000000..5dc93f1 --- /dev/null +++ b/app/admin/logic/setting/system/SystemLogic.php @@ -0,0 +1,65 @@ + '服务器操作系统', 'value' => PHP_OS], + ['param' => 'web服务器环境', 'value' => 'nginx/1.18'], + ['param' => 'PHP版本', 'value' => PHP_VERSION], + ]; + + $env = [ + [ 'option' => 'PHP版本', + 'require' => '7.2版本以上', + 'status' => (int)compare_php(PHP_VERSION), + 'remark' => '' + ] + ]; + + $auth = [ + [ + 'dir' => '/runtime', + 'require' => 'runtime目录可写', + 'status' => (int)check_dir_write('runtime'), + 'remark' => '' + ], + ]; + + return [ + 'server' => $server, + 'env' => $env, + 'auth' => $auth, + ]; + } + +} \ No newline at end of file diff --git a/app/admin/logic/setting/user/UserLogic.php b/app/admin/logic/setting/user/UserLogic.php new file mode 100755 index 0000000..9d00dd1 --- /dev/null +++ b/app/admin/logic/setting/user/UserLogic.php @@ -0,0 +1,108 @@ + ConfigService::get('default_image', 'user_avatar', $defaultAvatar), + ]; + return $config; + } + + + /** + * @notes 设置用户设置 + * @param array $params + * @return bool + * @author 乔峰 + * @date 2022/3/29 10:09 + */ + public function setConfig(array $params): bool + { + $avatar = FileService::setFileUrl($params['default_avatar']); + ConfigService::set('default_image', 'user_avatar', $avatar); + return true; + } + + + /** + * @notes 获取注册配置 + * @return array + * @author 乔峰 + * @date 2022/3/29 10:10 + */ + public function getRegisterConfig(): array + { + $config = [ + // 登录方式 + 'login_way' => ConfigService::get('login', 'login_way', config('project.login.login_way')), + // 注册强制绑定手机 + 'coerce_mobile' => ConfigService::get('login', 'coerce_mobile', config('project.login.coerce_mobile')), + // 政策协议 + 'login_agreement' => ConfigService::get('login', 'login_agreement', config('project.login.login_agreement')), + // 第三方登录 开关 + 'third_auth' => ConfigService::get('login', 'third_auth', config('project.login.third_auth')), + // 微信授权登录 + 'wechat_auth' => ConfigService::get('login', 'wechat_auth', config('project.login.wechat_auth')), + // qq授权登录 + 'qq_auth' => ConfigService::get('login', 'qq_auth', config('project.login.qq_auth')), + ]; + return $config; + } + + + /** + * @notes 设置登录注册 + * @param array $params + * @return bool + * @author 乔峰 + * @date 2022/3/29 10:10 + */ + public static function setRegisterConfig(array $params): bool + { + // 登录方式:1-账号密码登录;2-手机短信验证码登录 + ConfigService::set('login', 'login_way', $params['login_way']); + // 注册强制绑定手机 + ConfigService::set('login', 'coerce_mobile', $params['coerce_mobile']); + // 政策协议 + ConfigService::set('login', 'login_agreement', $params['login_agreement']); + // 第三方授权登录 + ConfigService::set('login', 'third_auth', $params['third_auth']); + // 微信授权登录 + ConfigService::set('login', 'wechat_auth', $params['wechat_auth']); + // qq登录 + ConfigService::set('login', 'qq_auth', $params['qq_auth']); + return true; + } + +} \ No newline at end of file diff --git a/app/admin/logic/setting/web/WebSettingLogic.php b/app/admin/logic/setting/web/WebSettingLogic.php new file mode 100755 index 0000000..5b5393d --- /dev/null +++ b/app/admin/logic/setting/web/WebSettingLogic.php @@ -0,0 +1,157 @@ + ConfigService::get('website', 'name'), + 'web_favicon' => FileService::getFileUrl(ConfigService::get('website', 'web_favicon')), + 'web_logo' => FileService::getFileUrl(ConfigService::get('website', 'web_logo')), + 'login_image' => FileService::getFileUrl(ConfigService::get('website', 'login_image')), + 'shop_name' => ConfigService::get('website', 'shop_name'), + 'shop_logo' => FileService::getFileUrl(ConfigService::get('website', 'shop_logo')), + + 'pc_logo' => FileService::getFileUrl(ConfigService::get('website', 'pc_logo')), + 'pc_title' => ConfigService::get('website', 'pc_title', ''), + 'pc_ico' => FileService::getFileUrl(ConfigService::get('website', 'pc_ico')), + 'pc_desc' => ConfigService::get('website', 'pc_desc', ''), + 'pc_keywords' => ConfigService::get('website', 'pc_keywords', ''), + ]; + } + + + /** + * @notes 设置网站信息 + * @param array $params + * @author 乔峰 + * @date 2021/12/28 15:43 + */ + public static function setWebsiteInfo(array $params) + { + $favicon = FileService::setFileUrl($params['web_favicon']); + $logo = FileService::setFileUrl($params['web_logo']); + $login = FileService::setFileUrl($params['login_image']); + $shopLogo = FileService::setFileUrl($params['shop_logo']); + $pcLogo = FileService::setFileUrl($params['pc_logo']); + $pcIco = FileService::setFileUrl($params['pc_ico'] ?? ''); + + ConfigService::set('website', 'name', $params['name']); + ConfigService::set('website', 'web_favicon', $favicon); + ConfigService::set('website', 'web_logo', $logo); + ConfigService::set('website', 'login_image', $login); + ConfigService::set('website', 'shop_name', $params['shop_name']); + ConfigService::set('website', 'shop_logo', $shopLogo); + ConfigService::set('website', 'pc_logo', $pcLogo); + + ConfigService::set('website', 'pc_title', $params['pc_title']); + ConfigService::set('website', 'pc_ico', $pcIco); + ConfigService::set('website', 'pc_desc', $params['pc_desc'] ?? ''); + ConfigService::set('website', 'pc_keywords', $params['pc_keywords'] ?? ''); + } + + + /** + * @notes 获取版权备案 + * @return array + * @author 乔峰 + * @date 2021/12/28 16:09 + */ + public static function getCopyright() : array + { + return ConfigService::get('copyright', 'config', []); + } + + + /** + * @notes 设置版权备案 + * @param array $params + * @return bool + * @author 乔峰 + * @date 2022/8/8 16:33 + */ + public static function setCopyright(array $params) + { + try { + if (!is_array($params['config'])) { + throw new \Exception('参数异常'); + } + ConfigService::set('copyright', 'config', $params['config'] ?? []); + return true; + } catch (\Exception $e) { + self::$error = $e->getMessage(); + return false; + } + } + + + /** + * @notes 设置政策协议 + * @param array $params + * @author ljj + * @date 2022/2/15 10:59 上午 + */ + public static function setAgreement(array $params) + { + $serviceContent = clear_file_domain($params['service_content'] ?? ''); + $privacyContent = clear_file_domain($params['privacy_content'] ?? ''); + ConfigService::set('agreement', 'service_title', $params['service_title'] ?? ''); + ConfigService::set('agreement', 'service_content', $serviceContent); + ConfigService::set('agreement', 'privacy_title', $params['privacy_title'] ?? ''); + ConfigService::set('agreement', 'privacy_content', $privacyContent); + } + + + /** + * @notes 获取政策协议 + * @return array + * @author ljj + * @date 2022/2/15 11:15 上午 + */ + public static function getAgreement() : array + { + $config = [ + 'service_title' => ConfigService::get('agreement', 'service_title'), + 'service_content' => ConfigService::get('agreement', 'service_content'), + 'privacy_title' => ConfigService::get('agreement', 'privacy_title'), + 'privacy_content' => ConfigService::get('agreement', 'privacy_content'), + ]; + + $config['service_content'] = get_file_domain($config['service_content']); + $config['privacy_content'] = get_file_domain($config['privacy_content']); + + return $config; + } +} \ No newline at end of file diff --git a/app/admin/logic/tools/GeneratorLogic.php b/app/admin/logic/tools/GeneratorLogic.php new file mode 100755 index 0000000..d88a288 --- /dev/null +++ b/app/admin/logic/tools/GeneratorLogic.php @@ -0,0 +1,503 @@ +findOrEmpty((int)$params['id']) + ->toArray(); + + $options = self::formatConfigByTableData($detail); + $detail['menu'] = $options['menu']; + $detail['delete'] = $options['delete']; + $detail['tree'] = $options['tree']; + $detail['relations'] = $options['relations']; + return $detail; + } + + + /** + * @notes 选择数据表 + * @param $params + * @param $adminId + * @return bool + * @author 乔峰 + * @date 2022/6/20 10:44 + */ + public static function selectTable($params, $adminId) + { + Db::startTrans(); + try { + foreach ($params['table'] as $item) { + // 添加主表基础信息 + $generateTable = self::initTable($item, $adminId); + // 获取数据表字段信息 + $column = self::getTableColumn($item['name']); + // 添加表字段信息 + self::initTableColumn($column, $generateTable['id']); + } + Db::commit(); + return true; + } catch (\Exception $e) { + Db::rollback(); + self::$error = $e->getMessage(); + return false; + } + } + + + /** + * @notes 编辑表信息 + * @param $params + * @return bool + * @author 乔峰 + * @date 2022/6/20 10:44 + */ + public static function editTable($params) + { + Db::startTrans(); + try { + // 格式化配置 + $options = self::formatConfigByTableData($params); + // 更新主表-数据表信息 + GenerateTable::update([ + 'id' => $params['id'], + 'table_name' => $params['table_name'], + 'table_comment' => $params['table_comment'], + 'template_type' => $params['template_type'], + 'author' => $params['author'] ?? '', + 'remark' => $params['remark'] ?? '', + 'generate_type' => $params['generate_type'], + 'module_name' => $params['module_name'], + 'class_dir' => $params['class_dir'] ?? '', + 'class_comment' => $params['class_comment'] ?? '', + 'menu' => $options['menu'], + 'delete' => $options['delete'], + 'tree' => $options['tree'], + 'relations' => $options['relations'], + ]); + + // 更新从表-数据表字段信息 + foreach ($params['table_column'] as $item) { + GenerateColumn::update([ + 'id' => $item['id'], + 'column_comment' => $item['column_comment'] ?? '', + 'is_required' => $item['is_required'] ?? 0, + 'is_insert' => $item['is_insert'] ?? 0, + 'is_update' => $item['is_update'] ?? 0, + 'is_lists' => $item['is_lists'] ?? 0, + 'is_query' => $item['is_query'] ?? 0, + 'query_type' => $item['query_type'], + 'view_type' => $item['view_type'], + 'dict_type' => $item['dict_type'] ?? '', + ]); + } + Db::commit(); + return true; + } catch (\Exception $e) { + Db::rollback(); + self::$error = $e->getMessage(); + return false; + } + } + + + /** + * @notes 删除表相关信息 + * @param $params + * @return bool + * @author 乔峰 + * @date 2022/6/16 9:30 + */ + public static function deleteTable($params) + { + Db::startTrans(); + try { + GenerateTable::whereIn('id', $params['id'])->delete(); + GenerateColumn::whereIn('table_id', $params['id'])->delete(); + Db::commit(); + return true; + } catch (\Exception $e) { + Db::rollback(); + self::$error = $e->getMessage(); + return false; + } + } + + + /** + * @notes 同步表字段 + * @param $params + * @return bool + * @author 乔峰 + * @date 2022/6/23 16:28 + */ + public static function syncColumn($params) + { + Db::startTrans(); + try { + // table 信息 + $table = GenerateTable::findOrEmpty($params['id']); + // 删除旧字段 + GenerateColumn::whereIn('table_id', $table['id'])->delete(); + // 获取当前数据表字段信息 + $column = self::getTableColumn($table['table_name']); + // 创建新字段数据 + self::initTableColumn($column, $table['id']); + + Db::commit(); + return true; + } catch (\Exception $e) { + Db::rollback(); + self::$error = $e->getMessage(); + return false; + } + } + + + /** + * @notes 生成代码 + * @param $params + * @return false|int[] + * @author 乔峰 + * @date 2022/6/24 9:43 + */ + public static function generate($params) + { + try { + // 获取数据表信息 + $tables = GenerateTable::with(['table_column']) + ->whereIn('id', $params['id']) + ->select()->toArray(); + $generator= Container::get(GenerateService::class); + $generator->delGenerateDirContent(); + $flag = array_unique(array_column($tables, 'table_name')); + $flag = implode(',', $flag); + $generator->setGenerateFlag(md5($flag . time()), false); + + // 循环生成 + foreach ($tables as $table) { + $generator->generate($table); + } + + $zipFile = ''; + // 生成压缩包 + if ($generator->getGenerateFlag()) { + $generator->zipFile(); + $generator->delGenerateFlag(); + $zipFile = $generator->getDownloadUrl(); + } + return ['file' => $zipFile]; + + } catch (\Exception $e) { + self::$error = $e->getMessage(); + return false; + } + } + + + /** + * @notes 预览 + * @param $params + * @return false + * @author 乔峰 + * @date 2022/6/23 16:27 + */ + public static function preview($params) + { + try { + // 获取数据表信息 + $table = GenerateTable::with(['table_column']) + ->whereIn('id', $params['id']) + ->findOrEmpty()->toArray(); + + return Container::get(GenerateService::class)->preview($table); + + } catch (\Exception $e) { + self::$error = $e->getMessage(); + return false; + } + } + + + /** + * @notes 获取表字段信息 + * @param $tableName + * @return array + * @author 乔峰 + * @date 2022/6/23 16:28 + */ + public static function getTableColumn($tableName) + { + $tableName = get_no_prefix_table_name($tableName); + return Db::name($tableName)->getFields(); + } + + + /** + * @notes 初始化代码生成数据表信息 + * @param $tableData + * @param $adminId + * @return GenerateTable|\think\Model + * @author 乔峰 + * @date 2022/6/23 16:28 + */ + public static function initTable($tableData, $adminId) + { + return GenerateTable::create([ + 'table_name' => $tableData['name'], + 'table_comment' => $tableData['comment'], + 'template_type' => GeneratorEnum::TEMPLATE_TYPE_SINGLE, + 'generate_type' => GeneratorEnum::GENERATE_TYPE_ZIP, + 'module_name' => 'admin', + 'admin_id' => $adminId, + // 菜单配置 + 'menu' => [ + 'pid' => 0, // 父级菜单id + 'type' => GeneratorEnum::GEN_SELF, // 构建方式 0-手动添加 1-自动构建 + 'name' => $tableData['comment'], // 菜单名称 + ], + // 删除配置 + 'delete' => [ + 'type' => GeneratorEnum::DELETE_TRUE, // 删除类型 + 'name' => GeneratorEnum::DELETE_NAME, // 默认删除字段名 + ], + // 关联配置 + 'relations' => [], + // 树形crud + 'tree' => [] + ]); + } + + + /** + * @notes 初始化代码生成字段信息 + * @param $column + * @param $tableId + * @throws \Exception + * @author 乔峰 + * @date 2022/6/23 16:28 + */ + public static function initTableColumn($column, $tableId) + { + $defaultColumn = ['id', 'create_time', 'update_time', 'delete_time']; + + $insertColumn = []; + foreach ($column as $value) { + $required = 0; + if ($value['notnull'] && !$value['primary'] && !in_array($value['name'], $defaultColumn)) { + $required = 1; + } + + $columnData = [ + 'table_id' => $tableId, + 'column_name' => $value['name'], + 'column_comment' => $value['comment'], + 'column_type' => self::getDbFieldType($value['type']), + 'is_required' => $required, + 'is_pk' => $value['primary'] ? 1 : 0, + ]; + + if (!in_array($value['name'], $defaultColumn)) { + $columnData['is_insert'] = 1; + $columnData['is_update'] = 1; + $columnData['is_lists'] = 1; + $columnData['is_query'] = 1; + } + $insertColumn[] = $columnData; + } + + (new GenerateColumn())->saveAll($insertColumn); + } + + + /** + * @notes 下载文件 + * @param $fileName + * @return false|string + * @author 乔峰 + * @date 2022/6/24 9:51 + */ + public static function download(string $fileName) + { + $cacheFileName = Cache::get('curd_file_name' . $fileName); + if (empty($cacheFileName)) { + self::$error = '请重新生成代码'; + return false; + } + $path = base_path() . '/runtime/generate/' . $fileName; + if (!file_exists($path)) { + self::$error = '下载失败'; + return false; + } + Cache::delete('curd_file_name' . $fileName); + return $path; + } + + + /** + * @notes 获取数据表字段类型 + * @param string $type + * @return string + * @author 乔峰 + * @date 2022/6/15 10:11 + */ + public static function getDbFieldType(string $type): string + { + if (0 === strpos($type, 'set') || 0 === strpos($type, 'enum')) { + $result = 'string'; + } elseif (preg_match('/(double|float|decimal|real|numeric)/is', $type)) { + $result = 'float'; + } elseif (preg_match('/(int|serial|bit)/is', $type)) { + $result = 'int'; + } elseif (preg_match('/bool/is', $type)) { + $result = 'bool'; + } elseif (0 === strpos($type, 'timestamp')) { + $result = 'timestamp'; + } elseif (0 === strpos($type, 'datetime')) { + $result = 'datetime'; + } elseif (0 === strpos($type, 'date')) { + $result = 'date'; + } else { + $result = 'string'; + } + return $result; + } + + + /** + * @notes + * @param $options + * @param $tableComment + * @return array + * @author 乔峰 + * @date 2022/12/13 18:23 + */ + public static function formatConfigByTableData($options) + { + // 菜单配置 + $menuConfig = $options['menu'] ?? []; + // 删除配置 + $deleteConfig = $options['delete'] ?? []; + // 关联配置 + $relationsConfig = $options['relations'] ?? []; + // 树表crud配置 + $treeConfig = $options['tree'] ?? []; + + $relations = []; + foreach ($relationsConfig as $relation) { + $relations[] = [ + 'name' => $relation['name'] ?? '', + 'model' => $relation['model'] ?? '', + 'type' => $relation['type'] ?? GeneratorEnum::RELATION_HAS_ONE, + 'local_key' => $relation['local_key'] ?? 'id', + 'foreign_key' => $relation['foreign_key'] ?? 'id', + ]; + } + + $options['menu'] = [ + 'pid' => intval($menuConfig['pid'] ?? 0), + 'type' => intval($menuConfig['type'] ?? GeneratorEnum::GEN_SELF), + 'name' => !empty($menuConfig['name']) ? $menuConfig['name'] : $options['table_comment'], + ]; + $options['delete'] = [ + 'type' => intval($deleteConfig['type'] ?? GeneratorEnum::DELETE_TRUE), + 'name' => !empty($deleteConfig['name']) ? $deleteConfig['name'] : GeneratorEnum::DELETE_NAME, + ]; + $options['relations'] = $relations; + $options['tree'] = [ + 'tree_id' => $treeConfig['tree_id'] ?? "", + 'tree_pid' =>$treeConfig['tree_pid'] ?? "", + 'tree_name' => $treeConfig['tree_name'] ?? '', + ]; + + return $options; + } + + + /** + * @notes 获取所有模型 + * @param string $module + * @return array + * @author 乔峰 + * @date 2022/12/14 11:04 + */ + public static function getAllModels($module = 'common') + { + if(empty($module)) { + return []; + } + $modulePath = base_path() . $module . '/model/'; + if(!is_dir($modulePath)) { + return []; + } + + $modulefiles = glob($modulePath . '*'); + $targetFiles = []; + foreach ($modulefiles as $file) { + $fileBaseName = basename($file, '.php'); + if (is_dir($file)) { + $file = glob($file . '/*'); + foreach ($file as $item) { + if (is_dir($item)) { + continue; + } + $targetFiles[] = sprintf( + "\\app\\" . $module . "\\model\\%s\\%s", + $fileBaseName, + basename($item, '.php') + ); + } + } else { + if ($fileBaseName == 'BaseModel') { + continue; + } + $targetFiles[] = sprintf( + "\\app\\" . $module . "\\model\\%s", + basename($file, '.php') + ); + } + } + + return $targetFiles; + } + +} \ No newline at end of file diff --git a/app/admin/logic/user/UserLogic.php b/app/admin/logic/user/UserLogic.php new file mode 100755 index 0000000..9f2bc08 --- /dev/null +++ b/app/admin/logic/user/UserLogic.php @@ -0,0 +1,66 @@ + $userId])->field($field) + ->findOrEmpty(); + + $user['channel'] = UserTerminalEnum::getTermInalDesc($user['channel']); + $user->sex = $user->getData('sex'); + return $user->toArray(); + } + + + /** + * @notes 更新用户信息 + * @param array $params + * @return User + * @author 乔峰 + * @date 2022/9/22 16:38 + */ + public static function setUserInfo(array $params) + { + return User::update([ + 'id' => $params['id'], + $params['field'] => $params['value'] + ]); + } + +} \ No newline at end of file diff --git a/app/admin/middleware/AuthMiddleware.php b/app/admin/middleware/AuthMiddleware.php new file mode 100644 index 0000000..0717440 --- /dev/null +++ b/app/admin/middleware/AuthMiddleware.php @@ -0,0 +1,63 @@ +controllerObject->isNotNeedLogin()) { + return $handler($request); + } + + //系统默认超级管理员,无需权限验证 + if (1 === $request->adminInfo['root']) { + return $handler($request); + } + + $adminAuthCache = new AdminAuthCache($request->adminInfo['admin_id']); + + // 当前访问路径 + $accessUri = strtolower($request->controller . '/' . $request->action); + // 全部路由 + $allUri = $this->formatUrl($adminAuthCache->getAllUri()); + + // 判断该当前访问的uri是否存在,不存在无需验证 + if (!in_array($accessUri, $allUri)) { + return $handler($request); + } + + // 当前管理员拥有的路由权限 + $AdminUris = $adminAuthCache->getAdminUri() ?? []; + $AdminUris = $this->formatUrl($AdminUris); + + if (in_array($accessUri, $AdminUris)) { + return $handler($request); + } + return JsonService::fail('权限不足,无法访问或操作'); + } + + /** + * @notes 格式化URL + * @param array $data + * @return array|string[] + * @author 乔峰 + * @date 2022/7/7 15:39 + */ + public function formatUrl(array $data) + { + return array_map(function ($item) { + return strtolower(Str::camel($item)); + }, $data); + } +} \ No newline at end of file diff --git a/app/admin/middleware/InitMiddleware.php b/app/admin/middleware/InitMiddleware.php new file mode 100644 index 0000000..de54878 --- /dev/null +++ b/app/admin/middleware/InitMiddleware.php @@ -0,0 +1,35 @@ +controller); + $controllerClass = new $controller; + if (($controllerClass instanceof BaseAdminController) === false) { + throw new ControllerExtendException($controller, '404'); + } + } catch (ClassNotFoundException $e) { + throw new HttpException(404, 'controller not exists:' . $e->getClass()); + } + + //创建控制器对象 + $request->controllerObject = new $controller; + + return $handler($request); + } +} \ No newline at end of file diff --git a/app/admin/middleware/LoginMiddleware.php b/app/admin/middleware/LoginMiddleware.php new file mode 100644 index 0000000..3b83198 --- /dev/null +++ b/app/admin/middleware/LoginMiddleware.php @@ -0,0 +1,64 @@ +header('token'); +// $controller = str_replace('.', '\\', $request->controller); + //判断接口是否免登录 + $isNotNeedLogin = $request->controllerObject->isNotNeedLogin(); + + //不直接判断$isNotNeedLogin结果,使不需要登录的接口通过,为了兼容某些接口可以登录或不登录访问 + if (empty($token) && !$isNotNeedLogin) { + //没有token并且该地址需要登录才能访问 + return JsonService::fail('请求参数缺token', [], 0, 0); + } + + $adminInfo = (new AdminTokenCache())->getAdminInfo($token); + if (empty($adminInfo) && !$isNotNeedLogin) { + //token过期无效并且该地址需要登录才能访问 + return JsonService::fail('登录超时,请重新登录', [], -1); + } + + //token临近过期,自动续期 + if ($adminInfo) { + //获取临近过期自动续期时长 + $beExpireDuration = Config::get('project.admin_token.be_expire_duration'); + //token续期 + if (time() > ($adminInfo['expire_time'] - $beExpireDuration)) { + $result = AdminTokenService::overtimeToken($token); + //续期失败(数据表被删除导致) + if (empty($result)) { + return JsonService::fail('登录过期', [], -1); + } + } + } + + //给request赋值,用于控制器 + $request->adminInfo = $adminInfo; + $request->adminId = $adminInfo['admin_id'] ?? 0; + + return $handler($request); + } +} \ No newline at end of file diff --git a/app/admin/service/AdminTokenService.php b/app/admin/service/AdminTokenService.php new file mode 100644 index 0000000..a522056 --- /dev/null +++ b/app/admin/service/AdminTokenService.php @@ -0,0 +1,116 @@ +find(); + + //获取token延长过期的时间 + $expireTime = $time + Config::get('project.admin_token.expire_duration'); + + $adminTokenCache = new AdminTokenCache(); + + //token处理 + if ($adminSession) { + if ($adminSession->expire_time < $time || $multipointLogin === 0) { + //清空缓存 + $adminTokenCache->deleteAdminInfo($adminSession->token); + //如果token过期或账号设置不支持多处登录,更新token + $adminSession->token = create_token($adminId); + } + $adminSession->expire_time = $expireTime; + $adminSession->update_time = $time; + + $adminSession->save(); + } else { + //找不到在该终端的token记录,创建token记录 + $adminSession = AdminSession::create([ + 'admin_id' => $adminId, + 'terminal' => $terminal, + 'token' => create_token($adminId), + 'expire_time' => $expireTime + ]); + } + + return $adminTokenCache->setAdminInfo($adminSession->token); + } + + /** + * @notes 延长token过期时间 + * @param $token + * @return array|false|mixed + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2021/7/5 14:25 + */ + public static function overtimeToken($token) + { + $time = time(); + $adminSession = AdminSession::where('token', '=', $token)->findOrEmpty(); + if ($adminSession->isEmpty()) { + return false; + } + //延长token过期时间 + $adminSession->expire_time = $time + Config::get('project.admin_token.expire_duration'); + $adminSession->update_time = $time; + $adminSession->save(); + return (new AdminTokenCache())->setAdminInfo($adminSession->token); + } + + /** + * @notes 设置token为过期 + * @param $token + * @return bool + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2021/7/5 14:31 + */ + public static function expireToken($token) + { + $adminSession = AdminSession::where('token', '=', $token) + ->with('admin') + ->findOrEmpty(); + + if ($adminSession->isEmpty()) { + return false; + } + + //当支持多处登录的时候,服务端不注销 + if ($adminSession->admin->multipoint_login === 1) { + return false; + } + + $time = time(); + $adminSession->expire_time = $time; + $adminSession->update_time = $time; + $adminSession->save(); + + return (new AdminTokenCache())->deleteAdminInfo($token); + } +} \ No newline at end of file diff --git a/app/admin/validate/FileValidate.php b/app/admin/validate/FileValidate.php new file mode 100644 index 0000000..39c2d15 --- /dev/null +++ b/app/admin/validate/FileValidate.php @@ -0,0 +1,101 @@ + 'require|number', + 'cid' => 'require|number', + 'ids' => 'require|array', + 'type' => 'require|in:10,20,30', + 'pid' => 'require|number', + 'name' => 'require|max:20' + ]; + + protected $message = [ + 'id.require' => '缺少id参数', + 'cid.require' => '缺少cid参数', + 'ids.require' => '缺少ids参数', + 'type.require' => '缺少type参数', + 'pid.require' => '缺少pid参数', + 'name.require' => '请填写分组名称', + 'name.max' => '分组名称长度须为20字符内', + ]; + + + /** + * @notes id验证场景 + * @return \app\admin\validate\FileValidate + * @author 乔峰 + * @date 2021/12/29 14:32 + */ + public function sceneId() + { + return $this->only(['id']); + } + + + /** + * @notes 重命名文件场景 + * @return FileValidate + * @author 乔峰 + * @date 2021/12/29 14:32 + */ + public function sceneRename() + { + return $this->only(['id', 'name']); + } + + + /** + * @notes 新增分类场景 + * @return FileValidate + * @author 乔峰 + * @date 2021/12/29 14:33 + */ + public function sceneAddCate() + { + return $this->only(['type', 'pid', 'name']); + } + + + /** + * @notes 编辑分类场景 + * @return FileValidate + * @author 乔峰 + * @date 2021/12/29 14:33 + */ + public function sceneEditCate() + { + return $this->only(['id', 'name']); + } + + + /** + * @notes 移动场景 + * @return FileValidate + * @author 乔峰 + * @date 2021/12/29 14:33 + */ + public function sceneMove() + { + return $this->only(['ids', 'cid']); + } + + + /** + * @notes 删除场景 + * @return FileValidate + * @author 乔峰 + * @date 2021/12/29 14:35 + */ + public function sceneDelete() + { + return $this->only(['ids']); + } +} \ No newline at end of file diff --git a/app/admin/validate/LoginValidate.php b/app/admin/validate/LoginValidate.php new file mode 100644 index 0000000..a6d7c7d --- /dev/null +++ b/app/admin/validate/LoginValidate.php @@ -0,0 +1,84 @@ + 'require|in:' . AdminTerminalEnum::PC . ',' . AdminTerminalEnum::MOBILE, + 'account' => 'require', + 'password' => 'require|password', + ]; + + protected $message = [ + 'account.require' => '请输入账号', + 'password.require' => '请输入密码' + ]; + + /** + * @notes @notes 密码验证 + * @param $password + * @param $other + * @param $data + * @return bool|string + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2021/7/2 14:00 + */ + public function password($password, $other, $data) + { + // 登录限制 + $config = [ + 'login_restrictions' => ConfigService::get('admin_login', 'login_restrictions'), + 'password_error_times' => ConfigService::get('admin_login', 'password_error_times'), + 'limit_login_time' => ConfigService::get('admin_login', 'limit_login_time'), + ]; + + $adminAccountSafeCache = new AdminAccountSafeCache(); + if ($config['login_restrictions'] == 1) { + $adminAccountSafeCache->count = $config['password_error_times']; + $adminAccountSafeCache->minute = $config['limit_login_time']; + } + + //后台账号安全机制,连续输错后锁定,防止账号密码暴力破解 + if ($config['login_restrictions'] == 1 && !$adminAccountSafeCache->isSafe()) { + return '密码连续' . $adminAccountSafeCache->count . '次输入错误,请' . $adminAccountSafeCache->minute . '分钟后重试'; + } + + $adminInfo = Admin::where('account', '=', $data['account']) + ->field(['password,disable']) + ->findOrEmpty(); + + if ($adminInfo->isEmpty()) { + return '账号不存在'; + } + + if ($adminInfo['disable'] === 1) { + return '账号已禁用'; + } + + if (empty($adminInfo['password'])) { + $adminAccountSafeCache->record(); + return '账号不存在'; + } + $passwordSalt = Config::get('project.unique_identification'); + if ($adminInfo['password'] !== create_password($password, $passwordSalt)) { + $adminAccountSafeCache->record(); + return '密码错误'; + } + + $adminAccountSafeCache->relieve(); + return true; + } +} \ No newline at end of file diff --git a/app/admin/validate/auth/AdminValidate.php b/app/admin/validate/auth/AdminValidate.php new file mode 100755 index 0000000..0d0f3ca --- /dev/null +++ b/app/admin/validate/auth/AdminValidate.php @@ -0,0 +1,168 @@ + 'require|checkAdmin', + 'account' => 'require|length:1,32|unique:'.Admin::class, + 'name' => 'require|length:1,16|unique:'.Admin::class, + 'password' => 'require|length:6,32|edit', + 'password_confirm' => 'requireWith:password|confirm', + 'role_id' => 'require', + 'disable' => 'require|in:0,1|checkAbleDisable', + 'multipoint_login' => 'require|in:0,1', + ]; + + protected $message = [ + 'id.require' => '管理员id不能为空', + 'account.require' => '账号不能为空', + 'account.length' => '账号长度须在1-32位字符', + 'account.unique' => '账号已存在', + 'password.require' => '密码不能为空', + 'password.length' => '密码长度须在6-32位字符', + 'password_confirm.requireWith' => '确认密码不能为空', + 'password_confirm.confirm' => '两次输入的密码不一致', + 'name.require' => '名称不能为空', + 'name.length' => '名称须在1-16位字符', + 'name.unique' => '名称已存在', + 'role_id.require' => '请选择角色', + 'disable.require' => '请选择状态', + 'disable.in' => '状态值错误', + 'multipoint_login.require' => '请选择是否支持多处登录', + 'multipoint_login.in' => '多处登录状态值为误', + ]; + + /** + * @notes 添加场景 + * @return AdminValidate + * @author 乔峰 + * @date 2021/12/29 15:46 + */ + public function sceneAdd() + { + return $this->remove(['password', 'edit']) + ->remove('id', 'require|checkAdmin') + ->remove('disable', 'checkAbleDisable'); + } + + /** + * @notes 详情场景 + * @return AdminValidate + * @author 乔峰 + * @date 2021/12/29 15:46 + */ + public function sceneDetail() + { + return $this->only(['id']); + } + + /** + * @notes 编辑场景 + * @return AdminValidate + * @author 乔峰 + * @date 2021/12/29 15:47 + */ + public function sceneEdit() + { + return $this->remove('password', 'require|length') + ->append('id', 'require|checkAdmin'); + } + + + /** + * @notes 删除场景 + * @return AdminValidate + * @author 乔峰 + * @date 2021/12/29 15:47 + */ + public function sceneDelete() + { + return $this->only(['id']); + } + + + + /** + * @notes 编辑情况下,检查是否填密码 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @author 乔峰 + * @date 2021/12/29 10:19 + */ + public function edit($value, $rule, $data) + { + if (empty($data['password']) && empty($data['password_confirm'])) { + return true; + } + $len = strlen($value); + if ($len < 6 || $len > 32) { + return '密码长度须在6-32位字符'; + } + return true; + } + + + /** + * @notes 检查指定管理员是否存在 + * @param $value + * @return bool|string + * @author 乔峰 + * @date 2021/12/29 10:19 + */ + public function checkAdmin($value) + { + $admin = Admin::findOrEmpty($value); + if ($admin->isEmpty()) { + return '管理员不存在'; + } + return true; + } + + + /** + * @notes 禁用校验 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @author 乔峰 + * @date 2022/8/11 9:59 + */ + public function checkAbleDisable($value, $rule, $data) + { + $admin = Admin::findOrEmpty($data['id']); + if ($admin->isEmpty()) { + return '管理员不存在'; + } + + if ($value && $admin['root']) { + return '超级管理员不允许被禁用'; + } + return true; + } + +} \ No newline at end of file diff --git a/app/admin/validate/auth/MenuValidate.php b/app/admin/validate/auth/MenuValidate.php new file mode 100755 index 0000000..39f30c0 --- /dev/null +++ b/app/admin/validate/auth/MenuValidate.php @@ -0,0 +1,195 @@ + 'require', + 'pid' => 'require|checkPid', + 'type' => 'require|in:M,C,A', + 'name' => 'require|length:1,30|checkUniqueName', + 'icon' => 'max:100', + 'sort' => 'require|egt:0', + 'perms' => 'max:100', + 'paths' => 'max:200', + 'component' => 'max:200', + 'selected' => 'max:200', + 'params' => 'max:200', + 'is_cache' => 'require|in:0,1', + 'is_show' => 'require|in:0,1', + 'is_disable' => 'require|in:0,1', + ]; + + + protected $message = [ + 'id.require' => '参数缺失', + 'pid.require' => '请选择上级菜单', + 'type.require' => '请选择菜单类型', + 'type.in' => '菜单类型参数值错误', + 'name.require' => '请填写菜单名称', + 'name.length' => '菜单名称长度需为1~30个字符', + 'icon.max' => '图标名称不能超过100个字符', + 'sort.require' => '请填写排序', + 'sort.egt' => '排序值需大于或等于0', + 'perms.max' => '权限字符不能超过100个字符', + 'paths.max' => '路由地址不能超过200个字符', + 'component.max' => '组件路径不能超过200个字符', + 'selected.max' => '选中菜单路径不能超过200个字符', + 'params.max' => '路由参数不能超过200个字符', + 'is_cache.require' => '请选择缓存状态', + 'is_cache.in' => '缓存状态参数值错误', + 'is_show.require' => '请选择显示状态', + 'is_show.in' => '显示状态参数值错误', + 'is_disable.require' => '请选择菜单状态', + 'is_disable.in' => '菜单状态参数值错误', + ]; + + + /** + * @notes 添加场景 + * @return MenuValidate + * @author 乔峰 + * @date 2022/6/29 18:26 + */ + public function sceneAdd() + { + return $this->remove('id', true); + } + + + /** + * @notes 详情场景 + * @return MenuValidate + * @author 乔峰 + * @date 2022/6/29 18:27 + */ + public function sceneDetail() + { + return $this->only(['id']); + } + + + /** + * @notes 删除场景 + * @return MenuValidate + * @author 乔峰 + * @date 2022/6/29 18:27 + */ + public function sceneDelete() + { + return $this->only(['id']) + ->append('id', 'checkAbleDelete'); + } + + + /** + * @notes 更新状态场景 + * @return MenuValidate + * @author 乔峰 + * @date 2022/7/6 17:04 + */ + public function sceneStatus() + { + return $this->only(['id', 'is_disable']); + } + + + /** + * @notes 校验菜单名称是否已存在 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @author 乔峰 + * @date 2022/6/29 18:24 + */ + protected function checkUniqueName($value, $rule, $data) + { + if ($data['type'] != 'M') { + return true; + } + $where[] = ['type', '=', $data['type']]; + $where[] = ['name', '=', $data['name']]; + + if (!empty($data['id'])) { + $where[] = ['id', '<>', $data['id']]; + } + + $check = SystemMenu::where($where)->findOrEmpty(); + + if (!$check->isEmpty()) { + return '菜单名称已存在'; + } + + return true; + } + + + /** + * @notes 是否有子级菜单 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @author 乔峰 + * @date 2022/6/30 9:40 + */ + protected function checkAbleDelete($value, $rule, $data) + { + $hasChild = SystemMenu::where(['pid' => $value])->findOrEmpty(); + if (!$hasChild->isEmpty()) { + //return '存在子菜单,不允许删除'; + } + + // 已绑定角色菜单不可以删除 + $isBindRole = SystemRole::hasWhere('roleMenuIndex', ['menu_id' => $value])->findOrEmpty(); + if (!$isBindRole->isEmpty()) { + //return '已分配菜单不可删除'; + } + + return true; + } + + + /** + * @notes 校验上级 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @author 乔峰 + * @date 2022/6/30 9:51 + */ + protected function checkPid($value, $rule, $data) + { + if (!empty($data['id']) && $data['id'] == $value) { + return '上级菜单不能选择自己'; + } + return true; + } + +} \ No newline at end of file diff --git a/app/admin/validate/auth/RoleValidate.php b/app/admin/validate/auth/RoleValidate.php new file mode 100755 index 0000000..9863473 --- /dev/null +++ b/app/admin/validate/auth/RoleValidate.php @@ -0,0 +1,119 @@ + 'require|checkRole', + 'name' => 'require|max:64|unique:' . SystemRole::class . ',name', + 'menu_id' => 'array', + ]; + + protected $message = [ + 'id.require' => '请选择角色', + 'name.require' => '请输入角色名称', + 'name.max' => '角色名称最长为16个字符', + 'name.unique' => '角色名称已存在', + 'menu_id.array' => '权限格式错误' + ]; + + /** + * @notes 添加场景 + * @return RoleValidate + * @author 乔峰 + * @date 2021/12/29 15:47 + */ + public function sceneAdd() + { + return $this->only(['name', 'menu_id']); + } + + /** + * @notes 详情场景 + * @return RoleValidate + * @author 乔峰 + * @date 2021/12/29 15:47 + */ + public function sceneDetail() + { + return $this->only(['id']); + } + + /** + * @notes 删除场景 + * @return RoleValidate + * @author 乔峰 + * @date 2021/12/29 15:48 + */ + public function sceneDel() + { + return $this->only(['id']) + ->append('id', 'checkAdmin'); + } + + + /** + * @notes 验证角色是否存在 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2021/12/29 15:48 + */ + public function checkRole($value, $rule, $data) + { + if (!SystemRole::find($value)) { + return '角色不存在'; + } + return true; + } + + + + /** + * @notes 验证角色是否被使用 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2021/12/29 15:49 + */ + public function checkAdmin($value, $rule, $data) + { + if (Admin::where(['role_id' => $value])->find()) { + return '有管理员在使用该角色,不允许删除'; + } + return true; + } + +} \ No newline at end of file diff --git a/app/admin/validate/auth/editSelfValidate.php b/app/admin/validate/auth/editSelfValidate.php new file mode 100755 index 0000000..992fc97 --- /dev/null +++ b/app/admin/validate/auth/editSelfValidate.php @@ -0,0 +1,71 @@ + 'require|length:1,16', + 'avatar' => 'require', + 'password_old' => 'length:6,32', + 'password' => 'length:6,32|checkPassword', + 'password_confirm' => 'requireWith:password|confirm', + ]; + + + protected $message = [ + 'name.require' => '请填写名称', + 'name.length' => '名称须在1-16位字符', + 'avatar.require' => '请选择头像', + 'password_now.length' => '密码长度须在6-32位字符', + 'password_confirm.requireWith' => '确认密码不能为空', + 'password_confirm.confirm' => '两次输入的密码不一致', + ]; + + + /** + * @notes 校验密码 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @author 乔峰 + * @date 2022/4/8 17:40 + */ + public function checkPassword($value, $rule, $data) + { + if (empty($data['password_old'])) { + return '请填写当前密码'; + } + + $admin = Admin::findOrEmpty($data['admin_id']); + if (!password_verify($data['password_old'],$admin['password'])) { + return '当前密码错误'; + } + + return true; + } + +} \ No newline at end of file diff --git a/app/admin/validate/channel/MnpSettingsValidate.php b/app/admin/validate/channel/MnpSettingsValidate.php new file mode 100755 index 0000000..4c5d72a --- /dev/null +++ b/app/admin/validate/channel/MnpSettingsValidate.php @@ -0,0 +1,36 @@ + 'require', + 'app_secret' => 'require', + ]; + + protected $message = [ + 'app_id.require' => '请填写AppID', + 'app_secret.require' => '请填写AppSecret', + ]; +} \ No newline at end of file diff --git a/app/admin/validate/channel/OfficialAccountReplyValidate.php b/app/admin/validate/channel/OfficialAccountReplyValidate.php new file mode 100755 index 0000000..57adb28 --- /dev/null +++ b/app/admin/validate/channel/OfficialAccountReplyValidate.php @@ -0,0 +1,72 @@ + 'require|integer', + 'reply_type' => 'require|in:1,2,3', + 'name' => 'require', + 'content_type' => 'require|in:1', + 'content' => 'require', + 'status' => 'require|in:0,1', + 'keyword' => 'requireIf:reply_type,2', + 'matching_type' => 'requireIf:reply_type,2|in:1,2', + 'sort' => 'requireIf:reply_type,2', + 'reply_num' => 'requireIf:reply_type,2|in:1', + 'new_sort' => 'require|integer|egt:0', + ]; + + protected $message = [ + 'reply_type.require' => '请输入回复类型', + 'reply_type.in' => '回复类型状态值错误', + 'name.require' => '请输入规则名称', + 'content_type.require' => '请选择内容类型', + 'content_type.in' => '内容类型状态值有误', + 'content.require' => '请输入回复内容', + 'status.require' => '请选择启用状态', + 'status.in' => '启用状态值错误', + 'keyword.requireIf' => '请输入关键词', + 'matching_type.requireIf' => '请选择匹配类型', + 'matching_type.in' => '匹配类型状态值错误', + 'sort.requireIf' => '请输入排序值', + 'sort.integer' => '排序值须为整型', + 'sort.egt' => '排序值须大于或等于0', + 'reply_num.requireIf' => '请选择回复数量', + 'reply_num.in' => '回复数量状态值错误', + 'id.require' => '参数缺失', + 'id.integer' => '参数格式错误', + 'new_sort.require' => '请输入新排序值', + 'new_sort.integer' => '新排序值须为整型', + 'new_sort.egt' => '新排序值须大于或等于0', + ]; + + protected $scene = [ + 'add' => ['reply_type', 'name', 'content_type', 'content', 'status', 'keyword', 'matching_type', 'sort', 'reply_num'], + 'detail' => ['id'], + 'delete' => ['id'], + 'sort' => ['id', 'new_sort'], + 'status' => ['id'], + 'edit' => ['id', 'reply_type', 'name', 'content_type', 'content', 'status','keyword', 'matching_type', 'sort', 'reply_num'] + ]; +} \ No newline at end of file diff --git a/app/admin/validate/channel/OfficialAccountSettingValidate.php b/app/admin/validate/channel/OfficialAccountSettingValidate.php new file mode 100755 index 0000000..e93f8a0 --- /dev/null +++ b/app/admin/validate/channel/OfficialAccountSettingValidate.php @@ -0,0 +1,38 @@ + 'require', + 'app_secret' => 'require', + 'encryption_type' => 'require|in:1,2,3', + ]; + + protected $message = [ + 'app_id.require' => '请填写AppID', + 'app_secret.require' => '请填写AppSecret', + 'encryption_type.require' => '请选择消息加密方式', + 'encryption_type.in' => '消息加密方式状态值错误', + ]; +} \ No newline at end of file diff --git a/app/admin/validate/channel/OpenSettingValidate.php b/app/admin/validate/channel/OpenSettingValidate.php new file mode 100755 index 0000000..1dae8ca --- /dev/null +++ b/app/admin/validate/channel/OpenSettingValidate.php @@ -0,0 +1,34 @@ + 'require', + 'app_secret' => 'require', + ]; + + protected $message = [ + 'app_id.require' => '请输入appId', + 'app_secret.require' => '请输入appSecret', + ]; +} \ No newline at end of file diff --git a/app/admin/validate/channel/WebPageSettingValidate.php b/app/admin/validate/channel/WebPageSettingValidate.php new file mode 100755 index 0000000..23c8833 --- /dev/null +++ b/app/admin/validate/channel/WebPageSettingValidate.php @@ -0,0 +1,33 @@ + 'require|in:0,1' + ]; + + protected $message = [ + 'status.require' => '请选择启用状态', + 'status.in' => '启用状态值有误', + ]; +} \ No newline at end of file diff --git a/app/admin/validate/crontab/CrontabValidate.php b/app/admin/validate/crontab/CrontabValidate.php new file mode 100755 index 0000000..c81a172 --- /dev/null +++ b/app/admin/validate/crontab/CrontabValidate.php @@ -0,0 +1,138 @@ + 'require', + 'type' => 'require|in:1', + 'command' => 'require', + 'status' => 'require|in:1,2,3', + 'expression' => 'require|checkExpression', + 'id' => 'require', + 'operate' => 'require' + ]; + + protected $message = [ + 'name.require' => '请输入定时任务名称', + 'type.require' => '请选择类型', + 'type.in' => '类型值错误', + 'command.require' => '请输入命令', + 'status.require' => '请选择状态', + 'status.in' => '状态值错误', + 'expression.require' => '请输入运行规则', + 'id.require' => '参数缺失', + 'operate.require' => '请选择操作', + ]; + + + /** + * @notes 添加定时任务场景 + * @return CrontabValidate + * @author 乔峰 + * @date 2022/3/29 14:39 + */ + public function sceneAdd() + { + return $this->remove('id', 'require')->remove('operate', 'require'); + } + + + /** + * @notes 查看定时任务详情场景 + * @return CrontabValidate + * @author 乔峰 + * @date 2022/3/29 14:39 + */ + public function sceneDetail() + { + return $this->only(['id']); + } + + + /** + * @notes 编辑定时任务 + * @return CrontabValidate + * @author 乔峰 + * @date 2022/3/29 14:39 + */ + public function sceneEdit() + { + return $this->remove('operate', 'require'); + } + + + /** + * @notes 删除定时任务场景 + * @return CrontabValidate + * @author 乔峰 + * @date 2022/3/29 14:40 + */ + public function sceneDelete() + { + return $this->only(['id']); + } + + + /** + * @notes CrontabValidate + * @return CrontabValidate + * @author 乔峰 + * @date 2022/3/29 14:40 + */ + public function sceneOperate() + { + return $this->only(['id', 'operate']); + } + + + /** + * @notes 获取规则执行时间场景 + * @return CrontabValidate + * @author 乔峰 + * @date 2022/3/29 14:40 + */ + public function sceneExpression() + { + return $this->only(['expression']); + } + + + /** + * @notes 校验运行规则 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @author 乔峰 + * @date 2022/3/29 14:40 + */ + public function checkExpression($value, $rule, $data) + { + if (CronExpression::isValidExpression($value) === false) { + return '定时任务运行规则错误'; + } + return true; + } +} \ No newline at end of file diff --git a/app/admin/validate/dept/DeptValidate.php b/app/admin/validate/dept/DeptValidate.php new file mode 100755 index 0000000..48327b3 --- /dev/null +++ b/app/admin/validate/dept/DeptValidate.php @@ -0,0 +1,178 @@ + 'require|checkDept', + 'pid' => 'require|integer', + 'name' => 'require|unique:'.Dept::class.'|length:1,30', + 'status' => 'require|in:0,1', + 'sort' => 'egt:0', + ]; + + + protected $message = [ + 'id.require' => '参数缺失', + 'name.require' => '请填写部门名称', + 'name.length' => '部门名称长度须在1-30位字符', + 'name.unique' => '部门名称已存在', + 'sort.egt' => '排序值不正确', + 'pid.require' => '请选择上级部门', + 'pid.integer' => '上级部门参数错误', + 'status.require' => '请选择部门状态', + ]; + + + /** + * @notes 添加场景 + * @return DeptValidate + * @author 乔峰 + * @date 2022/5/25 18:16 + */ + public function sceneAdd() + { + return $this->remove('id', true)->append('pid', 'checkDept'); + } + + + /** + * @notes 详情场景 + * @return DeptValidate + * @author 乔峰 + * @date 2022/5/25 18:16 + */ + public function sceneDetail() + { + return $this->only(['id']); + } + + + /** + * @notes 编辑场景 + * @return DeptValidate + * @author 乔峰 + * @date 2022/5/26 18:42 + */ + public function sceneEdit() + { + return $this->append('pid', 'checkPid'); + } + + + /** + * @notes 删除场景 + * @return DeptValidate + * @author 乔峰 + * @date 2022/5/25 18:16 + */ + public function sceneDelete() + { + return $this->only(['id'])->append('id', 'checkAbleDetele'); + } + + + /** + * @notes 校验部门 + * @param $value + * @return bool|string + * @author 乔峰 + * @date 2022/5/25 18:17 + */ + public function checkDept($value) + { + $dept = Dept::findOrEmpty($value); + if ($dept->isEmpty()) { + return '部门不存在'; + } + return true; + } + + + /** + * @notes 校验能否删除 + * @param $value + * @return bool|string + * @author 乔峰 + * @date 2022/5/26 14:22 + */ + public function checkAbleDetele($value) + { + $hasLower = Dept::where(['pid' => $value])->findOrEmpty(); + if (!$hasLower->isEmpty()) { + return '已关联下级部门,暂不可删除'; + } + + $check = AdminDept::where(['dept_id' => $value])->findOrEmpty(); + if (!$check->isEmpty()) { + return '已关联管理员,暂不可删除'; + } + + $dept = Dept::findOrEmpty($value); + if ($dept['pid'] == 0) { + return '顶级部门不可删除'; + } + + return true; + } + + /** + * @notes 校验部门 + * @param $value + * @param $rule + * @param array $data + * @return bool|string + * @author 乔峰 + * @date 2022/5/26 18:41 + */ + public function checkPid($value, $rule, $data = []) + { + // 当前编辑的部门id信息是否存在 + $dept = Dept::findOrEmpty($data['id']); + if ($dept->isEmpty()) { + return '当前部门信息缺失'; + } + + // 顶级部门不校验上级部门id + if ($dept['pid'] == 0) { + return true; + } + + if ($data['id'] == $value) { + return '上级部门不可是当前部门'; + } + + $leaderDept = Dept::findOrEmpty($value); + if ($leaderDept->isEmpty()) { + return '部门不存在'; + } + + return true; + } + + +} \ No newline at end of file diff --git a/app/admin/validate/dept/JobsValidate.php b/app/admin/validate/dept/JobsValidate.php new file mode 100755 index 0000000..194ab8f --- /dev/null +++ b/app/admin/validate/dept/JobsValidate.php @@ -0,0 +1,127 @@ + 'require|checkJobs', + 'name' => 'require|unique:'.Jobs::class.'|length:1,50', + 'code' => 'require|unique:'.Jobs::class, + 'status' => 'require|in:0,1', + 'sort' => 'egt:0', + ]; + + protected $message = [ + 'id.require' => '参数缺失', + 'name.require' => '请填写岗位名称', + 'name.length' => '岗位名称长度须在1-50位字符', + 'name.unique' => '岗位名称已存在', + 'code.require' => '请填写岗位编码', + 'code.unique' => '岗位编码已存在', + 'sort.egt' => '排序值不正确', + 'status.require' => '请选择岗位状态', + 'status.in' => '岗位状态值错误', + ]; + + + /** + * @notes 添加场景 + * @return JobsValidate + * @author 乔峰 + * @date 2022/5/26 9:53 + */ + public function sceneAdd() + { + return $this->remove('id', true); + } + + + /** + * @notes 详情场景 + * @return JobsValidate + * @author 乔峰 + * @date 2022/5/26 9:53 + */ + public function sceneDetail() + { + return $this->only(['id']); + } + + + public function sceneEdit() + { + } + + + /** + * @notes 删除场景 + * @return JobsValidate + * @author 乔峰 + * @date 2022/5/26 9:54 + */ + public function sceneDelete() + { + return $this->only(['id'])->append('id', 'checkAbleDetele'); + } + + + /** + * @notes 校验岗位 + * @param $value + * @return bool|string + * @author 乔峰 + * @date 2022/5/26 9:55 + */ + public function checkJobs($value) + { + $jobs = Jobs::findOrEmpty($value); + if ($jobs->isEmpty()) { + return '岗位不存在'; + } + return true; + } + + + /** + * @notes 校验能否删除 + * @param $value + * @return bool|string + * @author 乔峰 + * @date 2022/5/26 14:22 + */ + public function checkAbleDetele($value) + { + $check = AdminJobs::where(['jobs_id' => $value])->findOrEmpty(); + if (!$check->isEmpty()) { + return '已关联管理员,暂不可删除'; + } + return true; + } + +} \ No newline at end of file diff --git a/app/admin/validate/dict/DictDataValidate.php b/app/admin/validate/dict/DictDataValidate.php new file mode 100755 index 0000000..f1349c6 --- /dev/null +++ b/app/admin/validate/dict/DictDataValidate.php @@ -0,0 +1,119 @@ + 'require|checkDictData', + 'name' => 'require|length:1,255', + 'value' => 'require', + 'type_id' => 'require|checkDictType', + 'status' => 'require|in:0,1', + ]; + + + protected $message = [ + 'id.require' => '参数缺失', + 'name.require' => '请填写字典数据名称', + 'name.length' => '字典数据名称长度须在1-255位字符', + 'value.require' => '请填写字典数据值', + 'type_id.require' => '字典类型缺失', + 'status.require' => '请选择字典数据状态', + 'status.in' => '字典数据状态参数错误', + ]; + + + /** + * @notes 添加场景 + * @return DictDataValidate + * @author 乔峰 + * @date 2022/6/20 16:54 + */ + public function sceneAdd() + { + return $this->remove('id', true); + } + + + /** + * @notes ID场景 + * @return DictDataValidate + * @author 乔峰 + * @date 2022/6/20 16:54 + */ + public function sceneId() + { + return $this->only(['id']); + } + + + /** + * @notes 编辑场景 + * @return DictDataValidate + * @author 乔峰 + * @date 2022/6/20 18:36 + */ + public function sceneEdit() + { + return $this->remove('type_id', true); + } + + + /** + * @notes 校验字典数据 + * @param $value + * @return bool|string + * @author 乔峰 + * @date 2022/6/20 16:55 + */ + protected function checkDictData($value) + { + $article = DictData::findOrEmpty($value); + if ($article->isEmpty()) { + return '字典数据不存在'; + } + return true; + } + + + /** + * @notes 校验字典类型 + * @param $value + * @return bool|string + * @author 乔峰 + * @date 2022/6/20 17:03 + */ + protected function checkDictType($value) + { + $type = DictType::findOrEmpty($value); + if ($type->isEmpty()) { + return '字典类型不存在'; + } + return true; + } + +} \ No newline at end of file diff --git a/app/admin/validate/dict/DictTypeValidate.php b/app/admin/validate/dict/DictTypeValidate.php new file mode 100755 index 0000000..9cb930e --- /dev/null +++ b/app/admin/validate/dict/DictTypeValidate.php @@ -0,0 +1,133 @@ + 'require|checkDictType', + 'name' => 'require|length:1,255', + 'type' => 'require|unique:' . DictType::class, + 'status' => 'require|in:0,1', + 'remark' => 'max:200', + ]; + + + protected $message = [ + 'id.require' => '参数缺失', + 'name.require' => '请填写字典名称', + 'name.length' => '字典名称长度须在1~255位字符', + 'type.require' => '请填写字典类型', + 'type.unique' => '字典类型已存在', + 'status.require' => '请选择状态', + 'remark.max' => '备注长度不能超过200', + ]; + + + /** + * @notes 添加场景 + * @return DictTypeValidate + * @author 乔峰 + * @date 2022/6/20 16:00 + */ + public function sceneAdd() + { + return $this->remove('id', true); + } + + + /** + * @notes 详情场景 + * @return DictTypeValidate + * @author 乔峰 + * @date 2022/6/20 16:00 + */ + public function sceneDetail() + { + return $this->only(['id']); + } + + + public function sceneEdit() + { + } + + + /** + * @notes 删除场景 + * @return DictTypeValidate + * @author 乔峰 + * @date 2022/6/20 16:03 + */ + public function sceneDelete() + { + return $this->only(['id']) + ->append('id', 'checkAbleDelete'); + } + + + + /** + * @notes 检查字典类型是否存在 + * @param $value + * @return bool|string + * @author 乔峰 + * @date 2022/6/20 16:04 + */ + protected function checkDictType($value) + { + $dictType = DictType::findOrEmpty($value); + if ($dictType->isEmpty()) { + return '字典类型不存在'; + } + return true; + } + + + + /** + * @notes 验证是否可删除 + * @param $value + * @return bool|string + * @author 乔峰 + * @date 2022/6/20 16:04 + */ + protected function checkAbleDelete($value) + { + $dictData = DictData::whereIn('type_id', $value)->select(); + + foreach ($dictData as $item) { + if (!empty($item)) { + return '字典类型已被使用,请先删除绑定该字典类型的数据'; + } + } + + return true; + } + + + +} \ No newline at end of file diff --git a/app/admin/validate/notice/NoticeValidate.php b/app/admin/validate/notice/NoticeValidate.php new file mode 100755 index 0000000..2654262 --- /dev/null +++ b/app/admin/validate/notice/NoticeValidate.php @@ -0,0 +1,39 @@ + 'require', + ]; + + protected $message = [ + 'id.require' => '参数缺失', + ]; + + protected function sceneDetail() + { + return $this->only(['id']); + } + +} \ No newline at end of file diff --git a/app/admin/validate/notice/SmsConfigValidate.php b/app/admin/validate/notice/SmsConfigValidate.php new file mode 100755 index 0000000..0897c15 --- /dev/null +++ b/app/admin/validate/notice/SmsConfigValidate.php @@ -0,0 +1,51 @@ + 'require', + 'sign' => 'require', + 'app_id' => 'requireIf:type,tencent', + 'app_key' => 'requireIf:type,ali', + 'secret_id' => 'requireIf:type,tencent', + 'secret_key' => 'require', + 'status' => 'require', + ]; + + protected $message = [ + 'type.require' => '请选择类型', + 'sign.require' => '请输入签名', + 'app_id.requireIf' => '请输入app_id', + 'app_key.requireIf' => '请输入app_key', + 'secret_id.requireIf' => '请输入secret_id', + 'secret_key.require' => '请输入secret_key', + 'status.require' => '请选择状态', + ]; + + + protected function sceneDetail() + { + return $this->only(['type']); + } +} \ No newline at end of file diff --git a/app/admin/validate/setting/StorageValidate.php b/app/admin/validate/setting/StorageValidate.php new file mode 100644 index 0000000..8c1150c --- /dev/null +++ b/app/admin/validate/setting/StorageValidate.php @@ -0,0 +1,52 @@ + 'require', + 'status' => 'require', + ]; + + + + /** + * @notes 设置存储引擎参数场景 + * @return \app\admin\validate\setting\StorageValidate + * @author 乔峰 + * @date 2022/4/20 16:18 + */ + public function sceneSetup() + { + return $this->only(['engine', 'status']); + } + + + /** + * @notes 获取配置参数信息场景 + * @return StorageValidate + * @author 乔峰 + * @date 2022/4/20 16:18 + */ + public function sceneDetail() + { + return $this->only(['engine']); + } + + + /** + * @notes 切换存储引擎场景 + * @return StorageValidate + * @author 乔峰 + * @date 2022/4/20 16:18 + */ + public function sceneChange() + { + return $this->only(['engine']); + } +} \ No newline at end of file diff --git a/app/admin/validate/setting/TransactionSettingsValidate.php b/app/admin/validate/setting/TransactionSettingsValidate.php new file mode 100755 index 0000000..f03f859 --- /dev/null +++ b/app/admin/validate/setting/TransactionSettingsValidate.php @@ -0,0 +1,51 @@ + 'require|in:0,1', + 'cancel_unpaid_orders_times' => 'requireIf:cancel_unpaid_orders,1|integer|gt:0', + 'verification_orders' => 'require|in:0,1', + 'verification_orders_times' => 'requireIf:verification_orders,1|integer|gt:0', + ]; + + protected $message = [ + 'cancel_unpaid_orders.require' => '请选择系统取消待付款订单方式', + 'cancel_unpaid_orders.in' => '系统取消待付款订单状态值有误', + 'cancel_unpaid_orders_times.requireIf' => '系统取消待付款订单时间未填写', + 'cancel_unpaid_orders_times.integer' => '系统取消待付款订单时间须为整型', + 'cancel_unpaid_orders_times.gt' => '系统取消待付款订单时间须大于0', + + 'verification_orders.require' => '请选择系统自动核销订单方式', + 'verification_orders.in' => '系统自动核销订单状态值有误', + 'verification_orders_times.requireIf' => '系统自动核销订单时间未填写', + 'verification_orders_times.integer' => '系统自动核销订单时间须为整型', + 'verification_orders_times.gt' => '系统自动核销订单时间须大于0', + ]; + + public function sceneSetConfig() + { + return $this->only(['cancel_unpaid_orders','cancel_unpaid_orders_times','verification_orders','verification_orders_times']); + } +} \ No newline at end of file diff --git a/app/admin/validate/setting/UserConfigValidate.php b/app/admin/validate/setting/UserConfigValidate.php new file mode 100755 index 0000000..2d0743b --- /dev/null +++ b/app/admin/validate/setting/UserConfigValidate.php @@ -0,0 +1,58 @@ + 'requireIf:scene,register|array', + 'coerce_mobile' => 'requireIf:scene,register|in:0,1', + 'login_agreement' => 'in:0,1', + 'third_auth' => 'in:0,1', + 'wechat_auth' => 'in:0,1', + 'default_avatar' => 'require', + ]; + + + protected $message = [ + 'default_avatar.require' => '请上传用户默认头像', + 'login_way.requireIf' => '请选择登录方式', + 'login_way.array' => '登录方式值错误', + 'coerce_mobile.requireIf' => '请选择注册强制绑定手机', + 'coerce_mobile.in' => '注册强制绑定手机值错误', + 'wechat_auth.in' => '公众号微信授权登录值错误', + 'third_auth.in' => '第三方登录值错误', + 'login_agreement.in' => '政策协议值错误', + ]; + + //用户设置验证 + public function sceneUser() + { + return $this->only(['default_avatar']); + } + + //注册验证 + public function sceneRegister() + { + return $this->only(['login_way', 'coerce_mobile', 'login_agreement', 'third_auth', 'wechat_auth']); + } +} \ No newline at end of file diff --git a/app/admin/validate/setting/WebSettingValidate.php b/app/admin/validate/setting/WebSettingValidate.php new file mode 100755 index 0000000..73e47e8 --- /dev/null +++ b/app/admin/validate/setting/WebSettingValidate.php @@ -0,0 +1,50 @@ + 'require|max:30', + 'web_favicon' => 'require', + 'web_logo' => 'require', + 'login_image' => 'require', + 'shop_name' => 'require', + 'shop_logo' => 'require', + 'pc_logo' => 'require', + ]; + + protected $message = [ + 'name.require' => '请填写网站名称', + 'name.max' => '网站名称最长为12个字符', + 'web_favicon.require' => '请上传网站图标', + 'web_logo.require' => '请上传网站logo', + 'login_image.require' => '请上传登录页广告图', + 'shop_name.require' => '请填写前台名称', + 'shop_logo.require' => '请上传前台logo', + 'pc_logo.require' => '请上传PC端logo', + ]; + + protected $scene = [ + 'website' => ['name', 'web_favicon', 'web_logo', 'login_image', 'shop_name', 'shop_logo', 'pc_logo'], + ]; +} \ No newline at end of file diff --git a/app/admin/validate/tools/EditTableValidate.php b/app/admin/validate/tools/EditTableValidate.php new file mode 100644 index 0000000..5878728 --- /dev/null +++ b/app/admin/validate/tools/EditTableValidate.php @@ -0,0 +1,98 @@ + 'require|checkTableData', + 'table_name' => 'require', + 'table_comment' => 'require', + 'template_type' => 'require|in:0,1', + 'generate_type' => 'require|in:0,1', + 'module_name' => 'require', + 'table_column' => 'require|array|checkColumn', + ]; + + protected $message = [ + 'id.require' => '表id缺失', + 'table_name.require' => '请填写表名称', + 'table_comment.require' => '请填写表描述', + 'template_type.require' => '请选择模板类型', + 'template_type.in' => '模板类型参数错误', + 'generate_type.require' => '请选择生成方式', + 'generate_type.in' => '生成方式类型错误', + 'module_name.require' => '请填写模块名称', + 'table_column.require' => '表字段信息缺失', + 'table_column.array' => '表字段信息类型错误', + ]; + + + /** + * @notes 校验当前数据表是否存在 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @author 段誉 + * @date 2022/6/15 18:58 + */ + protected function checkTableData($value, $rule, $data) + { + $table = GenerateTable::findOrEmpty($value); + if ($table->isEmpty()) { + return '信息不存在'; + } + return true; + } + + + /** + * @notes 校验表字段参数 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @author 段誉 + * @date 2022/6/20 10:42 + */ + protected function checkColumn($value, $rule, $data) + { + foreach ($value as $item) { + if (!isset($item['id'])) { + return '表字段id参数缺失'; + } + if (!isset($item['query_type'])) { + return '请选择查询方式'; + } + if (!isset($item['view_type'])) { + return '请选择显示类型'; + } + } + return true; + } + +} \ No newline at end of file diff --git a/app/admin/validate/tools/GenerateTableValidate.php b/app/admin/validate/tools/GenerateTableValidate.php new file mode 100644 index 0000000..4773b36 --- /dev/null +++ b/app/admin/validate/tools/GenerateTableValidate.php @@ -0,0 +1,131 @@ + 'require|checkTableData', + 'table' => 'require|array|checkTable', + 'file' => 'require' + ]; + + protected $message = [ + 'id.require' => '参数缺失', + 'table.require' => '参数缺失', + 'table.array' => '参数类型错误', + 'file.require' => '下载失败', + ]; + + + /** + * @notes 选择数据表场景 + * @return GenerateTableValidate + * @author 段誉 + * @date 2022/6/15 18:58 + */ + public function sceneSelect() + { + return $this->only(['table']); + } + + + /** + * @notes 需要校验id的场景 + * @return GenerateTableValidate + * @author 段誉 + * @date 2022/6/15 18:58 + */ + public function sceneId() + { + return $this->only(['id']); + } + + + /** + * @notes 下载场景 + * @return GenerateTableValidate + * @author 段誉 + * @date 2022/6/24 10:02 + */ + public function sceneDownload() + { + return $this->only(['file']); + } + + + /** + * @notes 校验选择的数据表信息 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @author 段誉 + * @date 2022/6/15 18:58 + */ + protected function checkTable($value, $rule, $data) + { + foreach ($value as $item) { + if (!isset($item['name']) || !isset($item['comment'])) { + return '参数缺失'; + } + $exist = Db::query("SHOW TABLES LIKE'" . $item['name'] . "'"); + if (empty($exist)) { + return '当前数据库不存在' . $item['name'] . '表'; + } + } + return true; + } + + + /** + * @notes 校验当前数据表是否存在 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @author 段誉 + * @date 2022/6/15 18:58 + */ + protected function checkTableData($value, $rule, $data) + { + if (!is_array($value)) { + $value = [$value]; + } + + foreach ($value as $item) { + $table = GenerateTable::findOrEmpty($item); + if ($table->isEmpty()) { + return '信息不存在'; + } + } + + return true; + } + + +} \ No newline at end of file diff --git a/app/admin/validate/user/UserValidate.php b/app/admin/validate/user/UserValidate.php new file mode 100644 index 0000000..7ce67a6 --- /dev/null +++ b/app/admin/validate/user/UserValidate.php @@ -0,0 +1,110 @@ + 'require|checkUser', + 'field' => 'require|checkField', + 'value' => 'require', + ]; + + protected $message = [ + 'id.require' => '请选择用户', + 'field.require' => '请选择操作', + 'value.require' => '请输入内容', + ]; + + + /** + * @notes 详情场景 + * @return \app\admin\validate\user\UserValidate + * @author 乔峰 + * @date 2022/9/22 16:35 + */ + public function sceneDetail() + { + return $this->only(['id']); + } + + + /** + * @notes 用户信息校验 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/9/22 17:03 + */ + public function checkUser($value, $rule, $data) + { + $userIds = is_array($value) ? $value : [$value]; + + foreach ($userIds as $item) { + if (!User::find($item)) { + return '用户不存在!'; + } + } + return true; + } + + + /** + * @notes 校验是否可更新信息 + * @param $value + * @param $rule + * @param $data + * @return bool|string + * @author 乔峰 + * @date 2022/9/22 16:37 + */ + public function checkField($value, $rule, $data) + { + $allowField = ['account', 'sex', 'mobile', 'real_name']; + + if (!in_array($value, $allowField)) { + return '用户信息不允许更新'; + } + + switch ($value) { + case 'account': + //验证手机号码是否存在 + $account = User::where([ + ['id', '<>', $data['id']], + ['account', '=', $data['value']] + ])->findOrEmpty(); + + if (!$account->isEmpty()) { + return '账号已被使用'; + } + break; + + case 'mobile': + if (false == $this->validate($data['value'], 'mobile', $data)) { + return '手机号码格式错误'; + } + + //验证手机号码是否存在 + $mobile = User::where([ + ['id', '<>', $data['id']], + ['mobile', '=', $data['value']] + ])->findOrEmpty(); + + if (!$mobile->isEmpty()) { + return '手机号码已存在'; + } + break; + } + return true; + } +} \ No newline at end of file diff --git a/app/api/controller/IndexController.php b/app/api/controller/IndexController.php new file mode 100644 index 0000000..0457c30 --- /dev/null +++ b/app/api/controller/IndexController.php @@ -0,0 +1,12 @@ +getLocalIp(); + $this->key = $this->tagName . $ip; + } + + /** + * @notes 记录登录错误次数 + * @author 令狐冲 + * @date 2021/6/30 01:51 + */ + public function record() + { + if (Cache::get($this->key)) { + //缓存存在,记录错误次数 + Cache::set($this->key, Cache::get($this->key) + 1); + } else { + //缓存不存在,第一次设置缓存 + Cache::set($this->key, 1, $this->minute * 60); + } + } + + /** + * @notes 判断是否安全 + * @return bool + * @author 令狐冲 + * @date 2021/6/30 01:53 + */ + public function isSafe() + { + $count = Cache::get($this->key); + if ($count >= $this->count) { + return false; + } + return true; + } + + /** + * @notes 删除该ip记录错误次数 + * @author 令狐冲 + * @date 2021/6/30 01:55 + */ + public function relieve() + { + Cache::delete($this->key); + } +} \ No newline at end of file diff --git a/app/common/cache/AdminAuthCache.php b/app/common/cache/AdminAuthCache.php new file mode 100644 index 0000000..8dab287 --- /dev/null +++ b/app/common/cache/AdminAuthCache.php @@ -0,0 +1,104 @@ +adminId = $adminId; + // 全部权限 + $this->authConfigList = AuthLogic::getAllAuth(); + // 当前权限配置文件的md5 + $this->authMd5 = md5(json_encode($this->authConfigList)); + + $this->cacheMd5Key = $this->prefix . 'md5'; + $this->cacheAllKey = $this->prefix . 'all'; + $this->cacheUrlKey = $this->prefix . 'url_' . $this->adminId; + + $cacheAuthMd5 = Cache::get($this->cacheMd5Key); + $cacheAuth = Cache::get($this->cacheAllKey); + //权限配置和缓存权限对比,不一样说明权限配置文件已修改,清理缓存 + if ($this->authMd5 !== $cacheAuthMd5 || empty($cacheAuth)) { + $this->deleteTag(); + } + } + + + /** + * @notes 获取管理权限uri + * @param $adminId + * @return array|mixed + * @author 令狐冲 + * @date 2021/8/19 15:27 + */ + public function getAdminUri() + { + //从缓存获取,直接返回 + $urisAuth = Cache::get($this->cacheUrlKey); + if ($urisAuth) { + return $urisAuth; + } + + //获取角色关联的菜单id(菜单或权限) + $urisAuth = AuthLogic::getAuthByAdminId($this->adminId); + if (empty($urisAuth)) { + return []; + } + + Cache::set($this->cacheUrlKey, $urisAuth, 3600); + + //保存到缓存并读取返回 + return $urisAuth; + } + + + /** + * @notes 获取全部权限uri + * @return array|mixed + * @author cjhao + * @date 2021/9/13 11:41 + */ + public function getAllUri() + { + $cacheAuth = Cache::get($this->cacheAllKey); + if ($cacheAuth) { + return $cacheAuth; + } + // 获取全部权限 + $authList = AuthLogic::getAllAuth(); + //保存到缓存并读取返回 + $this->set($this->cacheMd5Key, $this->authMd5); + $this->set($this->cacheAllKey, $authList); + return $authList; + } + + + /** + * @notes 清理管理员缓存 + * @return bool + * @author cjhao + * @date 2021/10/13 18:47 + */ + public function clearAuthCache() + { + Cache::clear($this->cacheUrlKey); + return true; + } +} \ No newline at end of file diff --git a/app/common/cache/AdminTokenCache.php b/app/common/cache/AdminTokenCache.php new file mode 100644 index 0000000..6a630e1 --- /dev/null +++ b/app/common/cache/AdminTokenCache.php @@ -0,0 +1,100 @@ +prefix . $token); + if ($adminInfo) { + return $adminInfo; + } + + //从数据获取信息被设置缓存(可能后台清除缓存) + $adminInfo = $this->setAdminInfo($token); + if ($adminInfo) { + return $adminInfo; + } + + return false; + } + + /** + * @notes 通过有效token设置管理信息缓存 + * @param $token + * @return array|false|mixed + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 令狐冲 + * @date 2021/7/5 12:12 + */ + public function setAdminInfo($token) + { + $adminSession = AdminSession::where([['token', '=', $token], ['expire_time', '>', time()]]) + ->find(); + if (empty($adminSession)) { + return []; + } + $admin = Admin::where('id', '=', $adminSession->admin_id) + ->append(['role_id']) + ->find(); + + $roleName = ''; + $roleLists = SystemRole::column('name', 'id'); + if ($admin['root'] == 1) { + $roleName = '系统管理员'; + } else { + foreach ($admin['role_id'] as $roleId) { + $roleName .= $roleLists[$roleId] ?? ''; + $roleName .= '/'; + } + $roleName = trim($roleName, '/'); + } + + $adminInfo = [ + 'admin_id' => $admin->id, + 'root' => $admin->root, + 'name' => $admin->name, + 'account' => $admin->account, + 'role_name' => $roleName, + 'role_id' => $admin->role_id, + 'token' => $token, + 'terminal' => $adminSession->terminal, + 'expire_time' => $adminSession->expire_time, + ]; + Cache::set($this->prefix . $token, $adminInfo); + return $this->getAdminInfo($token); + } + + /** + * @notes 删除缓存 + * @param $token + * @return bool + * @author 令狐冲 + * @date 2021/7/3 16:57 + */ + public function deleteAdminInfo($token) + { + return Cache::delete($this->prefix . $token); + } +} \ No newline at end of file diff --git a/app/common/cache/BaseCache.php b/app/common/cache/BaseCache.php new file mode 100644 index 0000000..43e1d02 --- /dev/null +++ b/app/common/cache/BaseCache.php @@ -0,0 +1,47 @@ +tagName = get_class($this); + } + + + /** + * @notes 重写父类set,自动打上标签 + * @param string $key + * @param mixed $value + * @param null $ttl + * @return bool + * @author 乔峰 + * @date 2021/12/27 14:16 + */ + public function set($key, $value, $ttl = null): bool + { + return Cache::set($key, $value, $ttl); + } + + + /** + * @notes 清除缓存类所有缓存 + * @return bool + * @author 乔峰 + * @date 2021/12/27 14:16 + */ + public function deleteTag(): bool + { + return Cache::delete($this->tagName); + } +} \ No newline at end of file diff --git a/app/common/cache/ExportCache.php b/app/common/cache/ExportCache.php new file mode 100755 index 0000000..9ef56e3 --- /dev/null +++ b/app/common/cache/ExportCache.php @@ -0,0 +1,70 @@ +uniqid = md5(uniqid($this->tagName,true).mt_rand()); + } + + /** + * @notes 获取excel缓存目录 + * @return string + * @author 令狐冲 + * @date 2021/7/28 17:36 + */ + public function getSrc() + { + return public_path() . '/export/'.date('Y-m').'/'.$this->uniqid.'/'; + } + + + /** + * @notes 设置文件路径缓存地址 + * @param $fileName + * @return string + * @author 令狐冲 + * @date 2021/7/28 17:36 + */ + public function setFile($fileName) + { + $src = $this->getSrc(); + $key = md5($src . $fileName) . time(); + Cache::set($key, ['src' => $src, 'name' => $fileName], 300); + return $key; + } + + /** + * @notes 获取文件缓存的路径 + * @param $key + * @return mixed + * @author 令狐冲 + * @date 2021/7/26 18:36 + */ + public function getFile($key) + { + return Cache::get($key); + } + +} \ No newline at end of file diff --git a/app/common/controller/BaseLikeAdminController.php b/app/common/controller/BaseLikeAdminController.php new file mode 100644 index 0000000..0574905 --- /dev/null +++ b/app/common/controller/BaseLikeAdminController.php @@ -0,0 +1,96 @@ +request->controller() . ucwords($this->request->action())); +// $lists = invoke($listName); + } + return JsonService::dataLists($lists); + } + + + /** + * @notes 操作失败 + * @param string $msg + * @param array $data + * @param int $code + * @param int $show + * @author 乔峰 + * @date 2021/12/27 14:21 + */ + protected function fail(string $msg = 'fail', array $data = [], int $code = 0, int $show = 1) + { + return JsonService::fail($msg, $data, $code, $show); + } + + + + /** + * @notes 是否免登录验证 + * @return bool + * @author 乔峰 + * @date 2021/12/27 14:21 + */ + public function isNotNeedLogin() : bool + { + $notNeedLogin = $this->notNeedLogin; + if (empty($notNeedLogin)) { + return false; + } + $action = $this->request->action; + + if (!in_array(trim($action), $notNeedLogin)) { + return false; + } + return true; + } +} \ No newline at end of file diff --git a/app/common/enum/AdminTerminalEnum.php b/app/common/enum/AdminTerminalEnum.php new file mode 100755 index 0000000..e3dac5d --- /dev/null +++ b/app/common/enum/AdminTerminalEnum.php @@ -0,0 +1,28 @@ + '单表(增删改查)', + self::TEMPLATE_TYPE_TREE => '树表(增删改查)', + ]; + if ($value === true) { + return $data; + } + return $data[$value] ?? ''; + } +} \ No newline at end of file diff --git a/app/common/enum/LoginEnum.php b/app/common/enum/LoginEnum.php new file mode 100644 index 0000000..b2d2e09 --- /dev/null +++ b/app/common/enum/LoginEnum.php @@ -0,0 +1,18 @@ + '禁用', + self::NO => '正常' + ]; + if ($value === true) { + return $data; + } + return $data[$value]; + } +} \ No newline at end of file diff --git a/app/common/enum/user/UserEnum.php b/app/common/enum/user/UserEnum.php new file mode 100755 index 0000000..ee7b8bd --- /dev/null +++ b/app/common/enum/user/UserEnum.php @@ -0,0 +1,55 @@ + '未知', + self::SEX_MEN => '男', + self::SEX_WOMAN => '女', + ]; + if (true === $from) { + return $desc; + } + return $desc[$from] ?? ''; + } +} \ No newline at end of file diff --git a/app/common/enum/user/UserTerminalEnum.php b/app/common/enum/user/UserTerminalEnum.php new file mode 100755 index 0000000..d6327a0 --- /dev/null +++ b/app/common/enum/user/UserTerminalEnum.php @@ -0,0 +1,64 @@ + '微信小程序', + self::WECHAT_OA => '微信公众号', + self::H5 => '手机H5', + self::PC => '电脑PC', + self::IOS => '苹果APP', + self::ANDROID => '安卓APP', + ]; + if(true === $from){ + return $desc; + } + return $desc[$from] ?? ''; + } +} \ No newline at end of file diff --git a/app/common/exception/ControllerExtendException.php b/app/common/exception/ControllerExtendException.php new file mode 100644 index 0000000..7ac5546 --- /dev/null +++ b/app/common/exception/ControllerExtendException.php @@ -0,0 +1,14 @@ +message = '控制器需要继承模块的基础控制器:' . $message; + $this->model = $model; + } +} \ No newline at end of file diff --git a/app/common/exception/Handler.php b/app/common/exception/Handler.php new file mode 100644 index 0000000..a4955cd --- /dev/null +++ b/app/common/exception/Handler.php @@ -0,0 +1,40 @@ +getResponse(); + } + if ($request->expectsJson()) { + if (!$this->_debug) { + $json = ['code' => 500, 'msg' => '服务器错误!']; + return new Response(200, ['Content-Type' => 'application/json'], + json_encode($json, JSON_UNESCAPED_UNICODE | JSON_PRETTY_PRINT | JSON_UNESCAPED_SLASHES)); + } + } + return parent::render($request, $exception); + } +} \ No newline at end of file diff --git a/app/common/exception/HttpException.php b/app/common/exception/HttpException.php new file mode 100644 index 0000000..6f03df6 --- /dev/null +++ b/app/common/exception/HttpException.php @@ -0,0 +1,21 @@ +response = json(json_decode($message,true)); + parent::__construct($message, $code, $previous); + } + + public function getResponse(){ + return $this->response; + } +} \ No newline at end of file diff --git a/app/common/http/middleware/AdminAllowMiddleware.php b/app/common/http/middleware/AdminAllowMiddleware.php new file mode 100755 index 0000000..efe3263 --- /dev/null +++ b/app/common/http/middleware/AdminAllowMiddleware.php @@ -0,0 +1,54 @@ +method() == 'OPTIONS' ? response('') : $handler($request); + + // 给响应添加跨域相关的http头 + $response->withHeaders([ + 'Access-Control-Allow-Credentials' => 'true', + 'Access-Control-Allow-Origin' => $request->header('origin', '*'), + 'Access-Control-Allow-Methods' => $request->header('access-control-request-method', '*'), + 'Access-Control-Allow-Headers' => $request->header('access-control-request-headers', '*'), + ]); + + return $response; + } + +} \ No newline at end of file diff --git a/app/common/http/middleware/BaseMiddleware.php b/app/common/http/middleware/BaseMiddleware.php new file mode 100755 index 0000000..df8205b --- /dev/null +++ b/app/common/http/middleware/BaseMiddleware.php @@ -0,0 +1,30 @@ +get()->goCheck(); + + //请求参数设置 + $this->request = request(); + $this->params = $this->request->all(); + + //分页初始化 + $this->initPage(); + + //搜索初始化 + $this->initSearch(); + + //排序初始化 + $this->initSort(); + + //导出初始化 + $this->initExport(); + } + + /** + * @notes 分页参数初始化 + * @author 令狐冲 + * @date 2021/7/30 23:55 + */ + private function initPage() + { + $this->pageSizeMax = Config::get('project.lists.page_size_max'); + $this->pageSize = Config::get('project.lists.page_size'); + $this->pageType = $this->request->get('page_type', 1); + + if ($this->pageType == 1) { + //分页 + $this->pageNo = $this->request->get('page_no', 1) ?: 1; + $this->pageSize = $this->request->get('page_size', $this->pageSize) ?: $this->pageSize; + } else { + //不分页 + $this->pageNo = 1;//强制到第一页 + $this->pageSize = $this->pageSizeMax;// 直接取最大记录数 + } + + //limit查询参数设置 + $this->limitOffset = ($this->pageNo - 1) * $this->pageSize; + $this->limitLength = $this->pageSize; + } + + /** + * @notes 初始化搜索 + * @return array + * @author 令狐冲 + * @date 2021/7/31 00:00 + */ + private function initSearch() + { + if (!($this instanceof ListsSearchInterface)) { + return []; + } + $startTime = $this->request->get('start_time'); + $this->startTime = strtotime($startTime); + $endTime = $this->request->get('end_time'); + $this->endTime = strtotime($endTime); + + $this->start = $this->request->get('start'); + $this->end = $this->request->get('end'); + + return $this->searchWhere = $this->createWhere($this->setSearch()); + } + + /** + * @notes 初始化排序 + * @return array|string[] + * @author 令狐冲 + * @date 2021/7/31 00:03 + */ + private function initSort() + { + if (!($this instanceof ListsSortInterface)) { + return []; + } + + $this->field = $this->request->get('field', ''); + $this->orderBy = $this->request->get('order_by', ''); + + return $this->sortOrder = $this->createOrder($this->setSortFields(), $this->setDefaultOrder()); + } + + /** + * @notes 导出初始化 + * @return false|\think\response\Json + * @author 令狐冲 + * @date 2021/7/31 01:15 + */ + private function initExport() + { + $this->export = $this->request->get('export', ''); + + //不做导出操作 + if ($this->export != ExportEnum::INFO && $this->export != ExportEnum::EXPORT) { + return false; + } + + //导出操作,但是没有实现导出接口 + if (!($this instanceof ListsExcelInterface)) { + return JsonService::throw('该列表不支持导出'); + } + + $this->fileName = $this->request->get('file_name', '') ?: $this->setFileName(); + + //不导出文件,不初始化一下参数 + if ($this->export != ExportEnum::EXPORT) { + return false; + } + + //导出文件名设置 + $this->fileName .= '-' . date('Y-m-d-His') . '.xlsx'; + + //导出文件准备 + if ($this->export == ExportEnum::EXPORT) { + //指定导出范围(例:第2页到,第5页的数据) + if ($this->pageType == 1) { + $this->pageStart = $this->request->get('page_start', $this->pageStart); + $this->pageEnd = $this->request->get('page_end', $this->pageEnd); + //改变查询数量参数(例:第2页到,第5页的数据,查询->page(2,(5-2+1)*25) + $this->limitOffset = ($this->pageStart - 1) * $this->pageSize; + $this->limitLength = ($this->pageEnd - $this->pageStart + 1) * $this->pageSize; + } + + $count = $this->count(); + + //判断导出范围是否有数据 + if ($count == 0 || ceil($count / $this->pageSize) < $this->pageStart) { + $msg = $this->pageType ? '第' . $this->pageStart . '页到第' . $this->pageEnd . '页没有数据,无法导出' : '没有数据,无法导出'; + return JsonService::throw($msg); + } + } + } +} \ No newline at end of file diff --git a/app/common/lists/ListsExcelInterface.php b/app/common/lists/ListsExcelInterface.php new file mode 100644 index 0000000..f4b595f --- /dev/null +++ b/app/common/lists/ListsExcelInterface.php @@ -0,0 +1,25 @@ + $excelField) { + $fieldData = $row[$key]; + if (is_numeric($fieldData) && strlen($fieldData) >= 12) { + $fieldData .= "\t"; + } + $temp[$key] = $fieldData; + } + $data[] = $temp; + } + $spreadsheet = new Spreadsheet(); + $sheet = $spreadsheet->getActiveSheet(); + + //设置单元格内容 + foreach ($title as $key => $value) { + // 单元格内容写入 + $sheet->setCellValueByColumnAndRow($key + 1, 1, $value); + } + $row = 2; //从第二行开始 + foreach ($data as $item) { + $column = 1; + foreach ($item as $value) { + //单元格内容写入 + $sheet->setCellValueByColumnAndRow($column, $row, $value); + $column++; + } + $row++; + } + + $getHighestRowAndColumn = $sheet->getHighestRowAndColumn(); + $HighestRow = $getHighestRowAndColumn['row']; + $column = $getHighestRowAndColumn['column']; + $titleScope = 'A1:' . $column . '1';//第一(标题)范围(例:A1:D1) + + $sheet->getStyle($titleScope) + ->getFill() + ->setFillType(Fill::FILL_SOLID) // 设置填充样式 + ->getStartColor() + ->setARGB('00B0F0'); + // 设置文字颜色为白色 + $sheet->getStyle($titleScope)->getFont()->getColor() + ->setARGB('FFFFFF'); + +// $sheet->getStyle('B2')->getNumberFormat()->setFormatCode(NumberFormat::FORMAT_DATE_YYYYMMDD); + $spreadsheet->getActiveSheet()->getColumnDimension('B')->setAutoSize(true); + + $allCope = 'A1:' . $column . $HighestRow;//整个表格范围(例:A1:D5) + $sheet->getStyle($allCope)->getBorders()->getAllBorders()->setBorderStyle(Border::BORDER_THIN); + + $writer = IOFactory::createWriter($spreadsheet, 'Xlsx'); + + //创建excel文件 + $exportCache = new ExportCache(); + $src = $exportCache->getSrc(); + + if (!file_exists($src)) { + mkdir($src, 0775, true); + } + $writer->save($src . $this->fileName); + //设置本地excel缓存并返回下载地址 + return 'http://'.request()->host().'/admin/download/export?file='.$exportCache->setFile($this->fileName); + } + /** + * @notes 获取导出信息 + * @return array + * @author 令狐冲 + * @date 2021/7/29 16:08 + */ + public function excelInfo() + { + $count = $this->count(); + $sum_page = max(ceil($count / $this->pageSize), 1); + return [ + 'count' => $count, //所有数据记录数 + 'page_size' => $this->pageSize,//每页记录数 + 'sum_page' => $sum_page,//一共多少页 + 'max_page' => floor($this->pageSizeMax / $this->pageSize),//最多导出多少页 + 'all_max_size' => $this->pageSizeMax,//最多导出记录数 + 'page_start' => $this->pageStart,//导出范围页码开始值 + 'page_end' => min($sum_page, $this->pageEnd),//导出范围页码结束值 + 'file_name' => $this->fileName,//默认文件名 + ]; + } +} \ No newline at end of file diff --git a/app/common/lists/ListsExtendInterface.php b/app/common/lists/ListsExtendInterface.php new file mode 100644 index 0000000..8c8b1c4 --- /dev/null +++ b/app/common/lists/ListsExtendInterface.php @@ -0,0 +1,16 @@ + $whereFields) { + switch ($whereType) { + case '=': + case '<>': + case '>': + case '>=': + case '<': + case '<=': + case 'in': + foreach ($whereFields as $whereField) { + $paramsName = substr_symbol_behind($whereField); + if (!isset($this->params[$paramsName]) || $this->params[$paramsName] == '') { + continue; + } + $where[] = [$whereField, $whereType, $this->params[$paramsName]]; + } + break; + case '%like%': + foreach ($whereFields as $whereField) { + $paramsName = substr_symbol_behind($whereField); + if (!isset($this->params[$paramsName]) || empty($this->params[$paramsName])) { + continue; + } + $where[] = [$whereField, 'like', '%' . $this->params[$paramsName] . '%']; + } + break; + case '%like': + foreach ($whereFields as $whereField) { + $paramsName = substr_symbol_behind($whereField); + if (!isset($this->params[$paramsName]) || empty($this->params[$paramsName])) { + continue; + } + $where[] = [$whereField, 'like', '%' . $this->params[$paramsName]]; + } + break; + case 'like%': + foreach ($whereFields as $whereField) { + $paramsName = substr_symbol_behind($whereField); + if (!isset($this->params[$paramsName]) || empty($this->params[$paramsName])) { + continue; + } + $where[] = [$whereField, 'like', $this->params[$paramsName]]; + } + break; + case 'between_time': + if (!is_numeric($this->startTime) || !is_numeric($this->endTime)) { + break; + } + $where[] = [$whereFields, 'between', [$this->startTime, $this->endTime]]; + break; + case 'between': + if (empty($this->start) || empty($this->end)) { + break; + } + $where[] = [$whereFields, 'between', [$this->start, $this->end]]; + break; + case 'find_in_set': // find_in_set查询 + foreach ($whereFields as $whereField) { + $paramsName = substr_symbol_behind($whereField); + if (!isset($this->params[$paramsName]) || $this->params[$paramsName] == '') { + continue; + } + $where[] = [$whereField, 'find in set', $this->params[$paramsName]]; + } + break; + } + } + return $where; + } +} \ No newline at end of file diff --git a/app/common/lists/ListsSortInterface.php b/app/common/lists/ListsSortInterface.php new file mode 100644 index 0000000..a6c7b29 --- /dev/null +++ b/app/common/lists/ListsSortInterface.php @@ -0,0 +1,24 @@ +orderBy) || empty($this->field) || !in_array($this->field, array_keys($sortField))) { + return $defaultOrder; + } + + if (isset($sortField[$this->field])) { + $field = $sortField[$this->field]; + } else { + return $defaultOrder; + } + + if ($this->orderBy = 'desc') { + return [$field => 'desc']; + } + if ($this->orderBy = 'asc') { + return [$field => 'asc']; + } + return $defaultOrder; + } +} \ No newline at end of file diff --git a/app/common/logic/BaseLogic.php b/app/common/logic/BaseLogic.php new file mode 100755 index 0000000..959987f --- /dev/null +++ b/app/common/logic/BaseLogic.php @@ -0,0 +1,114 @@ +findOrEmpty()->toArray(); + if (empty($noticeSetting)) { + throw new \Exception('找不到对应场景的配置'); + } + // 合并额外参数 + $params = self::mergeParams($params); + $res = false; + self::setError('发送通知失败'); + + // 短信通知 + if (isset($noticeSetting['sms_notice']['status']) && $noticeSetting['sms_notice']['status'] == YesNoEnum::YES) { + $res = (new SmsMessageService())->send($params); + } + + return $res; + } catch (\Exception $e) { + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 整理参数 + * @param $params + * @return array + * @author 乔峰 + * @date 2022/9/15 15:28 + */ + public static function mergeParams($params) + { + // 用户相关 + if (!empty($params['params']['user_id'])) { + $user = User::findOrEmpty($params['params']['user_id'])->toArray(); + $params['params']['nickname'] = $user['nickname']; + $params['params']['user_name'] = $user['nickname']; + $params['params']['user_sn'] = $user['sn']; + $params['params']['mobile'] = $params['params']['mobile'] ?? $user['mobile']; + } + + // 跳转路径 + $jumpPath = self::getPathByScene($params['scene_id'], $params['params']['order_id'] ?? 0); + $params['url'] = $jumpPath['url']; + $params['page'] = $jumpPath['page']; + + return $params; + } + + + /** + * @notes 根据场景获取跳转链接 + * @param $sceneId + * @param $extraId + * @return string[] + * @author 乔峰 + * @date 2022/9/15 15:29 + */ + public static function getPathByScene($sceneId, $extraId) + { + // 小程序主页路径 + $page = '/pages/index/index'; + // 公众号主页路径 + $url = '/mobile/pages/index/index'; + return [ + 'url' => $url, + 'page' => $page, + ]; + } + + + /** + * @notes 替换消息内容中的变量占位符 + * @param $content + * @param $params + * @return array|mixed|string|string[] + * @author 乔峰 + * @date 2022/9/15 15:29 + */ + public static function contentFormat($content, $params) + { + foreach ($params['params'] as $k => $v) { + $search = '{' . $k . '}'; + $content = str_replace($search, $v, $content); + } + return $content; + } + + + /** + * @notes 添加通知记录 + * @param $params + * @param $noticeSetting + * @param $sendType + * @param $content + * @param string $extra + * @return NoticeRecord|\think\Model + * @author 乔峰 + * @date 2022/9/15 15:29 + */ + public static function addNotice($params, $noticeSetting, $sendType, $content, $extra = '') + { + return NoticeRecord::create([ + 'user_id' => $params['params']['user_id'] ?? 0, + 'title' => self::getTitleByScene($sendType, $noticeSetting), + 'content' => $content, + 'scene_id' => $noticeSetting['scene_id'], + 'read' => YesNoEnum::NO, + 'recipient' => $noticeSetting['recipient'], + 'send_type' => $sendType, + 'notice_type' => $noticeSetting['type'], + 'extra' => $extra, + ]); + } + + + /** + * @notes 通知记录标题 + * @param $sendType + * @param $noticeSetting + * @return string + * @author 乔峰 + * @date 2022/9/15 15:30 + */ + public static function getTitleByScene($sendType, $noticeSetting) + { + switch ($sendType) { + case NoticeEnum::SMS: + $title = ''; + break; + case NoticeEnum::OA: + $title = $noticeSetting['oa_notice']['name'] ?? ''; + break; + case NoticeEnum::MNP: + $title = $noticeSetting['mnp_notice']['name'] ?? ''; + break; + default: + $title = ''; + } + return $title; + } + +} \ No newline at end of file diff --git a/app/common/model/BaseModel.php b/app/common/model/BaseModel.php new file mode 100644 index 0000000..786b24c --- /dev/null +++ b/app/common/model/BaseModel.php @@ -0,0 +1,35 @@ + '定时任务', + CrontabEnum::DAEMON => '守护进程', + ]; + return $desc[$value] ?? ''; + } + + + /** + * @notes 状态获取器 + * @param $value + * @return string + * @author 乔峰 + * @date 2022/3/29 12:06 + */ + public function getStatusDescAttr($value) + { + $desc = [ + CrontabEnum::START => '运行', + CrontabEnum::STOP => '停止', + CrontabEnum::ERROR => '错误', + ]; + return $desc[$value] ?? ''; + } + + + /** + * @notes 最后执行时间获取器 + * @param $value + * @return string + * @author 乔峰 + * @date 2022/3/29 12:06 + */ + public function getLastTimeAttr($value) + { + return empty($value) ? '' : date('Y-m-d H:i:s', $value); + } +} \ No newline at end of file diff --git a/app/common/model/OperationLog.php b/app/common/model/OperationLog.php new file mode 100755 index 0000000..2cfe516 --- /dev/null +++ b/app/common/model/OperationLog.php @@ -0,0 +1,9 @@ +column('role_id'); + } + + + /** + * @notes 关联部门id + * @param $value + * @param $data + * @return array + * @author 乔峰 + * @date 2022/11/25 15:00 + */ + public function getDeptIdAttr($value, $data) + { + return AdminDept::where('admin_id', $data['id'])->column('dept_id'); + } + + + /** + * @notes 关联岗位id + * @param $value + * @param $data + * @return array + * @author 乔峰 + * @date 2022/11/25 15:01\ + */ + public function getJobsIdAttr($value, $data) + { + return AdminJobs::where('admin_id', $data['id'])->column('jobs_id'); + } + + + + /** + * @notes 获取禁用状态 + * @param $value + * @param $data + * @return string|string[] + * @author 令狐冲 + * @date 2021/7/7 01:25 + */ + public function getDisableDescAttr($value, $data) + { + return YesNoEnum::getDisableDesc($data['disable']); + } + + /** + * @notes 最后登录时间获取器 - 格式化:年-月-日 时:分:秒 + * @param $value + * @return string + * @author Tab + * @date 2021/7/13 11:35 + */ + public function getLoginTimeAttr($value) + { + return empty($value) ? '' : date('Y-m-d H:i:s', $value); + } + + /** + * @notes 头像获取器 - 头像路径添加域名 + * @param $value + * @return string + * @author Tab + * @date 2021/7/13 11:35 + */ + public function getAvatarAttr($value) + { + return empty($value) ? FileService::getFileUrl(config('project.default_image.admin_avatar')) : FileService::getFileUrl(trim($value, '/')); + } + +} \ No newline at end of file diff --git a/app/common/model/auth/AdminDept.php b/app/common/model/auth/AdminDept.php new file mode 100755 index 0000000..a6e7a73 --- /dev/null +++ b/app/common/model/auth/AdminDept.php @@ -0,0 +1,32 @@ + $adminId])->delete(); + } +} \ No newline at end of file diff --git a/app/common/model/auth/AdminJobs.php b/app/common/model/auth/AdminJobs.php new file mode 100755 index 0000000..240b11c --- /dev/null +++ b/app/common/model/auth/AdminJobs.php @@ -0,0 +1,32 @@ + $adminId])->delete(); + } +} \ No newline at end of file diff --git a/app/common/model/auth/AdminRole.php b/app/common/model/auth/AdminRole.php new file mode 100755 index 0000000..2d0afc4 --- /dev/null +++ b/app/common/model/auth/AdminRole.php @@ -0,0 +1,34 @@ + $adminId])->delete(); + } + +} \ No newline at end of file diff --git a/app/common/model/auth/AdminSession.php b/app/common/model/auth/AdminSession.php new file mode 100755 index 0000000..9f18a36 --- /dev/null +++ b/app/common/model/auth/AdminSession.php @@ -0,0 +1,32 @@ +hasOne(Admin::class, 'id', 'admin_id') + ->field('id,multipoint_login'); + } +} \ No newline at end of file diff --git a/app/common/model/auth/SystemMenu.php b/app/common/model/auth/SystemMenu.php new file mode 100755 index 0000000..7d94bbf --- /dev/null +++ b/app/common/model/auth/SystemMenu.php @@ -0,0 +1,31 @@ +hasMany(SystemRoleMenu::class, 'role_id'); + } +} \ No newline at end of file diff --git a/app/common/model/auth/SystemRoleMenu.php b/app/common/model/auth/SystemRoleMenu.php new file mode 100755 index 0000000..f5e0d03 --- /dev/null +++ b/app/common/model/auth/SystemRoleMenu.php @@ -0,0 +1,30 @@ +belongsTo(GenerateTable::class, 'id', 'table_id'); + } +} \ No newline at end of file diff --git a/app/common/model/tools/GenerateTable.php b/app/common/model/tools/GenerateTable.php new file mode 100644 index 0000000..181bbe4 --- /dev/null +++ b/app/common/model/tools/GenerateTable.php @@ -0,0 +1,59 @@ +hasMany(GenerateColumn::class, 'table_id', 'id'); + } + + /** + * @notes 模板类型描述 + * @param $value + * @param $data + * @return string|string[] + * @author 段誉 + * @date 2022/6/14 11:25 + */ + public function getTemplateTypeDescAttr($value, $data) + { + return GeneratorEnum::getTemplateTypeDesc($data['template_type']); + } + + + +} \ No newline at end of file diff --git a/app/common/model/user/User.php b/app/common/model/user/User.php new file mode 100644 index 0000000..8c5bad3 --- /dev/null +++ b/app/common/model/user/User.php @@ -0,0 +1,162 @@ +hasOne(UserAuth::class, 'user_id'); + } + + + /** + * @notes 搜索器-用户信息 + * @param $query + * @param $value + * @param $data + * @author 乔峰 + * @date 2022/9/22 16:12 + */ + public function searchKeywordAttr($query, $value, $data) + { + if ($value) { + $query->where('sn|nickname|mobile', 'like', '%' . $value . '%'); + } + } + + + /** + * @notes 搜索器-注册来源 + * @param $query + * @param $value + * @param $data + * @author 乔峰 + * @date 2022/9/22 16:13 + */ + public function searchChannelAttr($query, $value, $data) + { + if ($value) { + $query->where('channel', '=', $value); + } + } + + + /** + * @notes 搜索器-注册时间 + * @param $query + * @param $value + * @param $data + * @author 乔峰 + * @date 2022/9/22 16:13 + */ + public function searchCreateTimeStartAttr($query, $value, $data) + { + if ($value) { + $query->where('create_time', '>=', $value); + } + } + + + /** + * @notes 搜索器-注册时间 + * @param $query + * @param $value + * @param $data + * @author 乔峰 + * @date 2022/9/22 16:13 + */ + public function searchCreateTimeEndAttr($query, $value, $data) + { + if ($value) { + $query->where('create_time', '<=', $value); + } + } + + + /** + * @notes 头像获取器 - 用于头像地址拼接域名 + * @param $value + * @return string + * @author Tab + * @date 2021/7/17 14:28 + */ + public function getAvatarAttr($value) + { + return trim($value) ? FileService::getFileUrl($value) : ''; + } + + + /** + * @notes 获取器-性别描述 + * @param $value + * @param $data + * @return string|string[] + * @author 乔峰 + * @date 2022/9/7 15:15 + */ + public function getSexAttr($value, $data) + { + return UserEnum::getSexDesc($value); + } + + + /** + * @notes 登录时间 + * @param $value + * @return string + * @author 乔峰 + * @date 2022/9/23 18:15 + */ + public function getLoginTimeAttr($value) + { + return $value ? date('Y-m-d H:i:s', $value) : ''; + } + + /** + * @notes 生成用户编码 + * @param string $prefix + * @param int $length + * @return string + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + * @author 乔峰 + * @date 2022/9/16 10:33 + */ + public static function createUserSn($prefix = '', $length = 8) + { + $rand_str = ''; + for ($i = 0; $i < $length; $i++) { + $rand_str .= mt_rand(0, 9); + } + $sn = $prefix . $rand_str; + if (User::where(['sn' => $sn])->find()) { + return self::createUserSn($prefix, $length); + } + return $sn; + } +} \ No newline at end of file diff --git a/app/common/model/user/UserAuth.php b/app/common/model/user/UserAuth.php new file mode 100644 index 0000000..bbc719a --- /dev/null +++ b/app/common/model/user/UserAuth.php @@ -0,0 +1,17 @@ + $type, 'name' => $name])->findOrEmpty(); + + if ($data->isEmpty()) { + Config::create([ + 'type' => $type, + 'name' => $name, + 'value' => $value, + ]); + } else { + $data->value = $value; + $data->save(); + } + + // 返回原始值 + return $original; + } + + /** + * @notes 获取配置值 + * @param $type + * @param string $name + * @param null $default_value + * @return array|int|mixed|string + * @author Tab + * @date 2021/7/15 15:16 + */ + public static function get(string $type, string $name = '', $default_value = null) + { + if (!empty($name)) { + $value = Config::where(['type' => $type, 'name' => $name])->value('value'); + if (!is_null($value)) { + $json = json_decode($value, true); + $value = json_last_error() === JSON_ERROR_NONE ? $json : $value; + } + if ($value) { + return $value; + } + // 返回特殊值 0 '0' + if ($value === 0 || $value === '0') { + return $value; + } + // 返回默认值 + if ($default_value !== null) { + return $default_value; + } + // 返回本地配置文件中的值 + return config('project.' . $type . '.' . $name); + } + + // 取某个类型下的所有name的值 + $data = Config::where(['type' => $type])->column('value', 'name'); + foreach ($data as $k => $v) { + $json = json_decode($v, true); + if (json_last_error() === JSON_ERROR_NONE) { + $data[$k] = $json; + } + } + if ($data) { + return $data; + } + } +} \ No newline at end of file diff --git a/app/common/service/FileService.php b/app/common/service/FileService.php new file mode 100644 index 0000000..d79995d --- /dev/null +++ b/app/common/service/FileService.php @@ -0,0 +1,104 @@ +host(), 'http://') && !strstr(request()->host(), 'https://'))?'http://'.request()->host():request()->host(); + } else { + $storage = Cache::get('STORAGE_ENGINE'); + if (!$storage) { + $storage = ConfigService::get('storage', $default); + Cache::set('STORAGE_ENGINE', $storage); + } + $domain = $storage ? $storage['domain'] : ''; + } + + return self::format($domain, $uri); + } + + /** + * @notes 转相对路径 + * @param $uri + * @return mixed + * @author 乔峰 + * @date 2021/7/28 15:09 + */ + public static function setFileUrl($uri) + { + $default = ConfigService::get('storage', 'default', 'local'); + if ($default === 'local') { + $domain = 'http://'.request()->host(); + return str_replace($domain.'/', '', $uri); + } else { + $storage = ConfigService::get('storage', $default); + return str_replace($storage['domain'].'/', '', $uri); + } + } + + + /** + * @notes 格式化url + * @param $domain + * @param $uri + * @return string + * @author 乔峰 + * @date 2022/7/11 10:36 + */ + public static function format($domain, $uri) + { + // 处理域名 + $domainLen = strlen($domain); + $domainRight = substr($domain, $domainLen -1, 1); + if ('/' == $domainRight) { + $domain = substr_replace($domain,'',$domainLen -1, 1); + } + + // 处理uri + $uriLeft = substr($uri, 0, 1); + if('/' == $uriLeft) { + $uri = substr_replace($uri,'',0, 1); + } + + return trim($domain) . '/' . trim($uri); + } +} \ No newline at end of file diff --git a/app/common/service/JsonService.php b/app/common/service/JsonService.php new file mode 100644 index 0000000..7188e35 --- /dev/null +++ b/app/common/service/JsonService.php @@ -0,0 +1,118 @@ +export == ExportEnum::INFO && $lists instanceof ListsExcelInterface) { + return self::data($lists->excelInfo()); + } + + //获取导出文件的下载链接 + if ($lists->export == ExportEnum::EXPORT && $lists instanceof ListsExcelInterface) { + $exportDownloadUrl = $lists->createExcel($lists->setExcelFields(), $lists->lists()); + return self::success('', ['url' => $exportDownloadUrl], 2); + } + + $data = [ + 'lists' => $lists->lists(), + 'count' => $lists->count(), + 'page_no' => $lists->pageNo, + 'page_size' => $lists->pageSize, + ]; + $data['extend'] = []; + if ($lists instanceof ListsExtendInterface) { + $data['extend'] = $lists->extend(); + } + return self::success('', $data, 1, 0); + } +} \ No newline at end of file diff --git a/app/common/service/UploadService.php b/app/common/service/UploadService.php new file mode 100644 index 0000000..65a31a3 --- /dev/null +++ b/app/common/service/UploadService.php @@ -0,0 +1,150 @@ + ConfigService::get('storage', 'default', 'local'), + 'engine' => ConfigService::get('storage') ?? ['local'=>[]], + ]; + + // 2、执行文件上传 + $StorageDriver = new StorageDriver($config); + $StorageDriver->setUploadFile('file'); + $fileName = $StorageDriver->getFileName(); + $fileInfo = $StorageDriver->getFileInfo(); + + // 校验上传文件后缀 + if (!in_array(strtolower($fileInfo['ext']), config('project.file_image'))) { + throw new Exception("上传图片不允许上传". $fileInfo['ext'] . "文件"); + } + + // 上传文件 + $uriPath = '/'.$saveDir . '/' . date('Ymd'); + $saveDir = public_path().$uriPath; + if (!$StorageDriver->upload($saveDir)) { + throw new Exception($StorageDriver->getError()); + } + + // 3、处理文件名称 + if (strlen($fileInfo['name']) > 128) { + $name = substr($fileInfo['name'], 0, 123); + $nameEnd = substr($fileInfo['name'], strlen($fileInfo['name'])-5, strlen($fileInfo['name'])); + $fileInfo['name'] = $name . $nameEnd; + } + + // 4、写入数据库中 + $file = File::create([ + 'cid' => $cid, + 'type' => FileEnum::IMAGE_TYPE, + 'name' => $fileInfo['name'], + 'uri' => 'http://'.request()->host().$uriPath.'/' . str_replace("\\","/", $fileName), + 'source' => $source, + 'source_id' => $sourceId, + 'create_time' => time(), + ]); + + // 5、返回结果 + return [ + 'id' => $file['id'], + 'cid' => $file['cid'], + 'type' => $file['type'], + 'name' => $file['name'], + 'uri' => FileService::getFileUrl($file['uri']), + 'url' => $file['uri'] + ]; + + } catch (Exception $e) { + throw new Exception($e->getMessage()); + } + } + + + /** + * @notes 视频上传 + * @param $cid + * @param int $user_id + * @param string $saveDir + * @return array + * @throws Exception + * @author 乔峰 + * @date 2021/12/29 16:32 + */ + public static function video($cid, int $sourceId = 0, int $source = FileEnum::SOURCE_ADMIN, string $saveDir = 'uploads/video') + { + try { + $config = [ + 'default' => ConfigService::get('storage', 'default', 'local'), + 'engine' => ConfigService::get('storage') ?? ['local'=>[]], + ]; + + // 2、执行文件上传 + $StorageDriver = new StorageDriver($config); + $StorageDriver->setUploadFile('file'); + $fileName = $StorageDriver->getFileName(); + $fileInfo = $StorageDriver->getFileInfo(); + + // 校验上传文件后缀 + if (!in_array(strtolower($fileInfo['ext']), config('project.file_video'))) { + throw new Exception("上传视频不允许上传". $fileInfo['ext'] . "文件"); + } + + // 上传文件 + $saveDir = $saveDir . '/' . date('Ymd'); + if (!$StorageDriver->upload($saveDir)) { + throw new Exception($StorageDriver->getError()); + } + + // 3、处理文件名称 + if (strlen($fileInfo['name']) > 128) { + $name = substr($fileInfo['name'], 0, 123); + $nameEnd = substr($fileInfo['name'], strlen($fileInfo['name'])-5, strlen($fileInfo['name'])); + $fileInfo['name'] = $name . $nameEnd; + } + + // 4、写入数据库中 + $file = File::create([ + 'cid' => $cid, + 'type' => FileEnum::VIDEO_TYPE, + 'name' => $fileInfo['name'], + 'uri' => $saveDir . '/' . str_replace("\\","/", $fileName), + 'source' => $source, + 'source_id' => $sourceId, + 'create_time' => time(), + ]); + + // 5、返回结果 + return [ + 'id' => $file['id'], + 'cid' => $file['cid'], + 'type' => $file['type'], + 'name' => $file['name'], + 'uri' => FileService::getFileUrl($file['uri']), + 'url' => $file['uri'] + ]; + + } catch (Exception $e) { + throw new Exception($e->getMessage()); + } + } +} \ No newline at end of file diff --git a/app/common/service/generator/GenerateService.php b/app/common/service/generator/GenerateService.php new file mode 100644 index 0000000..164eb7f --- /dev/null +++ b/app/common/service/generator/GenerateService.php @@ -0,0 +1,236 @@ +generatePath = base_path() . '/runtime/generate/'; + $this->runtimePath = base_path() . '/runtime/'; + } + + + /** + * @notes 删除生成文件夹内容 + * @author 段誉 + * @date 2022/6/23 18:52 + */ + public function delGenerateDirContent() + { + // 删除runtime目录制定文件夹 + !is_dir($this->generatePath) && mkdir($this->generatePath, 0755, true); + del_target_dir($this->generatePath, false); + } + + + /** + * @notes 设置生成状态 + * @param $name + * @param false $status + * @author 段誉 + * @date 2022/6/23 18:53 + */ + public function setGenerateFlag($name, $status = false) + { + $this->flag = $name; + Cache::set($name, (int)$status, 3600); + } + + + /** + * @notes 获取生成状态标记 + * @return mixed|object|\think\App + * @author 段誉 + * @date 2022/6/23 18:53 + */ + public function getGenerateFlag() + { + return Cache::get($this->flag); + } + + + /** + * @notes 删除标记时间 + * @author 段誉 + * @date 2022/6/23 18:53 + */ + public function delGenerateFlag() + { + Cache::delete($this->flag); + } + + + /** + * @notes 生成器相关类 + * @return string[] + * @author 段誉 + * @date 2022/6/23 17:17 + */ + public function getGeneratorClass() + { + return [ + ControllerGenerator::class, + ListsGenerator::class, + ModelGenerator::class, + ValidateGenerator::class, + LogicGenerator::class, + VueApiGenerator::class, + VueIndexGenerator::class, + VueEditGenerator::class, + SqlGenerator::class, + ]; + } + + + /** + * @notes 生成文件 + * @param array $tableData + * @author 段誉 + * @date 2022/6/23 18:52 + */ + public function generate(array $tableData) + { + foreach ($this->getGeneratorClass() as $item) { + $generator = Container::get($item); + $generator->initGenerateData($tableData); + $generator->generate(); + // 是否为压缩包下载 + if ($generator->isGenerateTypeZip()) { + $this->setGenerateFlag($this->flag, true); + } + // 是否构建菜单 + if ($item == 'app\common\service\generator\core\SqlGenerator') { + $generator->isBuildMenu() && $generator->buildMenuHandle(); + } + } + } + + + /** + * @notes 预览文件 + * @param array $tableData + * @return array + * @author 段誉 + * @date 2022/6/23 18:52 + */ + public function preview(array $tableData) + { + $data = []; + foreach ($this->getGeneratorClass() as $item) { + $generator = Container::get($item); + $generator->initGenerateData($tableData); + $data[] = $generator->fileInfo(); + } + return $data; + } + + + /** + * @notes 压缩文件 + * @author 段誉 + * @date 2022/6/23 19:02 + */ + public function zipFile() + { + $fileName = 'curd-' . date('YmdHis') . '.zip'; + $this->zipTempName = $fileName; + $this->zipTempPath = $this->generatePath . $fileName; + $zip = new \ZipArchive(); + $zip->open($this->zipTempPath, \ZipArchive::CREATE); + $this->addFileZip($this->runtimePath, 'generate', $zip); + $zip->close(); + } + + + /** + * @notes 往压缩包写入文件 + * @param $basePath + * @param $dirName + * @param $zip + * @author 段誉 + * @date 2022/6/23 19:02 + */ + public function addFileZip($basePath, $dirName, $zip) + { + $handler = opendir($basePath . $dirName); + while (($filename = readdir($handler)) !== false) { + if ($filename != '.' && $filename != '..') { + if (is_dir($basePath . $dirName . '/' . $filename)) { + // 当前路径是文件夹 + $this->addFileZip($basePath, $dirName . '/' . $filename, $zip); + } else { + // 写入文件到压缩包 + $zip->addFile($basePath . $dirName . '/' . $filename, $dirName . '/' . $filename); + } + } + } + closedir($handler); + } + + + /** + * @notes 返回压缩包临时路径 + * @return mixed + * @author 段誉 + * @date 2022/6/24 9:41 + */ + public function getDownloadUrl() + { + // $vars = ['file' => $this->zipTempName]; + Cache::set('curd_file_name' . $this->zipTempName, $this->zipTempName, 3600); + return "admin/tools/generator/download?file=".$this->zipTempName; + } + +} \ No newline at end of file diff --git a/app/common/service/generator/core/BaseGenerator.php b/app/common/service/generator/core/BaseGenerator.php new file mode 100644 index 0000000..773f67b --- /dev/null +++ b/app/common/service/generator/core/BaseGenerator.php @@ -0,0 +1,483 @@ +basePath = base_path().'/'; + // $this->rootPath = app_path().'/'; + $this->templateDir = app_path() . '/common/service/generator/stub/'; + $this->generatorDir = $this->basePath . 'runtime/generate/'; + $this->checkDir($this->generatorDir); + } + + + /** + * @notes 初始化表表数据 + * @param array $tableData + * @author 段誉 + * @date 2022/6/22 18:03 + */ + public function initGenerateData(array $tableData) + { + // 设置当前表信息 + $this->setTableData($tableData); + // 设置模块名 + $this->setModuleName($tableData['module_name']); + // 设置类目录 + $this->setClassDir($tableData['class_dir'] ?? ''); + // 替换模板变量 + $this->replaceVariables(); + } + + + /** + * @notes 菜单配置 + * @author 段誉 + * @date 2022/12/13 15:14 + */ + public function setMenuConfig() + { + $this->menuConfig = [ + 'pid' => $this->tableData['menu']['pid'] ?? 0, + 'type' => $this->tableData['menu']['type'] ?? GeneratorEnum::DELETE_TRUE, + 'name' => $this->tableData['menu']['name'] ?? $this->tableData['table_comment'] + ]; + } + + + /** + * @notes 删除配置 + * @return array + * @author 段誉 + * @date 2022/12/13 15:09 + */ + public function setDeleteConfig() + { + $this->deleteConfig = [ + 'type' => $this->tableData['delete']['type'] ?? GeneratorEnum::DELETE_TRUE, + 'name' => $this->tableData['delete']['name'] ?? GeneratorEnum::DELETE_NAME, + ]; + } + + + /** + * @notes 关联模型配置 + * @author 段誉 + * @date 2022/12/14 11:28 + */ + public function setRelationConfig() + { + $this->relationConfig = empty($this->tableData['relations']) ? [] : $this->tableData['relations']; + } + + + /** + * @notes 设置树表配置 + * @author 段誉 + * @date 2022/12/20 14:30 + */ + public function setTreeConfig() + { + $this->treeConfig = [ + 'tree_id' => $this->tableData['tree']['tree_id'] ?? '', + 'tree_pid' => $this->tableData['tree']['tree_pid'] ?? '', + 'tree_name' => $this->tableData['tree']['tree_name'] ?? '', + ]; + } + + + /** + * @notes 生成文件到模块或runtime目录 + * @author 段誉 + * @date 2022/6/22 18:03 + */ + public function generate() + { + //生成方式 0-压缩包下载 1-生成到模块 + if ($this->tableData['generate_type']) { + // 生成路径 + $path = $this->getModuleGenerateDir() . $this->getGenerateName(); + } else { + // 生成到runtime目录 + $path = $this->getRuntimeGenerateDir() . $this->getGenerateName(); + } + // 写入内容 + file_put_contents($path, $this->content); + } + + + /** + * @notes 获取文件生成到模块的文件夹路径 + * @return mixed + * @author 段誉 + * @date 2022/6/22 18:05 + */ + abstract public function getModuleGenerateDir(); + + + /** + * @notes 获取文件生成到runtime的文件夹路径 + * @return mixed + * @author 段誉 + * @date 2022/6/22 18:05 + */ + abstract public function getRuntimeGenerateDir(); + + + /** + * @notes 替换模板变量 + * @return mixed + * @author 段誉 + * @date 2022/6/22 18:06 + */ + abstract public function replaceVariables(); + + + /** + * @notes 生成文件名 + * @return mixed + * @author 段誉 + * @date 2022/6/22 18:17 + */ + abstract public function getGenerateName(); + + + /** + * @notes 文件夹不存在则创建 + * @param string $path + * @author 段誉 + * @date 2022/6/22 18:07 + */ + public function checkDir(string $path) + { + !is_dir($path) && mkdir($path, 0755, true); + } + + + /** + * @notes 设置表信息 + * @param $tableData + * @author 段誉 + * @date 2022/6/22 18:07 + */ + public function setTableData($tableData) + { + $this->tableData = !empty($tableData) ? $tableData : []; + $this->tableColumn = $tableData['table_column'] ?? []; + // 菜单配置 + $this->setMenuConfig(); + // 删除配置 + $this->setDeleteConfig(); + // 关联模型配置 + $this->setRelationConfig(); + // 设置树表配置 + $this->setTreeConfig(); + } + + + /** + * @notes 设置模块名 + * @param string $moduleName + * @author 段誉 + * @date 2022/6/22 18:07 + */ + public function setModuleName(string $moduleName): void + { + $this->moduleName = strtolower($moduleName); + } + + + /** + * @notes 设置类目录 + * @param string $classDir + * @author 段誉 + * @date 2022/6/22 18:08 + */ + public function setClassDir(string $classDir): void + { + $this->classDir = $classDir; + } + + + /** + * @notes 设置生成文件内容 + * @param string $content + * @author 段誉 + * @date 2022/6/22 18:08 + */ + public function setContent(string $content): void + { + $this->content = $content; + } + + + /** + * @notes 获取模板路径 + * @param string $templateName + * @return string + * @author 段誉 + * @date 2022/6/22 18:09 + */ + public function getTemplatePath(string $templateName): string + { + return $this->templateDir . $templateName . '.stub'; + } + + + /** + * @notes 小驼峰命名 + * @return string + * @author 段誉 + * @date 2022/6/27 18:44 + */ + public function getLowerCamelName() + { + return Str::camel($this->getTableName()); + } + + + /** + * @notes 大驼峰命名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:09 + */ + public function getUpperCamelName() + { + return Str::studly($this->getTableName()); + } + + + /** + * @notes 表名小写 + * @return string + * @author 段誉 + * @date 2022/7/12 10:41 + */ + public function getLowerTableName() + { + return Str::lower($this->getTableName()); + } + + + /** + * @notes 获取表名 + * @return array|string|string[] + * @author 段誉 + * @date 2022/6/22 18:09 + */ + public function getTableName() + { + return get_no_prefix_table_name($this->tableData['table_name']); + } + + + /** + * @notes 获取表主键 + * @return mixed|string + * @author 段誉 + * @date 2022/6/22 18:09 + */ + public function getPkContent() + { + $pk = 'id'; + if (empty($this->tableColumn)) { + return $pk; + } + + foreach ($this->tableColumn as $item) { + if ($item['is_pk']) { + $pk = $item['column_name']; + } + } + return $pk; + } + + + /** + * @notes 获取作者信息 + * @return mixed|string + * @author 段誉 + * @date 2022/6/24 10:18 + */ + public function getAuthorContent() + { + return empty($this->tableData['author']) ? 'likeadmin' : $this->tableData['author']; + } + + + /** + * @notes 代码生成备注时间 + * @return false|string + * @author 段誉 + * @date 2022/6/24 10:28 + */ + public function getNoteDateContent() + { + return date('Y/m/d H:i'); + } + + + /** + * @notes 设置空额占位符 + * @param $content + * @param $blankpace + * @return string + * @author 段誉 + * @date 2022/6/22 18:09 + */ + public function setBlankSpace($content, $blankpace) + { + $content = explode(PHP_EOL, $content); + foreach ($content as $line => $text) { + $content[$line] = $blankpace . $text; + } + return (implode(PHP_EOL, $content)); + } + + + /** + * @notes 替换内容 + * @param $needReplace + * @param $waitReplace + * @param $template + * @return array|false|string|string[] + * @author 段誉 + * @date 2022/6/23 9:52 + */ + public function replaceFileData($needReplace, $waitReplace, $template) + { + return str_replace($needReplace, $waitReplace, file_get_contents($template)); + } + + + /** + * @notes 生成方式是否为压缩包 + * @return bool + * @author 段誉 + * @date 2022/6/23 17:02 + */ + public function isGenerateTypeZip() + { + return $this->tableData['generate_type'] == GeneratorEnum::GENERATE_TYPE_ZIP; + } + + + /** + * @notes 是否为树表crud + * @return bool + * @author 段誉 + * @date 2022/12/23 11:25 + */ + public function isTreeCrud() + { + return $this->tableData['template_type'] == GeneratorEnum::TEMPLATE_TYPE_TREE; + } + +} \ No newline at end of file diff --git a/app/common/service/generator/core/ControllerGenerator.php b/app/common/service/generator/core/ControllerGenerator.php new file mode 100644 index 0000000..0050743 --- /dev/null +++ b/app/common/service/generator/core/ControllerGenerator.php @@ -0,0 +1,222 @@ +getNameSpaceContent(), + $this->getUseContent(), + $this->getClassCommentContent(), + $this->getUpperCamelName(), + $this->moduleName, + $this->getPackageNameContent(), + $this->getExtendsControllerContent(), + $this->tableData['class_comment'], + $this->getAuthorContent(), + $this->getNoteDateContent(), + ]; + + $templatePath = $this->getTemplatePath('php/controller'); + + // 替换内容 + $content = $this->replaceFileData($needReplace, $waitReplace, $templatePath); + + $this->setContent($content); + } + + + /** + * @notes 获取命名空间内容 + * @return string + * @author 段誉 + * @date 2022/6/22 18:10 + */ + public function getNameSpaceContent() + { + if (!empty($this->classDir)) { + return "namespace app\\" . $this->moduleName . "\\controller\\" . $this->classDir . ';'; + } + return "namespace app\\" . $this->moduleName . "\\controller;"; + } + + + /** + * @notes 获取use模板内容 + * @return string + * @author 段誉 + * @date 2022/6/22 18:10 + */ + public function getUseContent() + { + if ($this->moduleName == 'admin') { + $tpl = "use app\\" . $this->moduleName . "\\controller\\BaseAdminController;" . PHP_EOL; + } else { + $tpl = "use app\\common\\controller\\BaseLikeAdminController;" . PHP_EOL; + } + + if (!empty($this->classDir)) { + $tpl .= "use app\\" . $this->moduleName . "\\lists\\" . $this->classDir . "\\" . $this->getUpperCamelName() . "Lists;" . PHP_EOL . + "use app\\" . $this->moduleName . "\\logic\\" . $this->classDir . "\\" . $this->getUpperCamelName() . "Logic;" . PHP_EOL . + "use app\\" . $this->moduleName . "\\validate\\" . $this->classDir . "\\" . $this->getUpperCamelName() . "Validate;"; + } else { + $tpl .= "use app\\" . $this->moduleName . "\\lists\\" . $this->getUpperCamelName() . "Lists;" . PHP_EOL . + "use app\\" . $this->moduleName . "\\logic\\" . $this->getUpperCamelName() . "Logic;" . PHP_EOL . + "use app\\" . $this->moduleName . "\\validate\\" . $this->getUpperCamelName() . "Validate;"; + } + return $tpl; + } + + + /** + * @notes 获取类描述内容 + * @return string + * @author 段誉 + * @date 2022/6/22 18:10 + */ + public function getClassCommentContent() + { + if (!empty($this->tableData['class_comment'])) { + $tpl = $this->tableData['class_comment'] . '控制器'; + } else { + $tpl = $this->getUpperCamelName() . '控制器'; + } + return $tpl; + } + + + /** + * @notes 获取包名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:10 + */ + public function getPackageNameContent() + { + return !empty($this->classDir) ? '\\' . $this->classDir : ''; + } + + + /** + * @notes 获取继承控制器 + * @return string + * @author 段誉 + * @date 2022/6/22 18:10 + */ + public function getExtendsControllerContent() + { + $tpl = 'BaseAdminController'; + if ($this->moduleName != 'admin') { + $tpl = 'BaseLikeAdminController'; + } + return $tpl; + } + + + /** + * @notes 获取文件生成到模块的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:10 + */ + public function getModuleGenerateDir() + { + $dir = $this->basePath . $this->moduleName . '/controller/'; + if (!empty($this->classDir)) { + $dir .= $this->classDir . '/'; + $this->checkDir($dir); + } + return $dir; + } + + + /** + * @notes 获取文件生成到runtime的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:11 + */ + public function getRuntimeGenerateDir() + { + $dir = $this->generatorDir . 'php/app/' . $this->moduleName . '/controller/'; + $this->checkDir($dir); + if (!empty($this->classDir)) { + $dir .= $this->classDir . '/'; + $this->checkDir($dir); + } + return $dir; + } + + + /** + * @notes 生成文件名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:11 + */ + public function getGenerateName() + { + return $this->getUpperCamelName() . 'Controller.php'; + } + + + /** + * @notes 文件信息 + * @return array + * @author 段誉 + * @date 2022/6/23 15:57 + */ + public function fileInfo(): array + { + return [ + 'name' => $this->getGenerateName(), + 'type' => 'php', + 'content' => $this->content + ]; + } + +} \ No newline at end of file diff --git a/app/common/service/generator/core/GenerateInterface.php b/app/common/service/generator/core/GenerateInterface.php new file mode 100644 index 0000000..74a7ceb --- /dev/null +++ b/app/common/service/generator/core/GenerateInterface.php @@ -0,0 +1,23 @@ +getNameSpaceContent(), + $this->getUseContent(), + $this->getClassCommentContent(), + $this->getUpperCamelName(), + $this->moduleName, + $this->getPackageNameContent(), + $this->getExtendsListsContent(), + $this->getPkContent(), + $this->getQueryConditionContent(), + $this->getFieldDataContent(), + $this->tableData['class_comment'], + $this->getAuthorContent(), + $this->getNoteDateContent(), + ]; + + $templatePath = $this->getTemplatePath('php/lists'); + if ($this->isTreeCrud()) { + // 插入树表相关 + array_push($needReplace, '{TREE_ID}', '{TREE_PID}'); + array_push($waitReplace, $this->treeConfig['tree_id'], $this->treeConfig['tree_pid']); + + $templatePath = $this->getTemplatePath('php/tree_lists'); + } + + // 替换内容 + $content = $this->replaceFileData($needReplace, $waitReplace, $templatePath); + + $this->setContent($content); + } + + + /** + * @notes 获取命名空间内容 + * @return string + * @author 段誉 + * @date 2022/6/22 18:12 + */ + public function getNameSpaceContent() + { + if (!empty($this->classDir)) { + return "namespace app\\" . $this->moduleName . "\\lists\\" . $this->classDir . ';'; + } + return "namespace app\\" . $this->moduleName . "\\lists;"; + } + + + /** + * @notes 获取use内容 + * @return string + * @author 段誉 + * @date 2022/6/22 18:12 + */ + public function getUseContent() + { + if ($this->moduleName == 'admin') { + $tpl = "use app\\" . $this->moduleName . "\\lists\\BaseAdminDataLists;" . PHP_EOL; + } else { + $tpl = "use app\\common\\lists\\BaseDataLists;" . PHP_EOL; + } + + if (!empty($this->classDir)) { + $tpl .= "use app\\common\\model\\" . $this->classDir . "\\" . $this->getUpperCamelName() . ';'; + } else { + $tpl .= "use app\\common\\model\\" . $this->getUpperCamelName() . ';'; + } + + return $tpl; + } + + + /** + * @notes 获取类描述 + * @return string + * @author 段誉 + * @date 2022/6/22 18:12 + */ + public function getClassCommentContent() + { + if (!empty($this->tableData['class_comment'])) { + $tpl = $this->tableData['class_comment'] . '列表'; + } else { + $tpl = $this->getUpperCamelName() . '列表'; + } + return $tpl; + } + + + /** + * @notes 获取包名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:12 + */ + public function getPackageNameContent() + { + return !empty($this->classDir) ? $this->classDir : ''; + } + + + /** + * @notes 获取继承控制器 + * @return string + * @author 段誉 + * @date 2022/6/22 18:12 + */ + public function getExtendsListsContent() + { + $tpl = 'BaseAdminDataLists'; + if ($this->moduleName != 'admin') { + $tpl = 'BaseDataLists'; + } + return $tpl; + } + + + /** + * @notes 获取查询条件内容 + * @return string + * @author 段誉 + * @date 2022/6/22 18:12 + */ + public function getQueryConditionContent() + { + $columnQuery = array_column($this->tableColumn, 'query_type'); + $query = array_unique($columnQuery); + + $conditon = ''; + + $specQueryHandle = ['between', 'like']; + + foreach ($query as $queryName) { + $columnValue = ''; + foreach ($this->tableColumn as $column) { + if (empty($column['query_type']) || $column['is_pk']) { + continue; + } + if ($queryName == $column['query_type'] && $column['is_query'] && !in_array($queryName, $specQueryHandle)) { + $columnValue .= "'" . $column['column_name'] . "', "; + } + } + if (!empty($columnValue)) { + $columnValue = substr($columnValue, 0, -2); + $conditon .= "'$queryName' => [" . trim($columnValue) . "]," . PHP_EOL; + } + } + + $likeColumn = ''; + $betweenColumn = ''; + $betweenTimeColumn = ''; + + // 另外处理between,like 等查询条件 + foreach ($this->tableColumn as $item) { + if (!$item['is_query']) { + continue; + } + // like + if ($item['query_type'] == 'like') { + $likeColumn .= "'" . $item['column_name'] . "', "; + continue; + } + // between + if ($item['query_type'] == 'between') { + if ($item['view_type'] == 'datetime') { + $betweenTimeColumn .= "'" . $item['column_name'] . "', "; + } else { + $betweenColumn .= "'" . $item['column_name'] . "', "; + } + } + } + + if (!empty($likeColumn)) { + $likeColumn = substr($likeColumn, 0, -2); + $conditon .= "'%like%' => " . "[" . trim($likeColumn) . "]," . PHP_EOL; + } + + if (!empty($betweenColumn)) { + $betweenColumn = substr($betweenColumn, 0, -2); + $conditon .= "'between' => " . "[" . trim($betweenColumn) . "]," . PHP_EOL; + } + + if (!empty($betweenTimeColumn)) { + $betweenTimeColumn = substr($betweenTimeColumn, 0, -2); + $conditon .= "'between_time' => " . "[" . trim($betweenTimeColumn) . "]," . PHP_EOL; + } + + $content = substr($conditon, 0, -1); + return $this->setBlankSpace($content, " "); + } + + + /** + * @notes 获取查询字段 + * @return false|string + * @author 段誉 + * @date 2022/6/22 18:13 + */ + public function getFieldDataContent() + { + $content = "'" . $this->getPkContent() . "', "; + $isExist = [$this->getPkContent()]; + foreach ($this->tableColumn as $column) { + if ($column['is_lists'] && !in_array($column['column_name'], $isExist)) { + $content .= "'" . $column['column_name'] . "', "; + $isExist[] = $column['column_name']; + } + + if ($this->isTreeCrud() && !in_array($column['column_name'], $isExist) + && in_array($column['column_name'], [$this->treeConfig['tree_id'], $this->treeConfig['tree_pid']]) + ) { + $content .= "'" . $column['column_name'] . "', "; + } + } + return substr($content, 0, -2); + } + + + /** + * @notes 获取文件生成到模块的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:13 + */ + public function getModuleGenerateDir() + { + $dir = $this->basePath . $this->moduleName . '/lists/'; + $this->checkDir($dir); + if (!empty($this->classDir)) { + $dir .= $this->classDir . '/'; + $this->checkDir($dir); + } + return $dir; + } + + + /** + * @notes 获取文件生成到runtime的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:13 + */ + public function getRuntimeGenerateDir() + { + $dir = $this->generatorDir . 'php/app/' . $this->moduleName . '/lists/'; + $this->checkDir($dir); + if (!empty($this->classDir)) { + $dir .= $this->classDir . '/'; + $this->checkDir($dir); + } + return $dir; + } + + + /** + * @notes 生成的文件名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:13 + */ + public function getGenerateName() + { + return $this->getUpperCamelName() . 'Lists.php'; + } + + + /** + * @notes 文件信息 + * @return array + * @author 段誉 + * @date 2022/6/23 15:57 + */ + public function fileInfo(): array + { + return [ + 'name' => $this->getGenerateName(), + 'type' => 'php', + 'content' => $this->content + ]; + } + + +} \ No newline at end of file diff --git a/app/common/service/generator/core/LogicGenerator.php b/app/common/service/generator/core/LogicGenerator.php new file mode 100644 index 0000000..c97ec1c --- /dev/null +++ b/app/common/service/generator/core/LogicGenerator.php @@ -0,0 +1,268 @@ +getNameSpaceContent(), + $this->getUseContent(), + $this->getClassCommentContent(), + $this->getUpperCamelName(), + $this->moduleName, + $this->getPackageNameContent(), + $this->getPkContent(), + $this->getCreateDataContent(), + $this->getUpdateDataContent(), + $this->tableData['class_comment'], + $this->getAuthorContent(), + $this->getNoteDateContent(), + ]; + + $templatePath = $this->getTemplatePath('php/logic'); + + // 替换内容 + $content = $this->replaceFileData($needReplace, $waitReplace, $templatePath); + + $this->setContent($content); + } + + + /** + * @notes 添加内容 + * @return string + * @author 段誉 + * @date 2022/6/22 18:14 + */ + public function getCreateDataContent() + { + $content = ''; + foreach ($this->tableColumn as $column) { + if (!$column['is_insert']) { + continue; + } + $content .= $this->addEditColumn($column); + } + if (empty($content)) { + return $content; + } + $content = substr($content, 0, -2); + return $this->setBlankSpace($content, " "); + } + + + /** + * @notes 编辑内容 + * @return string + * @author 段誉 + * @date 2022/6/22 18:14 + */ + public function getUpdateDataContent() + { + $columnContent = ''; + foreach ($this->tableColumn as $column) { + if (!$column['is_update']) { + continue; + } + $columnContent .= $this->addEditColumn($column); + } + + if (empty($columnContent)) { + return $columnContent; + } + + $columnContent = substr($columnContent, 0, -2); + $content = $columnContent; + return $this->setBlankSpace($content, " "); + } + + + /** + * @notes 添加编辑字段内容 + * @param $column + * @return mixed + * @author 段誉 + * @date 2022/6/27 15:37 + */ + public function addEditColumn($column) + { + if ($column['column_type'] == 'int' && $column['view_type'] == 'datetime') { + // 物理类型为int,显示类型选择日期的情况 + $content = "'" . $column['column_name'] . "' => " . 'strtotime($params[' . "'" . $column['column_name'] . "'" . ']),' . PHP_EOL; + } else { + $content = "'" . $column['column_name'] . "' => " . '$params[' . "'" . $column['column_name'] . "'" . '],' . PHP_EOL; + } + return $content; + } + + + /** + * @notes 获取命名空间内容 + * @return string + * @author 段誉 + * @date 2022/6/22 18:14 + */ + public function getNameSpaceContent() + { + if (!empty($this->classDir)) { + return "namespace app\\" . $this->moduleName . "\\logic\\" . $this->classDir . ';'; + } + return "namespace app\\" . $this->moduleName . "\\logic;"; + } + + + /** + * @notes 获取use内容 + * @return string + * @author 段誉 + * @date 2022/6/22 18:14 + */ + public function getUseContent() + { + $tpl = "use app\\common\\model\\" . $this->getUpperCamelName() . ';'; + if (!empty($this->classDir)) { + $tpl = "use app\\common\\model\\" . $this->classDir . "\\" . $this->getUpperCamelName() . ';'; + } + return $tpl; + } + + + /** + * @notes 获取类描述 + * @return string + * @author 段誉 + * @date 2022/6/22 18:14 + */ + public function getClassCommentContent() + { + if (!empty($this->tableData['class_comment'])) { + $tpl = $this->tableData['class_comment'] . '逻辑'; + } else { + $tpl = $this->getUpperCamelName() . '逻辑'; + } + return $tpl; + } + + + /** + * @notes 获取包名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:14 + */ + public function getPackageNameContent() + { + return !empty($this->classDir) ? '\\' . $this->classDir : ''; + } + + + /** + * @notes 获取文件生成到模块的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:15 + */ + public function getModuleGenerateDir() + { + $dir = $this->basePath . $this->moduleName . '/logic/'; + if (!empty($this->classDir)) { + $dir .= $this->classDir . '/'; + $this->checkDir($dir); + } + return $dir; + } + + + /** + * @notes 获取文件生成到runtime的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:15 + */ + public function getRuntimeGenerateDir() + { + $dir = $this->generatorDir . 'php/app/' . $this->moduleName . '/logic/'; + $this->checkDir($dir); + if (!empty($this->classDir)) { + $dir .= $this->classDir . '/'; + $this->checkDir($dir); + } + return $dir; + } + + + /** + * @notes 生成的文件名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:15 + */ + public function getGenerateName() + { + return $this->getUpperCamelName() . 'Logic.php'; + } + + + /** + * @notes 文件信息 + * @return array + * @author 段誉 + * @date 2022/6/23 15:57 + */ + public function fileInfo(): array + { + return [ + 'name' => $this->getGenerateName(), + 'type' => 'php', + 'content' => $this->content + ]; + } + + +} \ No newline at end of file diff --git a/app/common/service/generator/core/ModelGenerator.php b/app/common/service/generator/core/ModelGenerator.php new file mode 100644 index 0000000..3d32ff4 --- /dev/null +++ b/app/common/service/generator/core/ModelGenerator.php @@ -0,0 +1,275 @@ +getNameSpaceContent(), + $this->getClassCommentContent(), + $this->getUpperCamelName(), + $this->getPackageNameContent(), + $this->getTableName(), + $this->getUseContent(), + $this->getDeleteUseContent(), + $this->getDeleteTimeContent(), + $this->getRelationModel(), + ]; + + $templatePath = $this->getTemplatePath('php/model'); + + // 替换内容 + $content = $this->replaceFileData($needReplace, $waitReplace, $templatePath); + + $this->setContent($content); + } + + + /** + * @notes 获取命名空间模板内容 + * @return string + * @author 段誉 + * @date 2022/6/22 18:16 + */ + public function getNameSpaceContent() + { + if (!empty($this->classDir)) { + return "namespace app\\common\\model\\" . $this->classDir . ';'; + } + return "namespace app\\common\\model;"; + } + + + /** + * @notes 获取类描述 + * @return string + * @author 段誉 + * @date 2022/6/22 18:16 + */ + public function getClassCommentContent() + { + if (!empty($this->tableData['class_comment'])) { + $tpl = $this->tableData['class_comment'] . '模型'; + } else { + $tpl = $this->getUpperCamelName() . '模型'; + } + return $tpl; + } + + + /** + * @notes 获取包名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:16 + */ + public function getPackageNameContent() + { + return !empty($this->classDir) ? '\\' . $this->classDir : ''; + } + + + /** + * @notes 引用内容 + * @return string + * @author 段誉 + * @date 2022/12/12 17:32 + */ + public function getUseContent() + { + $tpl = ""; + if ($this->deleteConfig['type']) { + $tpl = "use think\\model\\concern\\SoftDelete;"; + } + return $tpl; + } + + + /** + * @notes 软删除引用 + * @return string + * @author 段誉 + * @date 2022/12/12 17:34 + */ + public function getDeleteUseContent() + { + $tpl = ""; + if ($this->deleteConfig['type']) { + $tpl = "use SoftDelete;"; + } + return $tpl; + } + + + /** + * @notes 软删除时间字段定义 + * @return string + * @author 段誉 + * @date 2022/12/12 17:38 + */ + public function getDeleteTimeContent() + { + $tpl = ""; + if ($this->deleteConfig['type']) { + $deleteTime = $this->deleteConfig['name']; + $tpl = 'protected $deleteTime = ' . "'". $deleteTime ."';"; + } + return $tpl; + } + + + /** + * @notes 关联模型 + * @return string + * @author 段誉 + * @date 2022/12/14 14:46 + */ + public function getRelationModel() + { + $tpl = ''; + if (empty($this->relationConfig)) { + return $tpl; + } + + // 遍历关联配置 + foreach ($this->relationConfig as $config) { + if (empty($config) || empty($config['name']) || empty($config['model'])) { + continue; + } + + $needReplace = [ + '{RELATION_NAME}', + '{AUTHOR}', + '{DATE}', + '{RELATION_MODEL}', + '{FOREIGN_KEY}', + '{LOCAL_KEY}', + ]; + + $waitReplace = [ + $config['name'], + $this->getAuthorContent(), + $this->getNoteDateContent(), + $config['model'], + $config['foreign_key'], + $config['local_key'], + ]; + + $templatePath = $this->getTemplatePath('php/model/' . $config['type']); + if (!file_exists($templatePath)) { + continue; + } + $tpl .= $this->replaceFileData($needReplace, $waitReplace, $templatePath) . PHP_EOL; + } + + return $tpl; + } + + + /** + * @notes 获取文件生成到模块的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:16 + */ + public function getModuleGenerateDir() + { + $dir = $this->basePath . 'common/model/'; + if (!empty($this->classDir)) { + $dir .= $this->classDir . '/'; + $this->checkDir($dir); + } + return $dir; + } + + + /** + * @notes 获取文件生成到runtime的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:17 + */ + public function getRuntimeGenerateDir() + { + $dir = $this->generatorDir . 'php/app/common/model/'; + $this->checkDir($dir); + if (!empty($this->classDir)) { + $dir .= $this->classDir . '/'; + $this->checkDir($dir); + } + return $dir; + } + + + /** + * @notes 生成的文件名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:17 + */ + public function getGenerateName() + { + return $this->getUpperCamelName() . '.php'; + } + + + /** + * @notes 文件信息 + * @return array + * @author 段誉 + * @date 2022/6/23 15:57 + */ + public function fileInfo(): array + { + return [ + 'name' => $this->getGenerateName(), + 'type' => 'php', + 'content' => $this->content + ]; + } + + +} \ No newline at end of file diff --git a/app/common/service/generator/core/SqlGenerator.php b/app/common/service/generator/core/SqlGenerator.php new file mode 100644 index 0000000..442ed3b --- /dev/null +++ b/app/common/service/generator/core/SqlGenerator.php @@ -0,0 +1,191 @@ +getMenuTableNameContent(), + $this->menuConfig['pid'], + $this->menuConfig['name'], + $this->getPermsNameContent(), + $this->getLowerTableName(), + $this->getLowerTableName(), + time(), + time() + ]; + + $templatePath = $this->getTemplatePath('sql/sql'); + + // 替换内容 + $content = $this->replaceFileData($needReplace, $waitReplace, $templatePath); + + $this->setContent($content); + } + + + /** + * @notes 路由权限内容 + * @return string + * @author 段誉 + * @date 2022/8/11 17:18 + */ + public function getPermsNameContent() + { + if (!empty($this->classDir)) { + return $this->classDir . '.' . Str::lower($this->getTableName()); + } + return Str::lower($this->getTableName()); + } + + + /** + * @notes 获取菜单表内容 + * @return string + * @author 段誉 + * @date 2022/7/7 15:57 + */ + public function getMenuTableNameContent() + { + $tablePrefix = config('database.connections.mysql.prefix'); + return $tablePrefix . 'system_menu'; + } + + + /** + * @notes 是否构建菜单 + * @return bool + * @author 段誉 + * @date 2022/7/8 14:24 + */ + public function isBuildMenu() + { + return $this->menuConfig['type'] == GeneratorEnum::GEN_AUTO; + } + + + /** + * @notes 构建菜单 + * @return bool + * @author 段誉 + * @date 2022/7/8 15:27 + */ + public function buildMenuHandle() + { + if (empty($this->content)) { + return false; + } + $sqls = explode(';', trim($this->content)); + //执行sql + foreach ($sqls as $sql) { + if (!empty(trim($sql))) { + Db::execute($sql . ';'); + } + } + return true; + } + + + /** + * @notes 获取文件生成到模块的文件夹路径 + * @return mixed|void + * @author 段誉 + * @date 2022/6/22 18:19 + */ + public function getModuleGenerateDir() + { + $dir = $this->generatorDir . 'sql/'; + $this->checkDir($dir); + return $dir; + } + + + /** + * @notes 获取文件生成到runtime的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:20 + */ + public function getRuntimeGenerateDir() + { + $dir = $this->generatorDir . 'sql/'; + $this->checkDir($dir); + return $dir; + } + + + /** + * @notes 生成的文件名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:20 + */ + public function getGenerateName() + { + return 'menu.sql'; + } + + + /** + * @notes 文件信息 + * @return array + * @author 段誉 + * @date 2022/6/23 15:57 + */ + public function fileInfo(): array + { + return [ + 'name' => $this->getGenerateName(), + 'type' => 'sql', + 'content' => $this->content + ]; + } + + +} \ No newline at end of file diff --git a/app/common/service/generator/core/ValidateGenerator.php b/app/common/service/generator/core/ValidateGenerator.php new file mode 100644 index 0000000..5366868 --- /dev/null +++ b/app/common/service/generator/core/ValidateGenerator.php @@ -0,0 +1,278 @@ +getNameSpaceContent(), + $this->getClassCommentContent(), + $this->getUpperCamelName(), + $this->moduleName, + $this->getPackageNameContent(), + $this->getPkContent(), + $this->getRuleContent(), + $this->tableData['class_comment'], + $this->getAuthorContent(), + $this->getNoteDateContent(), + $this->getAddParamsContent(), + $this->getEditParamsContent(), + $this->getFiledContent(), + ]; + + $templatePath = $this->getTemplatePath('php/validate'); + + // 替换内容 + $content = $this->replaceFileData($needReplace, $waitReplace, $templatePath); + + $this->setContent($content); + } + + + /** + * @notes 验证规则 + * @return mixed|string + * @author 段誉 + * @date 2022/6/22 18:18 + */ + public function getRuleContent() + { + $content = "'" . $this->getPkContent() . "' => 'require'," . PHP_EOL; + foreach ($this->tableColumn as $column) { + if ($column['is_required'] == 1) { + $content .= "'" . $column['column_name'] . "' => 'require'," . PHP_EOL; + } + } + $content = substr($content, 0, -1); + return $this->setBlankSpace($content, " "); + } + + + /** + * @notes 添加场景验证参数 + * @return string + * @author 段誉 + * @date 2022/12/7 15:26 + */ + public function getAddParamsContent() + { + $content = ""; + foreach ($this->tableColumn as $column) { + if ($column['is_required'] == 1 && $column['column_name'] != $this->getPkContent()) { + $content .= "'" . $column['column_name'] . "',"; + } + } + $content = substr($content, 0, -1); + + // 若无设置添加场景校验字段时, 排除主键 + if (!empty($content)) { + $content = 'return $this->only([' . $content . ']);'; + } else { + $content = 'return $this->remove(' . "'". $this->getPkContent() . "'" . ', true);'; + } + + return $this->setBlankSpace($content, ""); + } + + + /** + * @notes 编辑场景验证参数 + * @return string + * @author 段誉 + * @date 2022/12/7 15:20 + */ + public function getEditParamsContent() + { + $content = "'" . $this->getPkContent() . "'," ; + foreach ($this->tableColumn as $column) { + if ($column['is_required'] == 1) { + $content .= "'" . $column['column_name'] . "',"; + } + } + $content = substr($content, 0, -1); + if (!empty($content)) { + $content = 'return $this->only([' . $content . ']);'; + } + return $this->setBlankSpace($content, ""); + } + + + /** + * @notes 验证字段描述 + * @return string + * @author 段誉 + * @date 2022/12/9 15:09 + */ + public function getFiledContent() + { + $content = "'" . $this->getPkContent() . "' => '" . $this->getPkContent() . "'," . PHP_EOL; + foreach ($this->tableColumn as $column) { + if ($column['is_required'] == 1) { + $columnComment = $column['column_comment']; + if (empty($column['column_comment'])) { + $columnComment = $column['column_name']; + } + $content .= "'" . $column['column_name'] . "' => '" . $columnComment . "'," . PHP_EOL; + } + } + $content = substr($content, 0, -1); + return $this->setBlankSpace($content, " "); + } + + + /** + * @notes 获取命名空间模板内容 + * @return string + * @author 段誉 + * @date 2022/6/22 18:18 + */ + public function getNameSpaceContent() + { + if (!empty($this->classDir)) { + return "namespace app\\" . $this->moduleName . "\\validate\\" . $this->classDir . ';'; + } + return "namespace app\\" . $this->moduleName . "\\validate;"; + } + + + /** + * @notes 获取类描述 + * @return string + * @author 段誉 + * @date 2022/6/22 18:18 + */ + public function getClassCommentContent() + { + if (!empty($this->tableData['class_comment'])) { + $tpl = $this->tableData['class_comment'] . '验证器'; + } else { + $tpl = $this->getUpperCamelName() . '验证器'; + } + return $tpl; + } + + + /** + * @notes 获取包名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:18 + */ + public function getPackageNameContent() + { + return !empty($this->classDir) ? '\\' . $this->classDir : ''; + } + + + /** + * @notes 获取文件生成到模块的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:18 + */ + public function getModuleGenerateDir() + { + $dir = $this->basePath . $this->moduleName . '/validate/'; + if (!empty($this->classDir)) { + $dir .= $this->classDir . '/'; + $this->checkDir($dir); + } + return $dir; + } + + + /** + * @notes 获取文件生成到runtime的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:18 + */ + public function getRuntimeGenerateDir() + { + $dir = $this->generatorDir . 'php/app/' . $this->moduleName . '/validate/'; + $this->checkDir($dir); + if (!empty($this->classDir)) { + $dir .= $this->classDir . '/'; + $this->checkDir($dir); + } + return $dir; + } + + + /** + * @notes 生成的文件名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:19 + */ + public function getGenerateName() + { + return $this->getUpperCamelName() . 'Validate.php'; + } + + + /** + * @notes 文件信息 + * @return array + * @author 段誉 + * @date 2022/6/23 15:57 + */ + public function fileInfo(): array + { + return [ + 'name' => $this->getGenerateName(), + 'type' => 'php', + 'content' => $this->content + ]; + } + + +} \ No newline at end of file diff --git a/app/common/service/generator/core/VueApiGenerator.php b/app/common/service/generator/core/VueApiGenerator.php new file mode 100644 index 0000000..aba12d4 --- /dev/null +++ b/app/common/service/generator/core/VueApiGenerator.php @@ -0,0 +1,144 @@ +getCommentContent(), + $this->getUpperCamelName(), + $this->getRouteContent(), + ]; + + $templatePath = $this->getTemplatePath('vue/api'); + + // 替换内容 + $content = $this->replaceFileData($needReplace, $waitReplace, $templatePath); + + $this->setContent($content); + } + + + /** + * @notes 描述 + * @return mixed + * @author 段誉 + * @date 2022/6/22 18:19 + */ + public function getCommentContent() + { + return $this->tableData['table_comment']; + } + + + /** + * @notes 路由名称 + * @return array|string|string[] + * @author 段誉 + * @date 2022/6/22 18:19 + */ + public function getRouteContent() + { + $content = $this->getTableName(); + if (!empty($this->classDir)) { + $content = $this->classDir . '.' . $this->getTableName(); + } + return Str::lower($content); + } + + + /** + * @notes 获取文件生成到模块的文件夹路径 + * @return mixed|void + * @author 段誉 + * @date 2022/6/22 18:19 + */ + public function getModuleGenerateDir() + { + $dir = dirname(app_path()) . '/admin/src/api/'; + $this->checkDir($dir); + return $dir; + } + + + /** + * @notes 获取文件生成到runtime的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:20 + */ + public function getRuntimeGenerateDir() + { + $dir = $this->generatorDir . 'vue/src/api/'; + $this->checkDir($dir); + return $dir; + } + + + /** + * @notes 生成的文件名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:20 + */ + public function getGenerateName() + { + return $this->getLowerTableName() . '.ts'; + } + + + /** + * @notes 文件信息 + * @return array + * @author 段誉 + * @date 2022/6/23 15:57 + */ + public function fileInfo(): array + { + return [ + 'name' => $this->getGenerateName(), + 'type' => 'ts', + 'content' => $this->content + ]; + } + + +} \ No newline at end of file diff --git a/app/common/service/generator/core/VueEditGenerator.php b/app/common/service/generator/core/VueEditGenerator.php new file mode 100644 index 0000000..43b2e95 --- /dev/null +++ b/app/common/service/generator/core/VueEditGenerator.php @@ -0,0 +1,493 @@ +getFormViewContent(), + $this->getUpperCamelName(), + $this->getDictDataContent(), + $this->getDictDataApiContent(), + $this->getFormDataContent(), + $this->getFormValidateContent(), + $this->tableData['table_comment'], + $this->getPkContent(), + $this->getTableName(), + $this->getCheckBoxJoinContent(), + $this->getCheckBoxSplitContent(), + $this->getFormDateContent(), + $this->getLowerCamelName(), + $this->getImportListsContent(), + $this->getTreeConstContent(), + $this->getTreeListsContent(), + ]; + + $templatePath = $this->getTemplatePath('vue/edit'); + + // 替换内容 + $content = $this->replaceFileData($needReplace, $waitReplace, $templatePath); + + $this->setContent($content); + } + + + /** + * @notes 复选框处理 + * @return string + * @author 段誉 + * @date 2022/6/24 19:30 + */ + public function getCheckBoxJoinContent() + { + $content = ''; + foreach ($this->tableColumn as $column) { + if (empty($column['view_type']) || $column['is_pk']) { + continue; + } + if ($column['view_type'] != 'checkbox') { + continue; + } + $content .= $column['column_name'] . ': formData.' . $column['column_name'] . '.join(",")' . PHP_EOL; + } + if (!empty($content)) { + $content = substr($content, 0, -1); + } + return $content; + } + + + /** + * @notes 复选框处理 + * @return string + * @author 段誉 + * @date 2022/6/24 19:30 + */ + public function getCheckBoxSplitContent() + { + $content = ''; + foreach ($this->tableColumn as $column) { + if (empty($column['view_type']) || $column['is_pk']) { + continue; + } + if ($column['view_type'] != 'checkbox') { + continue; + } + $content .= '//@ts-ignore' . PHP_EOL; + $content .= 'data.' . $column['column_name'] . ' && ' .'(formData.' . $column['column_name'] . ' = String(data.' . $column['column_name'] . ').split(","))' . PHP_EOL; + } + if (!empty($content)) { + $content = substr($content, 0, -1); + } + return $this->setBlankSpace($content, ' '); + } + + + /** + * @notes 树表contst + * @return string + * @author 段誉 + * @date 2022/12/22 18:19 + */ + public function getTreeConstContent() + { + $content = ""; + if ($this->isTreeCrud()) { + $content = file_get_contents($this->getTemplatePath('vue/other_item/editTreeConst')); + } + return $content; + } + + + /** + * @notes 获取树表列表 + * @return string + * @author 段誉 + * @date 2022/12/22 18:26 + */ + public function getTreeListsContent() + { + $content = ''; + if (!$this->isTreeCrud()) { + return $content; + } + + $needReplace = [ + '{TREE_ID}', + '{TREE_NAME}', + '{UPPER_CAMEL_NAME}', + ]; + $waitReplace = [ + $this->treeConfig['tree_id'], + $this->treeConfig['tree_name'], + $this->getUpperCamelName(), + ]; + + $templatePath = $this->getTemplatePath('vue/other_item/editTreeLists'); + if (file_exists($templatePath)) { + $content = $this->replaceFileData($needReplace, $waitReplace, $templatePath); + } + + return $content; + } + + + /** + * @notes 表单日期处理 + * @return string + * @author 段誉 + * @date 2022/6/27 16:45 + */ + public function getFormDateContent() + { + $content = ''; + foreach ($this->tableColumn as $column) { + if (empty($column['view_type']) || $column['is_pk']) { + continue; + } + if ($column['view_type'] != 'datetime' || $column['column_type'] != 'int') { + continue; + } + $content .= '//@ts-ignore' . PHP_EOL; + $content .= 'formData.' . $column['column_name'] . ' = timeFormat(formData.' . $column['column_name'] . ','."'yyyy-mm-dd hh:MM:ss'".') ' . PHP_EOL; + } + if (!empty($content)) { + $content = substr($content, 0, -1); + } + return $this->setBlankSpace($content, ' '); + } + + + /** + * @notes 获取表单内容 + * @return string + * @author 段誉 + * @date 2022/6/23 11:57 + */ + public function getFormViewContent() + { + $content = ''; + foreach ($this->tableColumn as $column) { + if (!$column['is_insert'] || !$column['is_update'] || $column['is_pk']) { + continue; + } + $needReplace = [ + '{COLUMN_COMMENT}', + '{COLUMN_NAME}', + '{DICT_TYPE}', + ]; + $waitReplace = [ + $column['column_comment'], + $column['column_name'], + $column['dict_type'], + ]; + + $viewType = $column['view_type']; + // 树表,树状结构下拉框 + if ($this->isTreeCrud() && $column['column_name'] == $this->treeConfig['tree_pid']) { + $viewType = 'treeSelect'; + array_push($needReplace, '{TREE_ID}', '{TREE_NAME}'); + array_push($waitReplace, $this->treeConfig['tree_id'], $this->treeConfig['tree_name']); + } + + $templatePath = $this->getTemplatePath('vue/form_item/' . $viewType); + if (!file_exists($templatePath)) { + continue; + } + + // 单选框值处理 + if ($column['view_type'] == 'radio' || $column['view_type'] == 'select') { + $stubItemValue = 'item.value'; + $intFieldValue = ['tinyint', 'smallint', 'mediumint', 'int', 'integer', 'bigint']; + if (in_array($column['column_type'], $intFieldValue)) { + $stubItemValue = 'parseInt(item.value)'; + } + array_push($needReplace, '{ITEM_VALUE}'); + array_push($waitReplace, $stubItemValue); + } + + $content .= $this->replaceFileData($needReplace, $waitReplace, $templatePath) . PHP_EOL; + } + + if (!empty($content)) { + $content = substr($content, 0, -1); + } + + $content = $this->setBlankSpace($content, ' '); + return $content; + } + + + /** + * @notes 获取字典数据内容 + * @return string + * @author 段誉 + * @date 2022/6/23 11:58 + */ + public function getDictDataContent() + { + $content = ''; + $isExist = []; + foreach ($this->tableColumn as $column) { + if (empty($column['dict_type']) || $column['is_pk']) { + continue; + } + if (in_array($column['dict_type'], $isExist)) { + continue; + } + $content .= $column['dict_type'] . ': ' . "[]," . PHP_EOL; + $isExist[] = $column['dict_type']; + } + if (!empty($content)) { + $content = substr($content, 0, -1); + } + return $this->setBlankSpace($content, ' '); + } + + + /** + * @notes 获取字典数据api内容 + * @return false|string + * @author 段誉 + * @date 2022/6/23 11:58 + */ + public function getDictDataApiContent() + { + $content = ''; + $isExist = []; + foreach ($this->tableColumn as $column) { + if (empty($column['dict_type']) || $column['is_pk']) { + continue; + } + if (in_array($column['dict_type'], $isExist)) { + continue; + } + $needReplace = [ + '{UPPER_CAMEL_NAME}', + '{DICT_TYPE}', + ]; + $waitReplace = [ + $this->getUpperCamelName(), + $column['dict_type'], + ]; + $templatePath = $this->getTemplatePath('vue/other_item/dictDataApi'); + if (!file_exists($templatePath)) { + continue; + } + $content .= $this->replaceFileData($needReplace, $waitReplace, $templatePath) . '' . PHP_EOL; + + $isExist[] = $column['dict_type']; + } + $content = substr($content, 0, -1); + return $content; + } + + + /** + * @notes 获取表单默认字段内容 + * @return string + * @author 段誉 + * @date 2022/6/23 15:15 + */ + public function getFormDataContent() + { + $content = ''; + $isExist = []; + foreach ($this->tableColumn as $column) { + if (!$column['is_insert'] || !$column['is_update'] || $column['is_pk']) { + continue; + } + if (in_array($column['column_name'], $isExist)) { + continue; + } + + // 复选框类型返回数组 + if ($column['view_type'] == 'checkbox') { + $content .= $column['column_name'] . ': ' . "[]," . PHP_EOL; + } else { + $content .= $column['column_name'] . ': ' . "''," . PHP_EOL; + } + + $isExist[] = $column['column_name']; + } + if (!empty($content)) { + $content = substr($content, 0, -1); + } + return $this->setBlankSpace($content, ' '); + } + + + /** + * @notes 表单验证内容 + * @return false|string + * @author 段誉 + * @date 2022/6/23 15:16 + */ + public function getFormValidateContent() + { + $content = ''; + $isExist = []; + $specDictType = ['input', 'textarea', 'editor']; + + foreach ($this->tableColumn as $column) { + if (!$column['is_required'] || $column['is_pk']) { + continue; + } + if (in_array($column['column_name'], $isExist)) { + continue; + } + + $validateMsg = in_array($column['view_type'], $specDictType) ? '请输入' : '请选择'; + $validateMsg .= $column['column_comment']; + + $needReplace = [ + '{COLUMN_NAME}', + '{VALIDATE_MSG}', + ]; + $waitReplace = [ + $column['column_name'], + $validateMsg, + ]; + $templatePath = $this->getTemplatePath('vue/other_item/formValidate'); + if (!file_exists($templatePath)) { + continue; + } + + $content .= $this->replaceFileData($needReplace, $waitReplace, $templatePath) . ',' . PHP_EOL; + + $isExist[] = $column['column_name']; + } + $content = substr($content, 0, -2); + return $content; + } + + + /** + * @notes 树表时导入列表 + * @author 段誉 + * @date 2022/12/23 9:56 + */ + public function getImportListsContent() + { + $content = ""; + if ($this->isTreeCrud()) { + $content = "api". $this->getUpperCamelName(). 'Lists,'; + } + + if (empty($content)) { + return $content; + } + + return $this->setBlankSpace($content, ' '); + } + + + /** + * @notes 获取文件生成到模块的文件夹路径 + * @return mixed|void + * @author 段誉 + * @date 2022/6/22 18:19 + */ + public function getModuleGenerateDir() + { + $dir = dirname(app_path()) . '/admin/src/views/' . $this->getTableName() . '/'; + $this->checkDir($dir); + return $dir; + } + + + /** + * @notes 获取文件生成到runtime的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:20 + */ + public function getRuntimeGenerateDir() + { + $dir = $this->generatorDir . 'vue/src/views/' . $this->getTableName() . '/'; + $this->checkDir($dir); + return $dir; + } + + + /** + * @notes 生成的文件名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:20 + */ + public function getGenerateName() + { + return 'edit.vue'; + } + + + /** + * @notes 文件信息 + * @return array + * @author 段誉 + * @date 2022/6/23 15:57 + */ + public function fileInfo(): array + { + return [ + 'name' => $this->getGenerateName(), + 'type' => 'vue', + 'content' => $this->content + ]; + } + + +} \ No newline at end of file diff --git a/app/common/service/generator/core/VueIndexGenerator.php b/app/common/service/generator/core/VueIndexGenerator.php new file mode 100644 index 0000000..a3c5b4f --- /dev/null +++ b/app/common/service/generator/core/VueIndexGenerator.php @@ -0,0 +1,308 @@ +getSearchViewContent(), + $this->getListsViewContent(), + $this->getUpperCamelName(), + $this->getQueryParamsContent(), + $this->getDictDataContent(), + $this->getPkContent(), + $this->getTableName(), + $this->getPermsContent(), + $this->getPermsContent('edit'), + $this->getPermsContent('delete'), + $this->getLowerCamelName() + ]; + + $templatePath = $this->getTemplatePath('vue/index'); + + if ($this->isTreeCrud()) { + // 插入树表相关 + array_push($needReplace, '{TREE_ID}', '{TREE_PID}'); + array_push($waitReplace, $this->treeConfig['tree_id'], $this->treeConfig['tree_pid']); + $templatePath = $this->getTemplatePath('vue/index-tree'); + } + + // 替换内容 + $content = $this->replaceFileData($needReplace, $waitReplace, $templatePath); + + $this->setContent($content); + } + + + /** + * @notes 获取搜索内容 + * @return string + * @author 段誉 + * @date 2022/6/23 11:57 + */ + public function getSearchViewContent() + { + $content = ''; + foreach ($this->tableColumn as $column) { + if (!$column['is_query'] || $column['is_pk']) { + continue; + } + + $needReplace = [ + '{COLUMN_COMMENT}', + '{COLUMN_NAME}', + '{DICT_TYPE}', + ]; + $waitReplace = [ + $column['column_comment'], + $column['column_name'], + $column['dict_type'], + ]; + + $searchStubType = $column['view_type']; + if ($column['view_type'] == 'radio') { + $searchStubType = 'select'; + } + + $templatePath = $this->getTemplatePath('vue/search_item/' . $searchStubType); + if (!file_exists($templatePath)) { + continue; + } + $content .= $this->replaceFileData($needReplace, $waitReplace, $templatePath) . PHP_EOL; + } + + if (!empty($content)) { + $content = substr($content, 0, -1); + } + + $content = $this->setBlankSpace($content, ' '); + return $content; + } + + + /** + * @notes 获取列表内容 + * @return string + * @author 段誉 + * @date 2022/6/23 11:57 + */ + public function getListsViewContent() + { + $content = ''; + foreach ($this->tableColumn as $column) { + if (!$column['is_lists']) { + continue; + } + + $needReplace = [ + '{COLUMN_COMMENT}', + '{COLUMN_NAME}', + '{DICT_TYPE}', + ]; + $waitReplace = [ + $column['column_comment'], + $column['column_name'], + $column['dict_type'], + ]; + + $templatePath = $this->getTemplatePath('vue/table_item/default'); + if ($column['view_type'] == 'imageSelect') { + $templatePath = $this->getTemplatePath('vue/table_item/image'); + } + if (in_array($column['view_type'], ['select', 'radio', 'checkbox'])) { + $templatePath = $this->getTemplatePath('vue/table_item/options'); + } + if ($column['column_type'] == 'int' && $column['view_type'] == 'datetime') { + $templatePath = $this->getTemplatePath('vue/table_item/datetime'); + } + if (!file_exists($templatePath)) { + continue; + } + + $content .= $this->replaceFileData($needReplace, $waitReplace, $templatePath) . PHP_EOL; + } + if (!empty($content)) { + $content = substr($content, 0, -1); + } + return $this->setBlankSpace($content, ' '); + } + + + /** + * @notes 获取查询条件内容 + * @return string + * @author 段誉 + * @date 2022/6/23 11:57 + */ + public function getQueryParamsContent() + { + $content = ''; + $queryDate = false; + foreach ($this->tableColumn as $column) { + if (!$column['is_query'] || $column['is_pk']) { + continue; + } + $content .= $column['column_name'] . ": ''," . PHP_EOL; + if ($column['query_type'] == 'between' && $column['view_type'] == 'datetime') { + $queryDate = true; + } + } + if ($queryDate) { + $content .= "start_time: ''," . PHP_EOL; + $content .= "end_time: ''," . PHP_EOL; + } + $content = substr($content, 0, -2); + return $this->setBlankSpace($content, ' '); + } + + + /** + * @notes 获取字典数据内容 + * @return string + * @author 段誉 + * @date 2022/6/23 11:58 + */ + public function getDictDataContent() + { + $content = ''; + $isExist = []; + foreach ($this->tableColumn as $column) { + if (empty($column['dict_type']) || $column['is_pk']) { + continue; + } + if (in_array($column['dict_type'], $isExist)) { + continue; + } + $content .= $column['dict_type'] .","; + $isExist[] = $column['dict_type']; + } + if (!empty($content)) { + $content = substr($content, 0, -1); + } + return $this->setBlankSpace($content, ''); + } + + + /** + * @notes 权限规则 + * @param string $type + * @return string + * @author 段誉 + * @date 2022/7/7 9:47 + */ + public function getPermsContent($type = 'add') + { + if (!empty($this->classDir)) { + $classDir = $this->classDir . '.'; + } else { + $classDir = ''; + } + return trim($classDir . $this->getLowerTableName() . '/' . $type); + } + + + /** + * @notes 获取文件生成到模块的文件夹路径 + * @return mixed|void + * @author 段誉 + * @date 2022/6/22 18:19 + */ + public function getModuleGenerateDir() + { + $dir = dirname(app()->getRootPath()) . '/admin/src/views/' . $this->getLowerTableName() . '/'; + $this->checkDir($dir); + return $dir; + } + + + /** + * @notes 获取文件生成到runtime的文件夹路径 + * @return string + * @author 段誉 + * @date 2022/6/22 18:20 + */ + public function getRuntimeGenerateDir() + { + $dir = $this->generatorDir . 'vue/src/views/' . $this->getLowerTableName() . '/'; + $this->checkDir($dir); + return $dir; + } + + + /** + * @notes 生成的文件名 + * @return string + * @author 段誉 + * @date 2022/6/22 18:20 + */ + public function getGenerateName() + { + return 'index.vue'; + } + + + /** + * @notes 文件信息 + * @return array + * @author 段誉 + * @date 2022/6/23 15:57 + */ + public function fileInfo(): array + { + return [ + 'name' => $this->getGenerateName(), + 'type' => 'vue', + 'content' => $this->content + ]; + } + + +} \ No newline at end of file diff --git a/app/common/service/generator/stub/php/controller.stub b/app/common/service/generator/stub/php/controller.stub new file mode 100644 index 0000000..44622d2 --- /dev/null +++ b/app/common/service/generator/stub/php/controller.stub @@ -0,0 +1,105 @@ +dataLists(new {UPPER_CAMEL_NAME}Lists()); + } + + + /** + * @notes 添加{NOTES} + * @return \think\response\Json + * @author {AUTHOR} + * @date {DATE} + */ + public function add() + { + $params = (new {UPPER_CAMEL_NAME}Validate())->post()->goCheck('add'); + $result = {UPPER_CAMEL_NAME}Logic::add($params); + if (true === $result) { + return $this->success('添加成功', [], 1, 1); + } + return $this->fail({UPPER_CAMEL_NAME}Logic::getError()); + } + + + /** + * @notes 编辑{NOTES} + * @return \think\response\Json + * @author {AUTHOR} + * @date {DATE} + */ + public function edit() + { + $params = (new {UPPER_CAMEL_NAME}Validate())->post()->goCheck('edit'); + $result = {UPPER_CAMEL_NAME}Logic::edit($params); + if (true === $result) { + return $this->success('编辑成功', [], 1, 1); + } + return $this->fail({UPPER_CAMEL_NAME}Logic::getError()); + } + + + /** + * @notes 删除{NOTES} + * @return \think\response\Json + * @author {AUTHOR} + * @date {DATE} + */ + public function delete() + { + $params = (new {UPPER_CAMEL_NAME}Validate())->post()->goCheck('delete'); + {UPPER_CAMEL_NAME}Logic::delete($params); + return $this->success('删除成功', [], 1, 1); + } + + + /** + * @notes 获取{NOTES}详情 + * @return \think\response\Json + * @author {AUTHOR} + * @date {DATE} + */ + public function detail() + { + $params = (new {UPPER_CAMEL_NAME}Validate())->goCheck('detail'); + $result = {UPPER_CAMEL_NAME}Logic::detail($params); + return $this->data($result); + } + + +} \ No newline at end of file diff --git a/app/common/service/generator/stub/php/lists.stub b/app/common/service/generator/stub/php/lists.stub new file mode 100644 index 0000000..ab69032 --- /dev/null +++ b/app/common/service/generator/stub/php/lists.stub @@ -0,0 +1,76 @@ +searchWhere) + ->field([{FIELD_DATA}]) + ->limit($this->limitOffset, $this->limitLength) + ->order(['{PK}' => 'desc']) + ->select() + ->toArray(); + } + + + /** + * @notes 获取{NOTES}数量 + * @return int + * @author {AUTHOR} + * @date {DATE} + */ + public function count(): int + { + return {UPPER_CAMEL_NAME}::where($this->searchWhere)->count(); + } + +} \ No newline at end of file diff --git a/app/common/service/generator/stub/php/logic.stub b/app/common/service/generator/stub/php/logic.stub new file mode 100644 index 0000000..4d4b7a6 --- /dev/null +++ b/app/common/service/generator/stub/php/logic.stub @@ -0,0 +1,106 @@ +getMessage()); + return false; + } + } + + + /** + * @notes 编辑{NOTES} + * @param array $params + * @return bool + * @author {AUTHOR} + * @date {DATE} + */ + public static function edit(array $params): bool + { + Db::startTrans(); + try { + {UPPER_CAMEL_NAME}::where('{PK}', $params['{PK}'])->update([ +{UPDATE_DATA} + ]); + + Db::commit(); + return true; + } catch (\Exception $e) { + Db::rollback(); + self::setError($e->getMessage()); + return false; + } + } + + + /** + * @notes 删除{NOTES} + * @param array $params + * @return bool + * @author {AUTHOR} + * @date {DATE} + */ + public static function delete(array $params): bool + { + return {UPPER_CAMEL_NAME}::destroy($params['{PK}']); + } + + + /** + * @notes 获取{NOTES}详情 + * @param $params + * @return array + * @author {AUTHOR} + * @date {DATE} + */ + public static function detail($params): array + { + return {UPPER_CAMEL_NAME}::findOrEmpty($params['{PK}'])->toArray(); + } +} \ No newline at end of file diff --git a/app/common/service/generator/stub/php/model.stub b/app/common/service/generator/stub/php/model.stub new file mode 100644 index 0000000..61acb2f --- /dev/null +++ b/app/common/service/generator/stub/php/model.stub @@ -0,0 +1,34 @@ +hasMany({RELATION_MODEL}::class, '{FOREIGN_KEY}', '{LOCAL_KEY}'); + } \ No newline at end of file diff --git a/app/common/service/generator/stub/php/model/has_one.stub b/app/common/service/generator/stub/php/model/has_one.stub new file mode 100644 index 0000000..21bd83b --- /dev/null +++ b/app/common/service/generator/stub/php/model/has_one.stub @@ -0,0 +1,11 @@ + + /** + * @notes 关联{RELATION_NAME} + * @return \think\model\relation\HasOne + * @author {AUTHOR} + * @date {DATE} + */ + public function {RELATION_NAME}() + { + return $this->hasOne({RELATION_MODEL}::class, '{FOREIGN_KEY}', '{LOCAL_KEY}'); + } \ No newline at end of file diff --git a/app/common/service/generator/stub/php/tree_lists.stub b/app/common/service/generator/stub/php/tree_lists.stub new file mode 100644 index 0000000..cee3870 --- /dev/null +++ b/app/common/service/generator/stub/php/tree_lists.stub @@ -0,0 +1,77 @@ +searchWhere) + ->field([{FIELD_DATA}]) + ->order(['{PK}' => 'desc']) + ->select() + ->toArray(); + + return linear_to_tree($lists, 'children', '{TREE_ID}', '{TREE_PID}'); + } + + + /** + * @notes 获取{NOTES}数量 + * @return int + * @author {AUTHOR} + * @date {DATE} + */ + public function count(): int + { + return {UPPER_CAMEL_NAME}::where($this->searchWhere)->count(); + } + +} \ No newline at end of file diff --git a/app/common/service/generator/stub/php/validate.stub b/app/common/service/generator/stub/php/validate.stub new file mode 100644 index 0000000..a984d47 --- /dev/null +++ b/app/common/service/generator/stub/php/validate.stub @@ -0,0 +1,94 @@ +only(['{PK}']); + } + + + /** + * @notes 详情场景 + * @return {UPPER_CAMEL_NAME}Validate + * @author {AUTHOR} + * @date {DATE} + */ + public function sceneDetail() + { + return $this->only(['{PK}']); + } + +} \ No newline at end of file diff --git a/app/common/service/generator/stub/sql/sql.stub b/app/common/service/generator/stub/sql/sql.stub new file mode 100644 index 0000000..c7861cb --- /dev/null +++ b/app/common/service/generator/stub/sql/sql.stub @@ -0,0 +1,13 @@ +INSERT INTO `{MENU_TABLE}`(`pid`, `type`, `name`, `icon`, `sort`, `perms`, `paths`, `component`, `selected`, `params`, `is_cache`, `is_show`, `is_disable`, `create_time`, `update_time`) + VALUES ({PARTNER_ID}, 'C', '{LISTS_NAME}', '', 1, '{PERMS_NAME}/lists', '{PATHS_NAME}', '{COMPONENT_NAME}/index', '', '', 0, 1, 0, {CREATE_TIME}, {UPDATE_TIME}); + +SELECT @pid := LAST_INSERT_ID(); + +INSERT INTO `{MENU_TABLE}`(`pid`, `type`, `name`, `icon`, `sort`, `perms`, `paths`, `component`, `selected`, `params`, `is_cache`, `is_show`, `is_disable`, `create_time`, `update_time`) + VALUES (@pid, 'A', '添加', '', 1, '{PERMS_NAME}/add', '', '', '', '', 0, 1, 0, {CREATE_TIME}, {UPDATE_TIME}); + +INSERT INTO `{MENU_TABLE}`(`pid`, `type`, `name`, `icon`, `sort`, `perms`, `paths`, `component`, `selected`, `params`, `is_cache`, `is_show`, `is_disable`, `create_time`, `update_time`) + VALUES (@pid, 'A', '编辑', '', 1, '{PERMS_NAME}/edit', '', '', '', '', 0, 1, 0, {CREATE_TIME}, {UPDATE_TIME}); + +INSERT INTO `{MENU_TABLE}`(`pid`, `type`, `name`, `icon`, `sort`, `perms`, `paths`, `component`, `selected`, `params`, `is_cache`, `is_show`, `is_disable`, `create_time`, `update_time`) + VALUES (@pid, 'A', '删除', '', 1, '{PERMS_NAME}/delete', '', '', '', '', 0, 1, 0, {CREATE_TIME}, {UPDATE_TIME}); diff --git a/app/common/service/generator/stub/vue/api.stub b/app/common/service/generator/stub/vue/api.stub new file mode 100644 index 0000000..1858e9a --- /dev/null +++ b/app/common/service/generator/stub/vue/api.stub @@ -0,0 +1,26 @@ +import request from '@/utils/request' + +// {COMMENT}列表 +export function api{UPPER_CAMEL_NAME}Lists(params: any) { + return request.get({ url: '/{ROUTE}/lists', params }) +} + +// 添加{COMMENT} +export function api{UPPER_CAMEL_NAME}Add(params: any) { + return request.post({ url: '/{ROUTE}/add', params }) +} + +// 编辑{COMMENT} +export function api{UPPER_CAMEL_NAME}Edit(params: any) { + return request.post({ url: '/{ROUTE}/edit', params }) +} + +// 删除{COMMENT} +export function api{UPPER_CAMEL_NAME}Delete(params: any) { + return request.post({ url: '/{ROUTE}/delete', params }) +} + +// {COMMENT}详情 +export function api{UPPER_CAMEL_NAME}Detail(params: any) { + return request.get({ url: '/{ROUTE}/detail', params }) +} \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/edit.stub b/app/common/service/generator/stub/vue/edit.stub new file mode 100644 index 0000000..3928bf8 --- /dev/null +++ b/app/common/service/generator/stub/vue/edit.stub @@ -0,0 +1,103 @@ + + + diff --git a/app/common/service/generator/stub/vue/form_item/checkbox.stub b/app/common/service/generator/stub/vue/form_item/checkbox.stub new file mode 100644 index 0000000..98e6329 --- /dev/null +++ b/app/common/service/generator/stub/vue/form_item/checkbox.stub @@ -0,0 +1,11 @@ + + + + {{ item.name }} + + + \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/form_item/datetime.stub b/app/common/service/generator/stub/vue/form_item/datetime.stub new file mode 100644 index 0000000..48d0dfa --- /dev/null +++ b/app/common/service/generator/stub/vue/form_item/datetime.stub @@ -0,0 +1,10 @@ + + + + diff --git a/app/common/service/generator/stub/vue/form_item/datetime2.stub b/app/common/service/generator/stub/vue/form_item/datetime2.stub new file mode 100644 index 0000000..b182bd9 --- /dev/null +++ b/app/common/service/generator/stub/vue/form_item/datetime2.stub @@ -0,0 +1,6 @@ + + + diff --git a/app/common/service/generator/stub/vue/form_item/editor.stub b/app/common/service/generator/stub/vue/form_item/editor.stub new file mode 100644 index 0000000..f150681 --- /dev/null +++ b/app/common/service/generator/stub/vue/form_item/editor.stub @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/form_item/imageSelect.stub b/app/common/service/generator/stub/vue/form_item/imageSelect.stub new file mode 100644 index 0000000..22703fc --- /dev/null +++ b/app/common/service/generator/stub/vue/form_item/imageSelect.stub @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/form_item/input.stub b/app/common/service/generator/stub/vue/form_item/input.stub new file mode 100644 index 0000000..ffe3884 --- /dev/null +++ b/app/common/service/generator/stub/vue/form_item/input.stub @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/form_item/radio.stub b/app/common/service/generator/stub/vue/form_item/radio.stub new file mode 100644 index 0000000..2768885 --- /dev/null +++ b/app/common/service/generator/stub/vue/form_item/radio.stub @@ -0,0 +1,11 @@ + + + + {{ item.name }} + + + \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/form_item/select.stub b/app/common/service/generator/stub/vue/form_item/select.stub new file mode 100644 index 0000000..5c2fa50 --- /dev/null +++ b/app/common/service/generator/stub/vue/form_item/select.stub @@ -0,0 +1,10 @@ + + + + + \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/form_item/textarea.stub b/app/common/service/generator/stub/vue/form_item/textarea.stub new file mode 100644 index 0000000..6f80318 --- /dev/null +++ b/app/common/service/generator/stub/vue/form_item/textarea.stub @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/form_item/treeSelect.stub b/app/common/service/generator/stub/vue/form_item/treeSelect.stub new file mode 100644 index 0000000..54c5dd9 --- /dev/null +++ b/app/common/service/generator/stub/vue/form_item/treeSelect.stub @@ -0,0 +1,13 @@ + + + \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/index-tree.stub b/app/common/service/generator/stub/vue/index-tree.stub new file mode 100644 index 0000000..1058e7b --- /dev/null +++ b/app/common/service/generator/stub/vue/index-tree.stub @@ -0,0 +1,167 @@ + + + + diff --git a/app/common/service/generator/stub/vue/index.stub b/app/common/service/generator/stub/vue/index.stub new file mode 100644 index 0000000..3c9a71b --- /dev/null +++ b/app/common/service/generator/stub/vue/index.stub @@ -0,0 +1,123 @@ + + + + diff --git a/app/common/service/generator/stub/vue/other_item/dictDataApi.stub b/app/common/service/generator/stub/vue/other_item/dictDataApi.stub new file mode 100644 index 0000000..fb9176a --- /dev/null +++ b/app/common/service/generator/stub/vue/other_item/dictDataApi.stub @@ -0,0 +1,6 @@ + dictDataLists({ + type_value: '{DICT_TYPE}', + page_type: 0 + }).then((res: any) => { + dictData.{DICT_TYPE} = res.lists + }) \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/other_item/editTreeConst.stub b/app/common/service/generator/stub/vue/other_item/editTreeConst.stub new file mode 100644 index 0000000..e98eb7d --- /dev/null +++ b/app/common/service/generator/stub/vue/other_item/editTreeConst.stub @@ -0,0 +1 @@ +const treeList = ref([]) \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/other_item/editTreeLists.stub b/app/common/service/generator/stub/vue/other_item/editTreeLists.stub new file mode 100644 index 0000000..5de63f3 --- /dev/null +++ b/app/common/service/generator/stub/vue/other_item/editTreeLists.stub @@ -0,0 +1,8 @@ +const getLists = async () => { + const data: any = await api{UPPER_CAMEL_NAME}Lists() + const item = { {TREE_ID}: 0, {TREE_NAME}: '顶级', children: [] } + item.children = data.lists + treeList.value.push(item) +} + +getLists() \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/other_item/formValidate.stub b/app/common/service/generator/stub/vue/other_item/formValidate.stub new file mode 100644 index 0000000..336052c --- /dev/null +++ b/app/common/service/generator/stub/vue/other_item/formValidate.stub @@ -0,0 +1,5 @@ + {COLUMN_NAME}: [{ + required: true, + message: '{VALIDATE_MSG}', + trigger: ['blur'] + }] \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/search_item/datetime.stub b/app/common/service/generator/stub/vue/search_item/datetime.stub new file mode 100644 index 0000000..e0e6882 --- /dev/null +++ b/app/common/service/generator/stub/vue/search_item/datetime.stub @@ -0,0 +1,6 @@ + + + diff --git a/app/common/service/generator/stub/vue/search_item/input.stub b/app/common/service/generator/stub/vue/search_item/input.stub new file mode 100644 index 0000000..9c09034 --- /dev/null +++ b/app/common/service/generator/stub/vue/search_item/input.stub @@ -0,0 +1,3 @@ + + + \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/search_item/select.stub b/app/common/service/generator/stub/vue/search_item/select.stub new file mode 100644 index 0000000..854152f --- /dev/null +++ b/app/common/service/generator/stub/vue/search_item/select.stub @@ -0,0 +1,11 @@ + + + + + + \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/table_item/datetime.stub b/app/common/service/generator/stub/vue/table_item/datetime.stub new file mode 100644 index 0000000..2cbbc6d --- /dev/null +++ b/app/common/service/generator/stub/vue/table_item/datetime.stub @@ -0,0 +1,5 @@ + + + \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/table_item/default.stub b/app/common/service/generator/stub/vue/table_item/default.stub new file mode 100644 index 0000000..54ae4c7 --- /dev/null +++ b/app/common/service/generator/stub/vue/table_item/default.stub @@ -0,0 +1 @@ + \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/table_item/image.stub b/app/common/service/generator/stub/vue/table_item/image.stub new file mode 100644 index 0000000..8ce9342 --- /dev/null +++ b/app/common/service/generator/stub/vue/table_item/image.stub @@ -0,0 +1,5 @@ + + + \ No newline at end of file diff --git a/app/common/service/generator/stub/vue/table_item/options.stub b/app/common/service/generator/stub/vue/table_item/options.stub new file mode 100644 index 0000000..8820cd8 --- /dev/null +++ b/app/common/service/generator/stub/vue/table_item/options.stub @@ -0,0 +1,5 @@ + + + \ No newline at end of file diff --git a/app/common/service/storage/Driver.php b/app/common/service/storage/Driver.php new file mode 100755 index 0000000..6740ebd --- /dev/null +++ b/app/common/service/storage/Driver.php @@ -0,0 +1,128 @@ +config = $config; + $this->engine = $this->getEngineClass($storage); + } + + /** + * 设置上传的文件信息 + * @param string $name + * @return mixed + */ + public function setUploadFile($name = 'iFile') + { + return $this->engine->setUploadFile($name); + } + + /** + * 设置上传的文件信息 + * @param string $filePath + * @return mixed + */ + public function setUploadFileByReal($filePath) + { + return $this->engine->setUploadFileByReal($filePath); + } + + /** + * 执行文件上传 + * @param $save_dir (保存路径) + * @return mixed + */ + public function upload($save_dir) + { + return $this->engine->upload($save_dir); + } + + /** + * Notes: 抓取网络资源 + * @param $url + * @param $key + * @author 张无忌(2021/3/2 14:16) + * @return mixed + */ + public function fetch($url, $key) { + return $this->engine->fetch($url, $key); + } + + /** + * 执行文件删除 + * @param $fileName + * @return mixed + */ + public function delete($fileName) + { + return $this->engine->delete($fileName); + } + + /** + * 获取错误信息 + * @return mixed + */ + public function getError() + { + return $this->engine->getError(); + } + + /** + * 获取文件路径 + * @return mixed + */ + public function getFileName() + { + return $this->engine->getFileName(); + } + + /** + * 返回文件信息 + * @return mixed + */ + public function getFileInfo() + { + return $this->engine->getFileInfo(); + } + + /** + * 获取当前的存储引擎 + * @param null|string $storage 指定存储方式,如不指定则为系统默认 + * @return mixed + * @throws Exception + */ + private function getEngineClass($storage = null) + { + $engineName = is_null($storage) ? $this->config['default'] : $storage; + $classSpace = __NAMESPACE__ . '\\engine\\' . ucfirst($engineName); + if (!class_exists($classSpace)) { + throw new Exception('未找到存储引擎类: ' . $engineName); + } + if($engineName == 'local') { + return new $classSpace(); + } + + return new $classSpace($this->config['engine'][$engineName]); + } + +} diff --git a/app/common/service/storage/engine/Aliyun.php b/app/common/service/storage/engine/Aliyun.php new file mode 100755 index 0000000..ad418b4 --- /dev/null +++ b/app/common/service/storage/engine/Aliyun.php @@ -0,0 +1,114 @@ +config = $config; + } + + /** + * 执行上传 + * @param $save_dir (保存路径) + * @return bool|mixed + */ + public function upload($save_dir) + { + try { + $ossClient = new OssClient( + $this->config['access_key'], + $this->config['secret_key'], + $this->config['domain'] + ); + $ossClient->uploadFile( + $this->config['bucket'], + $this->fileName, + $this->getRealPath() + ); + } catch (OssException $e) { + $this->error = $e->getMessage(); + return false; + } + return true; + } + + /** + * Notes: 抓取远程资源 + * @param $url + * @param null $key + * @return mixed|void + * @author 张无忌(2021/3/2 14:36) + */ + public function fetch($url, $key = null) + { + try { + $ossClient = new OssClient( + $this->config['access_key'], + $this->config['secret_key'], + $this->config['domain'], + true + ); + + $content = file_get_contents($url); + $ossClient->putObject( + $this->config['bucket'], + $key, + $content + ); + } catch (OssException $e) { + $this->error = $e->getMessage(); + return false; + } + return true; + } + + /** + * 删除文件 + * @param $fileName + * @return bool|mixed + */ + public function delete($fileName) + { + try { + $ossClient = new OssClient( + $this->config['access_key'], + $this->config['access_key'], + $this->config['domain'], + true + ); + $ossClient->deleteObject($this->config['bucket'], $fileName); + } catch (OssException $e) { + $this->error = $e->getMessage(); + return false; + } + return true; + } + + /** + * 返回文件路径 + * @return mixed + */ + public function getFileName() + { + return $this->fileName; + } + +} diff --git a/app/common/service/storage/engine/Local.php b/app/common/service/storage/engine/Local.php new file mode 100755 index 0000000..8d26a38 --- /dev/null +++ b/app/common/service/storage/engine/Local.php @@ -0,0 +1,59 @@ +file->move($save_dir.'/'.$this->fileName); + if (empty($info)) { + $this->error = $this->file->getError(); + return false; + } + return true; + } + + public function fetch($url, $key=null) {} + + /** + * 删除文件 + * @param $fileName + * @return bool|mixed + */ + public function delete($fileName) + { + $check = strpos($fileName, '/'); + if ($check !== false && $check == 0) { + // 文件所在目录 + $fileName = substr_replace($fileName,"",0,1); + } + $filePath = public_path() . "{$fileName}"; + return !file_exists($filePath) ?: unlink($filePath); + } + + /** + * 返回文件路径 + * @return mixed + */ + public function getFileName() + { + return $this->fileName; + } +} diff --git a/app/common/service/storage/engine/Qcloud.php b/app/common/service/storage/engine/Qcloud.php new file mode 100755 index 0000000..5328013 --- /dev/null +++ b/app/common/service/storage/engine/Qcloud.php @@ -0,0 +1,116 @@ +config = $config; + // 创建COS控制类 + $this->createCosClient(); + } + + /** + * 创建COS控制类 + */ + private function createCosClient() + { + $this->cosClient = new Client([ + 'region' => $this->config['region'], + 'credentials' => [ + 'secretId' => $this->config['access_key'], + 'secretKey' => $this->config['secret_key'], + ], + ]); + } + + /** + * 执行上传 + * @param $save_dir (保存路径) + * @return bool|mixed + */ + public function upload($save_dir) + { + // 上传文件 + // putObject(上传接口,最大支持上传5G文件) + try { + $result = $this->cosClient->putObject([ + 'Bucket' => $this->config['bucket'], + 'Key' => $save_dir . '/' . $this->fileName, + 'Body' => fopen($this->getRealPath(), 'rb') + ]); + return true; + } catch (Exception $e) { + $this->error = $e->getMessage(); + return false; + } + } + + /** + * notes: 抓取远程资源(最大支持上传5G文件) + * @param $url + * @param null $key + * @author 张无忌(2021/3/2 14:36) + * @return mixed|void + */ + public function fetch($url, $key=null) { + try { + $this->cosClient->putObject([ + 'Bucket' => $this->config['bucket'], + 'Key' => $key, + 'Body' => fopen($url, 'rb') + ]); + return true; + } catch (Exception $e) { + $this->error = $e->getMessage(); + return false; + } + } + + /** + * 删除文件 + * @param $fileName + * @return bool|mixed + */ + public function delete($fileName) + { + try { + $this->cosClient->deleteObject(array( + 'Bucket' => $this->config['bucket'], + 'Key' => $fileName + )); + return true; + } catch (Exception $e) { + $this->error = $e->getMessage(); + return false; + } + } + + /** + * 返回文件路径 + * @return mixed + */ + public function getFileName() + { + return $this->fileName; + } + +} diff --git a/app/common/service/storage/engine/Qiniu.php b/app/common/service/storage/engine/Qiniu.php new file mode 100755 index 0000000..c04a03a --- /dev/null +++ b/app/common/service/storage/engine/Qiniu.php @@ -0,0 +1,136 @@ +config = $config; + } + + /** + * @notes 执行上传 + * @param $save_dir + * @return bool|mixed + * @author 张无忌 + * @date 2021/7/27 16:02 + */ + public function upload($save_dir) + { + // 要上传图片的本地路径 + $realPath = $this->getRealPath(); + + // 构建鉴权对象 + $auth = new Auth($this->config['access_key'], $this->config['secret_key']); + + // 要上传的空间 + $token = $auth->uploadToken($this->config['bucket']); + + // 初始化 UploadManager 对象并进行文件的上传 + $uploadMgr = new UploadManager(); + + try { + // 调用 UploadManager 的 putFile 方法进行文件的上传 + $key = $save_dir . '/' . $this->fileName; + list(, $error) = $uploadMgr->putFile($token, $key, $realPath); + + if ($error !== null) { + $this->error = $error->message(); + return false; + } + return true; + } catch (Exception $e) { + $this->error = $e->getMessage(); + return false; + } + } + + /** + * @notes 抓取远程资源 + * @param $url + * @param null $key + * @return bool|mixed + * @author 张无忌 + * @date 2021/7/27 16:02 + */ + public function fetch($url, $key=null) + { + try { + if (substr($url, 0, 1) !== '/' || strstr($url, 'http://') || strstr($url, 'https://')) { + $auth = new Auth($this->config['access_key'], $this->config['secret_key']); + $bucketManager = new BucketManager($auth); + list(, $err) = $bucketManager->fetch($url, $this->config['bucket'], $key); + } else { + $auth = new Auth($this->config['access_key'], $this->config['secret_key']); + $token = $auth->uploadToken($this->config['bucket']); + $uploadMgr = new UploadManager(); + list(, $err) = $uploadMgr->putFile($token, $key, $url); + } + + if ($err !== null) { + $this->error = $err->message(); + return false; + } + + return true; + } catch (Exception $e) { + $this->error = $e->getMessage(); + return false; + } + } + + /** + * @notes 删除文件 + * @param $fileName + * @return bool|mixed + * @author 张无忌 + * @date 2021/7/27 16:02 + */ + public function delete($fileName) + { + // 构建鉴权对象 + $auth = new Auth($this->config['access_key'], $this->config['secret_key']); + // 初始化 UploadManager 对象并进行文件的上传 + $bucketMgr = new BucketManager($auth); + + try { + list($res, $error) = $bucketMgr->delete($this->config['bucket'], $fileName); + if ($error !== null) { + $this->error = $error->message(); + return false; + } + return true; + } catch (Exception $e) { + $this->error = $e->getMessage(); + return false; + } + } + + /** + * 返回文件路径 + * @return mixed + */ + public function getFileName() + { + return $this->fileName; + } +} diff --git a/app/common/service/storage/engine/Server.php b/app/common/service/storage/engine/Server.php new file mode 100755 index 0000000..bd19040 --- /dev/null +++ b/app/common/service/storage/engine/Server.php @@ -0,0 +1,151 @@ +file = request()->file($name); + + if (empty($this->file)) { + throw new Exception('未找到上传文件的信息'); + } + $this->file->extension = pathinfo($this->file->getUploadName(), PATHINFO_EXTENSION); + $finfo = finfo_open(FILEINFO_MIME_TYPE); + $this->file->getMime = finfo_file($finfo, $this->file->getPathname()); + // 校验上传文件后缀 + $limit = array_merge(config('project.file_image'), config('project.file_video')); + if (!in_array(strtolower($this->file->extension), $limit)) { + throw new Exception('不允许上传' . $this->file->extension . '后缀文件'); + } + + // 文件信息 + $this->fileInfo = [ + 'ext' => $this->file->extension, + 'size' => $this->file->getSize(), + 'mime' => $this->file->getMime, + 'name' => $this->file->getUploadName(), + 'realPath' => $this->file->getRealPath(), + ]; + // 生成保存文件名 + $this->fileName = $this->buildSaveName(); + } + + + /** + * 设置上传的文件信息 + * @param string $filePath + */ + public function setUploadFileByReal($filePath) + { + // 设置为系统内部上传 + $this->isInternal = true; + // 文件信息 + $this->fileInfo = [ + 'name' => basename($filePath), + 'size' => filesize($filePath), + 'tmp_name' => $filePath, + 'error' => 0, + ]; + // 生成保存文件名 + $this->fileName = $this->buildSaveName(); + } + + /** + * Notes: 抓取网络资源 + * @param $url + * @param $key + * @author 张无忌(2021/3/2 14:15) + * @return mixed + */ + abstract protected function fetch($url, $key); + + /** + * 文件上传 + * @param $save_dir (保存路径) + * @return mixed + */ + abstract protected function upload($save_dir); + + /** + * 文件删除 + * @param $fileName + * @return mixed + */ + abstract protected function delete($fileName); + + /** + * 返回上传后文件路径 + * @return mixed + */ + abstract public function getFileName(); + + /** + * 返回文件信息 + * @return mixed + */ + public function getFileInfo() + { + return $this->fileInfo; + } + + protected function getRealPath() + { + return $this->fileInfo['realPath']; + } + + /** + * 返回错误信息 + * @return mixed + */ + public function getError() + { + return $this->error; + } + + /** + * 生成保存文件名 + */ + private function buildSaveName() + { + // 要上传图片的本地路径 + $realPath = $this->getRealPath(); + // 扩展名 + $ext = pathinfo($this->getFileInfo()['name'], PATHINFO_EXTENSION); + // 自动生成文件名 + return date('YmdHis') . substr(md5($realPath), 0, 5) + . str_pad(rand(0, 9999), 4, '0', STR_PAD_LEFT) . ".{$ext}"; + } + +} diff --git a/app/common/validate/BaseValidate.php b/app/common/validate/BaseValidate.php new file mode 100755 index 0000000..5be1236 --- /dev/null +++ b/app/common/validate/BaseValidate.php @@ -0,0 +1,87 @@ +method() == 'POST') { + JsonService::throw('请求方式错误,请使用post请求方式'); + } + $this->method = 'POST'; + return $this; + } + + /** + * @notes 设置请求方式 + * @author 令狐冲 + * @date 2021/12/27 14:13 + */ + public function get() + { + if (!request()->method() == 'GET') { + JsonService::throw('请求方式错误,请使用get请求方式'); + } + return $this; + } + + + /** + * @notes 切面验证接收到的参数 + * @param null $scene 场景验证 + * @param array $validateData 验证参数,可追加和覆盖掉接收的参数 + * @return array + * @author 令狐冲 + * @date 2021/12/27 14:13 + */ + public function goCheck($scene = null, array $validateData = []): array + { + //接收参数 + if ($this->method == 'GET') { + $params = request()->get(); + } else { + $params = request()->post(); + } + //合并验证参数 + $params = array_merge($params, $validateData); + + //场景 + if ($scene) { + $result = $this->scene($scene)->check($params); + } else { + $result = $this->check($params); + } + + if (!$result) { + $exception = is_array($this->error) ? implode(';', $this->error) : $this->error; + JsonService::throw($exception); + } + // 3.成功返回数据 + return $params; + } +} \ No newline at end of file diff --git a/app/common/validate/ListsValidate.php b/app/common/validate/ListsValidate.php new file mode 100755 index 0000000..6bf1fcd --- /dev/null +++ b/app/common/validate/ListsValidate.php @@ -0,0 +1,66 @@ + 'integer|gt:0', + 'page_size' => 'integer|gt:0|pageSizeMax', + 'page_start' => 'integer|gt:0', + 'page_end' => 'integer|gt:0|egt:page_start', + 'page_type' => 'in:0,1', + 'order_by' => 'in:desc,asc', + 'start_time' => 'date', + 'end_time' => 'date|gt:start_time', + 'start' => 'number', + 'end' => 'number', + 'export' => 'in:1,2', + ]; + + protected $message = [ + 'page_end.egt' => '导出范围设置不正确,请重新选择', + 'end_time.gt' => '搜索的时间范围不正确', + ]; + + /** + * @notes 查询数据量判断 + * @param $value + * @param $rule + * @param $data + * @return bool + * @author 令狐冲 + * @date 2021/7/30 15:13 + */ + public function pageSizeMax($value, $rule, $data) + { + $pageSizeMax = Config::get('project.lists.page_size_max'); + if ($pageSizeMax < $value) { + return '已超出系统限制数量,请分页查询或导出,' . '当前最多记录数为:' . $pageSizeMax; + } + return true; + } + + +} \ No newline at end of file diff --git a/app/functions.php b/app/functions.php new file mode 100644 index 0000000..53126ae --- /dev/null +++ b/app/functions.php @@ -0,0 +1,199 @@ += 0 ? true : false; +} + +/** + * @notes 检查文件是否可写 + * @param string $dir + * @return bool + * @author 乔峰 + * @date 2021/12/28 18:27 + */ +function check_dir_write(string $dir = '') : bool +{ + $route = base_path() . '/' . $dir; + return is_writable($route); +} + +/** + * @notes 随机生成token值 + * @param string $extra + * @return string + * @author 乔峰 + * @date 2021/12/28 18:24 + */ +function create_token(string $extra = '') : string +{ + return md5($extra . time()); +} + +/** + * @notes 生成密码加密密钥 + * @param string $plaintext + * @param string $salt + * @return string + * @author 段誉 + * @date 2021/12/28 18:24 + */ +function create_password(string $plaintext, string $salt) : string +{ + return md5($salt . md5($plaintext . $salt)); +} + +/** + * 多级线性结构排序 + * 转换前: + * [{"id":1,"pid":0,"name":"a"},{"id":2,"pid":0,"name":"b"},{"id":3,"pid":1,"name":"c"}, + * {"id":4,"pid":2,"name":"d"},{"id":5,"pid":4,"name":"e"},{"id":6,"pid":5,"name":"f"}, + * {"id":7,"pid":3,"name":"g"}] + * 转换后: + * [{"id":1,"pid":0,"name":"a","level":1},{"id":3,"pid":1,"name":"c","level":2},{"id":7,"pid":3,"name":"g","level":3}, + * {"id":2,"pid":0,"name":"b","level":1},{"id":4,"pid":2,"name":"d","level":2},{"id":5,"pid":4,"name":"e","level":3}, + * {"id":6,"pid":5,"name":"f","level":4}] + * @param array $data 线性结构数组 + * @param string $symbol 名称前面加符号 + * @param string $name 名称 + * @param string $id_name 数组id名 + * @param string $parent_id_name 数组祖先id名 + * @param int $level 此值请勿给参数 + * @param int $parent_id 此值请勿给参数 + * @return array + */ +function linear_to_tree($data, $sub_key_name = 'sub', $id_name = 'id', $parent_id_name = 'pid', $parent_id = 0) +{ + $tree = []; + foreach ($data as $row) { + if ($row[$parent_id_name] == $parent_id) { + $temp = $row; + $child = linear_to_tree($data, $sub_key_name, $id_name, $parent_id_name, $row[$id_name]); + if ($child) { + $temp[$sub_key_name] = $child; + } + $tree[] = $temp; + } + } + return $tree; +} + +function createDir($path){ + if (is_dir($path)) { + return true; + } + + $parent = dirname($path); + if (!is_dir($parent)) { + if (!createDir($parent)) { + return false; + } + } + return mkdir($path); +} + +if (!function_exists('download')) { + /** + * 获取\think\response\Download对象实例 + * @param string $filename 要下载的文件 + * @param string $name 显示文件名 + * @param bool $content 是否为内容 + * @param int $expire 有效期(秒) + * @return \think\response\File + */ + function download(string $filename, string $name = '', bool $content = false, int $expire = 180) + { + + return response()->download($filename,$name); + } +} + +/** + * @notes 删除目标目录 + * @param $path + * @param $delDir + * @return bool|void + * @author 段誉 + * @date 2022/4/8 16:30 + */ +function del_target_dir($path, $delDir) +{ + //没找到,不处理 + if (!file_exists($path)) { + return false; + } + + //打开目录句柄 + $handle = opendir($path); + if ($handle) { + while (false !== ($item = readdir($handle))) { + if ($item != "." && $item != "..") { + if (is_dir("$path/$item")) { + del_target_dir("$path/$item", $delDir); + } else { + unlink("$path/$item"); + } + } + } + closedir($handle); + if ($delDir) { + return rmdir($path); + } + } else { + if (file_exists($path)) { + return unlink($path); + } + return false; + } +} + +/** + * @notes 获取无前缀数据表名 + * @param $tableName + * @return mixed|string + * @author 段誉 + * @date 2022/12/12 15:23 + */ +function get_no_prefix_table_name($tableName) +{ + $tablePrefix = config('database.connections.mysql.prefix'); + $prefixIndex = strpos($tableName, $tablePrefix); + if ($prefixIndex !== 0 || $prefixIndex === false) { + return $tableName; + } + $tableName = substr_replace($tableName, '', 0, strlen($tablePrefix)); + return trim($tableName); +} \ No newline at end of file diff --git a/composer.json b/composer.json new file mode 100644 index 0000000..ab51cde --- /dev/null +++ b/composer.json @@ -0,0 +1,75 @@ +{ + "name": "workerman/webman", + "type": "project", + "keywords": [ + "high performance", + "http service" + ], + "homepage": "https://www.workerman.net", + "license": "MIT", + "description": "High performance HTTP Service Framework.", + "authors": [ + { + "name": "walkor", + "email": "walkor@workerman.net", + "homepage": "https://www.workerman.net", + "role": "Developer" + } + ], + "support": { + "email": "walkor@workerman.net", + "issues": "https://github.com/walkor/webman/issues", + "forum": "https://wenda.workerman.net/", + "wiki": "https://workerman.net/doc/webman", + "source": "https://github.com/walkor/webman" + }, + "require": { + "php": ">=7.2", + "workerman/webman-framework": "^1.4.7", + "monolog/monolog": "^2.0", + "webman/think-orm": "^1.0", + "vlucas/phpdotenv": "^5.4", + "psr/container": "^1.1.1", + "ext-json": "*", + "phpoffice/phpspreadsheet": "^1.19", + "aliyuncs/oss-sdk-php": "^2.6", + "webman/console": "^1.2.12", + "qiniu/php-sdk": "7.4", + "qcloud/cos-sdk-v5": "^2.6", + "dragonmantank/cron-expression": "^3.3", + "tinywan/storage": "^0.3.4", + "webman/log": "^1.1", + "taoser/webman-validate": "^1.7", + "php-di/php-di": "^6", + "doctrine/annotations": "^1.14", + "illuminate/redis": "^10.22", + "symfony/cache": "^5.4", + "max/var-dumper": "^0.1.1", + "textalk/websocket": "^1.5" + }, + "suggest": { + "ext-event": "For better performance. " + }, + "autoload": { + "psr-4": { + "": "./", + "app\\": "./app", + "App\\": "./app", + "app\\View\\Components\\": "./app/view/components" + }, + "files": [ + "./support/helpers.php" + ] + }, + "scripts": { + "post-package-install": [ + "support\\Plugin::install" + ], + "post-package-update": [ + "support\\Plugin::install" + ], + "pre-package-uninstall": [ + "support\\Plugin::uninstall" + ] + } +} diff --git a/composer.lock b/composer.lock new file mode 100644 index 0000000..860c272 --- /dev/null +++ b/composer.lock @@ -0,0 +1,3807 @@ +{ + "_readme": [ + "This file locks the dependencies of your project to a known state", + "Read more about it at https://getcomposer.org/doc/01-basic-usage.md#installing-dependencies", + "This file is @generated automatically" + ], + "content-hash": "180fd50c18ae54063515e2bae5b5e329", + "packages": [ + { + "name": "aliyuncs/oss-sdk-php", + "version": "v2.6.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/aliyuncs/oss-sdk-php/v2.6.0/aliyuncs-oss-sdk-php-v2.6.0.zip", + "reference": "572d0f8e099e8630ae7139ed3fdedb926c7a760f", + "shasum": "" + }, + "require": { + "php": ">=5.3" + }, + "require-dev": { + "phpunit/phpunit": "*", + "satooshi/php-coveralls": "*" + }, + "type": "library", + "autoload": { + "psr-4": { + "OSS\\": "src/OSS" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Aliyuncs", + "homepage": "http://www.aliyun.com" + } + ], + "description": "Aliyun OSS SDK for PHP", + "homepage": "http://www.aliyun.com/product/oss/", + "time": "2022-08-03T08:06:01+00:00" + }, + { + "name": "doctrine/annotations", + "version": "1.14.3", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/doctrine/annotations/1.14.3/doctrine-annotations-1.14.3.zip", + "reference": "fb0d71a7393298a7b232cbf4c8b1f73f3ec3d5af", + "shasum": "" + }, + "require": { + "doctrine/lexer": "^1 || ^2", + "ext-tokenizer": "*", + "php": "^7.1 || ^8.0", + "psr/cache": "^1 || ^2 || ^3" + }, + "require-dev": { + "doctrine/cache": "^1.11 || ^2.0", + "doctrine/coding-standard": "^9 || ^10", + "phpstan/phpstan": "~1.4.10 || ^1.8.0", + "phpunit/phpunit": "^7.5 || ^8.5 || ^9.5", + "symfony/cache": "^4.4 || ^5.4 || ^6", + "vimeo/psalm": "^4.10" + }, + "suggest": { + "php": "PHP 8.0 or higher comes with attributes, a native replacement for annotations" + }, + "type": "library", + "autoload": { + "psr-4": { + "Doctrine\\Common\\Annotations\\": "lib/Doctrine/Common/Annotations" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Guilherme Blanco", + "email": "guilhermeblanco@gmail.com" + }, + { + "name": "Roman Borschel", + "email": "roman@code-factory.org" + }, + { + "name": "Benjamin Eberlei", + "email": "kontakt@beberlei.de" + }, + { + "name": "Jonathan Wage", + "email": "jonwage@gmail.com" + }, + { + "name": "Johannes Schmitt", + "email": "schmittjoh@gmail.com" + } + ], + "description": "Docblock Annotations Parser", + "homepage": "https://www.doctrine-project.org/projects/annotations.html", + "keywords": [ + "annotations", + "docblock", + "parser" + ], + "time": "2023-02-01T09:20:38+00:00" + }, + { + "name": "doctrine/deprecations", + "version": "v1.1.1", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/doctrine/deprecations/v1.1.1/doctrine-deprecations-v1.1.1.zip", + "reference": "612a3ee5ab0d5dd97b7cf3874a6efe24325efac3", + "shasum": "" + }, + "require": { + "php": "^7.1 || ^8.0" + }, + "require-dev": { + "doctrine/coding-standard": "^9", + "phpstan/phpstan": "1.4.10 || 1.10.15", + "phpstan/phpstan-phpunit": "^1.0", + "phpunit/phpunit": "^7.5 || ^8.5 || ^9.5", + "psalm/plugin-phpunit": "0.18.4", + "psr/log": "^1 || ^2 || ^3", + "vimeo/psalm": "4.30.0 || 5.12.0" + }, + "suggest": { + "psr/log": "Allows logging deprecations via PSR-3 logger implementation" + }, + "type": "library", + "autoload": { + "psr-4": { + "Doctrine\\Deprecations\\": "lib/Doctrine/Deprecations" + } + }, + "license": [ + "MIT" + ], + "description": "A small layer on top of trigger_error(E_USER_DEPRECATED) or PSR-3 logging with options to disable all deprecations or selectively for packages.", + "homepage": "https://www.doctrine-project.org/", + "time": "2023-06-03T09:27:29+00:00" + }, + { + "name": "doctrine/inflector", + "version": "2.0.8", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/doctrine/inflector/2.0.8/doctrine-inflector-2.0.8.zip", + "reference": "f9301a5b2fb1216b2b08f02ba04dc45423db6bff", + "shasum": "" + }, + "require": { + "php": "^7.2 || ^8.0" + }, + "require-dev": { + "doctrine/coding-standard": "^11.0", + "phpstan/phpstan": "^1.8", + "phpstan/phpstan-phpunit": "^1.1", + "phpstan/phpstan-strict-rules": "^1.3", + "phpunit/phpunit": "^8.5 || ^9.5", + "vimeo/psalm": "^4.25 || ^5.4" + }, + "type": "library", + "autoload": { + "psr-4": { + "Doctrine\\Inflector\\": "lib/Doctrine/Inflector" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Guilherme Blanco", + "email": "guilhermeblanco@gmail.com" + }, + { + "name": "Roman Borschel", + "email": "roman@code-factory.org" + }, + { + "name": "Benjamin Eberlei", + "email": "kontakt@beberlei.de" + }, + { + "name": "Jonathan Wage", + "email": "jonwage@gmail.com" + }, + { + "name": "Johannes Schmitt", + "email": "schmittjoh@gmail.com" + } + ], + "description": "PHP Doctrine Inflector is a small library that can perform string manipulations with regard to upper/lowercase and singular/plural forms of words.", + "homepage": "https://www.doctrine-project.org/projects/inflector.html", + "keywords": [ + "inflection", + "inflector", + "lowercase", + "manipulation", + "php", + "plural", + "singular", + "strings", + "uppercase", + "words" + ], + "time": "2023-06-16T13:40:37+00:00" + }, + { + "name": "doctrine/lexer", + "version": "2.1.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/doctrine/lexer/2.1.0/doctrine-lexer-2.1.0.zip", + "reference": "39ab8fcf5a51ce4b85ca97c7a7d033eb12831124", + "shasum": "" + }, + "require": { + "doctrine/deprecations": "^1.0", + "php": "^7.1 || ^8.0" + }, + "require-dev": { + "doctrine/coding-standard": "^9 || ^10", + "phpstan/phpstan": "^1.3", + "phpunit/phpunit": "^7.5 || ^8.5 || ^9.5", + "psalm/plugin-phpunit": "^0.18.3", + "vimeo/psalm": "^4.11 || ^5.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Doctrine\\Common\\Lexer\\": "src" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Guilherme Blanco", + "email": "guilhermeblanco@gmail.com" + }, + { + "name": "Roman Borschel", + "email": "roman@code-factory.org" + }, + { + "name": "Johannes Schmitt", + "email": "schmittjoh@gmail.com" + } + ], + "description": "PHP Doctrine Lexer parser library that can be used in Top-Down, Recursive Descent Parsers.", + "homepage": "https://www.doctrine-project.org/projects/lexer.html", + "keywords": [ + "annotations", + "docblock", + "lexer", + "parser", + "php" + ], + "time": "2022-12-14T08:49:07+00:00" + }, + { + "name": "dragonmantank/cron-expression", + "version": "v3.3.3", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/dragonmantank/cron-expression/v3.3.3/dragonmantank-cron-expression-v3.3.3.zip", + "reference": "adfb1f505deb6384dc8b39804c5065dd3c8c8c0a", + "shasum": "" + }, + "require": { + "php": "^7.2|^8.0", + "webmozart/assert": "^1.0" + }, + "replace": { + "mtdowling/cron-expression": "^1.0" + }, + "require-dev": { + "phpstan/extension-installer": "^1.0", + "phpstan/phpstan": "^1.0", + "phpstan/phpstan-webmozart-assert": "^1.0", + "phpunit/phpunit": "^7.0|^8.0|^9.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Cron\\": "src/Cron/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Chris Tankersley", + "email": "chris@ctankersley.com", + "homepage": "https://github.com/dragonmantank" + } + ], + "description": "CRON for PHP: Calculate the next or previous run date and determine if a CRON expression is due", + "keywords": [ + "cron", + "schedule" + ], + "time": "2023-08-10T19:36:49+00:00" + }, + { + "name": "ezyang/htmlpurifier", + "version": "v4.16.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/ezyang/htmlpurifier/v4.16.0/ezyang-htmlpurifier-v4.16.0.zip", + "reference": "523407fb06eb9e5f3d59889b3978d5bfe94299c8", + "shasum": "" + }, + "require": { + "php": "~5.6.0 || ~7.0.0 || ~7.1.0 || ~7.2.0 || ~7.3.0 || ~7.4.0 || ~8.0.0 || ~8.1.0 || ~8.2.0" + }, + "require-dev": { + "cerdic/css-tidy": "^1.7 || ^2.0", + "simpletest/simpletest": "dev-master" + }, + "suggest": { + "cerdic/css-tidy": "If you want to use the filter 'Filter.ExtractStyleBlocks'.", + "ext-bcmath": "Used for unit conversion and imagecrash protection", + "ext-iconv": "Converts text to and from non-UTF-8 encodings", + "ext-tidy": "Used for pretty-printing HTML" + }, + "type": "library", + "autoload": { + "files": [ + "library/HTMLPurifier.composer.php" + ], + "psr-0": { + "HTMLPurifier": "library/" + }, + "exclude-from-classmap": [ + "/library/HTMLPurifier/Language/" + ] + }, + "license": [ + "LGPL-2.1-or-later" + ], + "authors": [ + { + "name": "Edward Z. Yang", + "email": "admin@htmlpurifier.org", + "homepage": "http://ezyang.com" + } + ], + "description": "Standards compliant HTML filter written in PHP", + "homepage": "http://htmlpurifier.org/", + "keywords": [ + "html" + ], + "time": "2022-09-18T07:06:19+00:00" + }, + { + "name": "graham-campbell/result-type", + "version": "v1.1.1", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/graham-campbell/result-type/v1.1.1/graham-campbell-result-type-v1.1.1.zip", + "reference": "672eff8cf1d6fe1ef09ca0f89c4b287d6a3eb831", + "shasum": "" + }, + "require": { + "php": "^7.2.5 || ^8.0", + "phpoption/phpoption": "^1.9.1" + }, + "require-dev": { + "phpunit/phpunit": "^8.5.32 || ^9.6.3 || ^10.0.12" + }, + "type": "library", + "autoload": { + "psr-4": { + "GrahamCampbell\\ResultType\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Graham Campbell", + "email": "hello@gjcampbell.co.uk", + "homepage": "https://github.com/GrahamCampbell" + } + ], + "description": "An Implementation Of The Result Type", + "keywords": [ + "Graham Campbell", + "GrahamCampbell", + "Result Type", + "Result-Type", + "result" + ], + "time": "2023-02-25T20:23:15+00:00" + }, + { + "name": "guzzlehttp/command", + "version": "1.3.0", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/guzzlehttp/command/1.3.0/guzzlehttp-command-1.3.0.zip", + "reference": "3372bcfd79d4b357b6871665bf06155515e8d844", + "shasum": "" + }, + "require": { + "guzzlehttp/guzzle": "^7.5.1", + "guzzlehttp/promises": "^1.5.3 || ^2.0", + "guzzlehttp/psr7": "^1.9.1 || ^2.4.5", + "php": "^7.2.5 || ^8.0" + }, + "require-dev": { + "bamarni/composer-bin-plugin": "^1.8.1", + "phpunit/phpunit": "^8.5.19 || ^9.5.8" + }, + "type": "library", + "extra": { + "bamarni-bin": { + "bin-links": true, + "forward-command": false + } + }, + "autoload": { + "psr-4": { + "GuzzleHttp\\Command\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Graham Campbell", + "email": "hello@gjcampbell.co.uk", + "homepage": "https://github.com/GrahamCampbell" + }, + { + "name": "Michael Dowling", + "email": "mtdowling@gmail.com", + "homepage": "https://github.com/mtdowling" + }, + { + "name": "Jeremy Lindblom", + "email": "jeremeamia@gmail.com", + "homepage": "https://github.com/jeremeamia" + }, + { + "name": "Tobias Nyholm", + "email": "tobias.nyholm@gmail.com", + "homepage": "https://github.com/Nyholm" + } + ], + "description": "Provides the foundation for building command-based web service clients", + "time": "2023-05-21T14:15:09+00:00" + }, + { + "name": "guzzlehttp/guzzle", + "version": "7.8.0", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/guzzlehttp/guzzle/7.8.0/guzzlehttp-guzzle-7.8.0.zip", + "reference": "1110f66a6530a40fe7aea0378fe608ee2b2248f9", + "shasum": "" + }, + "require": { + "ext-json": "*", + "guzzlehttp/promises": "^1.5.3 || ^2.0.1", + "guzzlehttp/psr7": "^1.9.1 || ^2.5.1", + "php": "^7.2.5 || ^8.0", + "psr/http-client": "^1.0", + "symfony/deprecation-contracts": "^2.2 || ^3.0" + }, + "provide": { + "psr/http-client-implementation": "1.0" + }, + "require-dev": { + "bamarni/composer-bin-plugin": "^1.8.1", + "ext-curl": "*", + "php-http/client-integration-tests": "dev-master#2c025848417c1135031fdf9c728ee53d0a7ceaee as 3.0.999", + "php-http/message-factory": "^1.1", + "phpunit/phpunit": "^8.5.29 || ^9.5.23", + "psr/log": "^1.1 || ^2.0 || ^3.0" + }, + "suggest": { + "ext-curl": "Required for CURL handler support", + "ext-intl": "Required for Internationalized Domain Name (IDN) support", + "psr/log": "Required for using the Log middleware" + }, + "type": "library", + "extra": { + "bamarni-bin": { + "bin-links": true, + "forward-command": false + } + }, + "autoload": { + "files": [ + "src/functions_include.php" + ], + "psr-4": { + "GuzzleHttp\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Graham Campbell", + "email": "hello@gjcampbell.co.uk", + "homepage": "https://github.com/GrahamCampbell" + }, + { + "name": "Michael Dowling", + "email": "mtdowling@gmail.com", + "homepage": "https://github.com/mtdowling" + }, + { + "name": "Jeremy Lindblom", + "email": "jeremeamia@gmail.com", + "homepage": "https://github.com/jeremeamia" + }, + { + "name": "George Mponos", + "email": "gmponos@gmail.com", + "homepage": "https://github.com/gmponos" + }, + { + "name": "Tobias Nyholm", + "email": "tobias.nyholm@gmail.com", + "homepage": "https://github.com/Nyholm" + }, + { + "name": "Márk Sági-Kazár", + "email": "mark.sagikazar@gmail.com", + "homepage": "https://github.com/sagikazarmark" + }, + { + "name": "Tobias Schultze", + "email": "webmaster@tubo-world.de", + "homepage": "https://github.com/Tobion" + } + ], + "description": "Guzzle is a PHP HTTP client library", + "keywords": [ + "client", + "curl", + "framework", + "http", + "http client", + "psr-18", + "psr-7", + "rest", + "web service" + ], + "time": "2023-08-27T10:20:53+00:00" + }, + { + "name": "guzzlehttp/guzzle-services", + "version": "1.4.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/guzzlehttp/guzzle-services/1.4.0/guzzlehttp-guzzle-services-1.4.0.zip", + "reference": "f4bb1c205152a56741624b88753732e01a60565c", + "shasum": "" + }, + "require": { + "guzzlehttp/command": "^1.3", + "guzzlehttp/guzzle": "^7.7", + "guzzlehttp/psr7": "^1.9.1 || ^2.4.5", + "guzzlehttp/uri-template": "^1.0.1", + "php": "^7.2.5 || ^8.0" + }, + "require-dev": { + "bamarni/composer-bin-plugin": "^1.8.1", + "phpunit/phpunit": "^8.5.19 || ^9.5.8" + }, + "suggest": { + "gimler/guzzle-description-loader": "^0.0.4" + }, + "type": "library", + "extra": { + "bamarni-bin": { + "bin-links": true, + "forward-command": false + } + }, + "autoload": { + "psr-4": { + "GuzzleHttp\\Command\\Guzzle\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Graham Campbell", + "email": "hello@gjcampbell.co.uk", + "homepage": "https://github.com/GrahamCampbell" + }, + { + "name": "Michael Dowling", + "email": "mtdowling@gmail.com", + "homepage": "https://github.com/mtdowling" + }, + { + "name": "Stefano Kowalke", + "email": "blueduck@mail.org", + "homepage": "https://github.com/Konafets" + }, + { + "name": "Tobias Nyholm", + "email": "tobias.nyholm@gmail.com", + "homepage": "https://github.com/Nyholm" + } + ], + "description": "Provides an implementation of the Guzzle Command library that uses Guzzle service descriptions to describe web services, serialize requests, and parse responses into easy to use model structures.", + "time": "2023-05-21T14:21:30+00:00" + }, + { + "name": "guzzlehttp/promises", + "version": "2.0.1", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/guzzlehttp/promises/2.0.1/guzzlehttp-promises-2.0.1.zip", + "reference": "111166291a0f8130081195ac4556a5587d7f1b5d", + "shasum": "" + }, + "require": { + "php": "^7.2.5 || ^8.0" + }, + "require-dev": { + "bamarni/composer-bin-plugin": "^1.8.1", + "phpunit/phpunit": "^8.5.29 || ^9.5.23" + }, + "type": "library", + "extra": { + "bamarni-bin": { + "bin-links": true, + "forward-command": false + } + }, + "autoload": { + "psr-4": { + "GuzzleHttp\\Promise\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Graham Campbell", + "email": "hello@gjcampbell.co.uk", + "homepage": "https://github.com/GrahamCampbell" + }, + { + "name": "Michael Dowling", + "email": "mtdowling@gmail.com", + "homepage": "https://github.com/mtdowling" + }, + { + "name": "Tobias Nyholm", + "email": "tobias.nyholm@gmail.com", + "homepage": "https://github.com/Nyholm" + }, + { + "name": "Tobias Schultze", + "email": "webmaster@tubo-world.de", + "homepage": "https://github.com/Tobion" + } + ], + "description": "Guzzle promises library", + "keywords": [ + "promise" + ], + "time": "2023-08-03T15:11:55+00:00" + }, + { + "name": "guzzlehttp/psr7", + "version": "2.6.1", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/guzzlehttp/psr7/2.6.1/guzzlehttp-psr7-2.6.1.zip", + "reference": "be45764272e8873c72dbe3d2edcfdfcc3bc9f727", + "shasum": "" + }, + "require": { + "php": "^7.2.5 || ^8.0", + "psr/http-factory": "^1.0", + "psr/http-message": "^1.1 || ^2.0", + "ralouphie/getallheaders": "^3.0" + }, + "provide": { + "psr/http-factory-implementation": "1.0", + "psr/http-message-implementation": "1.0" + }, + "require-dev": { + "bamarni/composer-bin-plugin": "^1.8.1", + "http-interop/http-factory-tests": "^0.9", + "phpunit/phpunit": "^8.5.29 || ^9.5.23" + }, + "suggest": { + "laminas/laminas-httphandlerrunner": "Emit PSR-7 responses" + }, + "type": "library", + "extra": { + "bamarni-bin": { + "bin-links": true, + "forward-command": false + } + }, + "autoload": { + "psr-4": { + "GuzzleHttp\\Psr7\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Graham Campbell", + "email": "hello@gjcampbell.co.uk", + "homepage": "https://github.com/GrahamCampbell" + }, + { + "name": "Michael Dowling", + "email": "mtdowling@gmail.com", + "homepage": "https://github.com/mtdowling" + }, + { + "name": "George Mponos", + "email": "gmponos@gmail.com", + "homepage": "https://github.com/gmponos" + }, + { + "name": "Tobias Nyholm", + "email": "tobias.nyholm@gmail.com", + "homepage": "https://github.com/Nyholm" + }, + { + "name": "Márk Sági-Kazár", + "email": "mark.sagikazar@gmail.com", + "homepage": "https://github.com/sagikazarmark" + }, + { + "name": "Tobias Schultze", + "email": "webmaster@tubo-world.de", + "homepage": "https://github.com/Tobion" + }, + { + "name": "Márk Sági-Kazár", + "email": "mark.sagikazar@gmail.com", + "homepage": "https://sagikazarmark.hu" + } + ], + "description": "PSR-7 message implementation that also provides common utility methods", + "keywords": [ + "http", + "message", + "psr-7", + "request", + "response", + "stream", + "uri", + "url" + ], + "time": "2023-08-27T10:13:57+00:00" + }, + { + "name": "guzzlehttp/uri-template", + "version": "v1.0.2", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/guzzlehttp/uri-template/v1.0.2/guzzlehttp-uri-template-v1.0.2.zip", + "reference": "61bf437fc2197f587f6857d3ff903a24f1731b5d", + "shasum": "" + }, + "require": { + "php": "^7.2.5 || ^8.0", + "symfony/polyfill-php80": "^1.17" + }, + "require-dev": { + "bamarni/composer-bin-plugin": "^1.8.1", + "phpunit/phpunit": "^8.5.19 || ^9.5.8", + "uri-template/tests": "1.0.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "GuzzleHttp\\UriTemplate\\": "src" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Graham Campbell", + "email": "hello@gjcampbell.co.uk", + "homepage": "https://github.com/GrahamCampbell" + }, + { + "name": "Michael Dowling", + "email": "mtdowling@gmail.com", + "homepage": "https://github.com/mtdowling" + }, + { + "name": "George Mponos", + "email": "gmponos@gmail.com", + "homepage": "https://github.com/gmponos" + }, + { + "name": "Tobias Nyholm", + "email": "tobias.nyholm@gmail.com", + "homepage": "https://github.com/Nyholm" + } + ], + "description": "A polyfill class for uri_template of PHP", + "keywords": [ + "guzzlehttp", + "uri-template" + ], + "time": "2023-08-27T10:19:19+00:00" + }, + { + "name": "illuminate/collections", + "version": "v10.22.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/illuminate/collections/v10.22.0/illuminate-collections-v10.22.0.zip", + "reference": "f494398dbaaead9e5ff16a18002d11634e8358e6", + "shasum": "" + }, + "require": { + "illuminate/conditionable": "^10.0", + "illuminate/contracts": "^10.0", + "illuminate/macroable": "^10.0", + "php": "^8.1" + }, + "suggest": { + "symfony/var-dumper": "Required to use the dump method (^6.2)." + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "10.x-dev" + } + }, + "autoload": { + "files": [ + "helpers.php" + ], + "psr-4": { + "Illuminate\\Support\\": "" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Taylor Otwell", + "email": "taylor@laravel.com" + } + ], + "description": "The Illuminate Collections package.", + "homepage": "https://laravel.com", + "time": "2023-08-11T14:48:51+00:00" + }, + { + "name": "illuminate/conditionable", + "version": "v10.22.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/illuminate/conditionable/v10.22.0/illuminate-conditionable-v10.22.0.zip", + "reference": "d0958e4741fc9d6f516a552060fd1b829a85e009", + "shasum": "" + }, + "require": { + "php": "^8.0.2" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "10.x-dev" + } + }, + "autoload": { + "psr-4": { + "Illuminate\\Support\\": "" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Taylor Otwell", + "email": "taylor@laravel.com" + } + ], + "description": "The Illuminate Conditionable package.", + "homepage": "https://laravel.com", + "time": "2023-02-03T08:06:17+00:00" + }, + { + "name": "illuminate/contracts", + "version": "v10.22.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/illuminate/contracts/v10.22.0/illuminate-contracts-v10.22.0.zip", + "reference": "eb1a7e72e159136a832f2c0467de5570bdc208ae", + "shasum": "" + }, + "require": { + "php": "^8.1", + "psr/container": "^1.1.1|^2.0.1", + "psr/simple-cache": "^1.0|^2.0|^3.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "10.x-dev" + } + }, + "autoload": { + "psr-4": { + "Illuminate\\Contracts\\": "" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Taylor Otwell", + "email": "taylor@laravel.com" + } + ], + "description": "The Illuminate Contracts package.", + "homepage": "https://laravel.com", + "time": "2023-07-26T21:27:34+00:00" + }, + { + "name": "illuminate/macroable", + "version": "v10.22.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/illuminate/macroable/v10.22.0/illuminate-macroable-v10.22.0.zip", + "reference": "dff667a46ac37b634dcf68909d9d41e94dc97c27", + "shasum": "" + }, + "require": { + "php": "^8.1" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "10.x-dev" + } + }, + "autoload": { + "psr-4": { + "Illuminate\\Support\\": "" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Taylor Otwell", + "email": "taylor@laravel.com" + } + ], + "description": "The Illuminate Macroable package.", + "homepage": "https://laravel.com", + "time": "2023-06-05T12:46:42+00:00" + }, + { + "name": "illuminate/redis", + "version": "v10.22.0", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/illuminate/redis/v10.22.0/illuminate-redis-v10.22.0.zip", + "reference": "896a0f1940147417f8ce88cced703691bc5548b0", + "shasum": "" + }, + "require": { + "illuminate/collections": "^10.0", + "illuminate/contracts": "^10.0", + "illuminate/macroable": "^10.0", + "illuminate/support": "^10.0", + "php": "^8.1" + }, + "suggest": { + "ext-redis": "Required to use the phpredis connector (^4.0|^5.0).", + "predis/predis": "Required to use the predis connector (^2.0.2)." + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "10.x-dev" + } + }, + "autoload": { + "psr-4": { + "Illuminate\\Redis\\": "" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Taylor Otwell", + "email": "taylor@laravel.com" + } + ], + "description": "The Illuminate Redis package.", + "homepage": "https://laravel.com", + "time": "2023-08-02T14:11:43+00:00" + }, + { + "name": "illuminate/support", + "version": "v10.22.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/illuminate/support/v10.22.0/illuminate-support-v10.22.0.zip", + "reference": "7ff0d0d70a7b8275816398a88870e062a01ebb8b", + "shasum": "" + }, + "require": { + "doctrine/inflector": "^2.0", + "ext-ctype": "*", + "ext-filter": "*", + "ext-mbstring": "*", + "illuminate/collections": "^10.0", + "illuminate/conditionable": "^10.0", + "illuminate/contracts": "^10.0", + "illuminate/macroable": "^10.0", + "nesbot/carbon": "^2.67", + "php": "^8.1", + "voku/portable-ascii": "^2.0" + }, + "conflict": { + "tightenco/collect": "<5.5.33" + }, + "suggest": { + "illuminate/filesystem": "Required to use the composer class (^10.0).", + "league/commonmark": "Required to use Str::markdown() and Stringable::markdown() (^2.0.2).", + "ramsey/uuid": "Required to use Str::uuid() (^4.7).", + "symfony/process": "Required to use the composer class (^6.2).", + "symfony/uid": "Required to use Str::ulid() (^6.2).", + "symfony/var-dumper": "Required to use the dd function (^6.2).", + "vlucas/phpdotenv": "Required to use the Env class and env helper (^5.4.1)." + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "10.x-dev" + } + }, + "autoload": { + "files": [ + "helpers.php" + ], + "psr-4": { + "Illuminate\\Support\\": "" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Taylor Otwell", + "email": "taylor@laravel.com" + } + ], + "description": "The Illuminate Support package.", + "homepage": "https://laravel.com", + "time": "2023-09-04T15:58:19+00:00" + }, + { + "name": "laravel/serializable-closure", + "version": "v1.3.1", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/laravel/serializable-closure/v1.3.1/laravel-serializable-closure-v1.3.1.zip", + "reference": "e5a3057a5591e1cfe8183034b0203921abe2c902", + "shasum": "" + }, + "require": { + "php": "^7.3|^8.0" + }, + "require-dev": { + "nesbot/carbon": "^2.61", + "pestphp/pest": "^1.21.3", + "phpstan/phpstan": "^1.8.2", + "symfony/var-dumper": "^5.4.11" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.x-dev" + } + }, + "autoload": { + "psr-4": { + "Laravel\\SerializableClosure\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Taylor Otwell", + "email": "taylor@laravel.com" + }, + { + "name": "Nuno Maduro", + "email": "nuno@laravel.com" + } + ], + "description": "Laravel Serializable Closure provides an easy and secure way to serialize closures in PHP.", + "keywords": [ + "closure", + "laravel", + "serializable" + ], + "time": "2023-07-14T13:56:28+00:00" + }, + { + "name": "maennchen/zipstream-php", + "version": "3.1.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/maennchen/zipstream-php/3.1.0/maennchen-zipstream-php-3.1.0.zip", + "reference": "b8174494eda667f7d13876b4a7bfef0f62a7c0d1", + "shasum": "" + }, + "require": { + "ext-mbstring": "*", + "ext-zlib": "*", + "php-64bit": "^8.1" + }, + "require-dev": { + "ext-zip": "*", + "friendsofphp/php-cs-fixer": "^3.16", + "guzzlehttp/guzzle": "^7.5", + "mikey179/vfsstream": "^1.6", + "php-coveralls/php-coveralls": "^2.5", + "phpunit/phpunit": "^10.0", + "vimeo/psalm": "^5.0" + }, + "suggest": { + "guzzlehttp/psr7": "^2.4", + "psr/http-message": "^2.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "ZipStream\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Paul Duncan", + "email": "pabs@pablotron.org" + }, + { + "name": "Jonatan Männchen", + "email": "jonatan@maennchen.ch" + }, + { + "name": "Jesse Donat", + "email": "donatj@gmail.com" + }, + { + "name": "András Kolesár", + "email": "kolesar@kolesar.hu" + } + ], + "description": "ZipStream is a library for dynamically streaming dynamic zip files from PHP without writing to the disk at all on the server.", + "keywords": [ + "stream", + "zip" + ], + "time": "2023-06-21T14:59:35+00:00" + }, + { + "name": "markbaker/complex", + "version": "3.0.2", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/markbaker/complex/3.0.2/markbaker-complex-3.0.2.zip", + "reference": "95c56caa1cf5c766ad6d65b6344b807c1e8405b9", + "shasum": "" + }, + "require": { + "php": "^7.2 || ^8.0" + }, + "require-dev": { + "dealerdirect/phpcodesniffer-composer-installer": "dev-master", + "phpcompatibility/php-compatibility": "^9.3", + "phpunit/phpunit": "^7.0 || ^8.0 || ^9.0", + "squizlabs/php_codesniffer": "^3.7" + }, + "type": "library", + "autoload": { + "psr-4": { + "Complex\\": "classes/src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Mark Baker", + "email": "mark@lange.demon.co.uk" + } + ], + "description": "PHP Class for working with complex numbers", + "homepage": "https://github.com/MarkBaker/PHPComplex", + "keywords": [ + "complex", + "mathematics" + ], + "time": "2022-12-06T16:21:08+00:00" + }, + { + "name": "markbaker/matrix", + "version": "3.0.1", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/markbaker/matrix/3.0.1/markbaker-matrix-3.0.1.zip", + "reference": "728434227fe21be27ff6d86621a1b13107a2562c", + "shasum": "" + }, + "require": { + "php": "^7.1 || ^8.0" + }, + "require-dev": { + "dealerdirect/phpcodesniffer-composer-installer": "dev-master", + "phpcompatibility/php-compatibility": "^9.3", + "phpdocumentor/phpdocumentor": "2.*", + "phploc/phploc": "^4.0", + "phpmd/phpmd": "2.*", + "phpunit/phpunit": "^7.0 || ^8.0 || ^9.0", + "sebastian/phpcpd": "^4.0", + "squizlabs/php_codesniffer": "^3.7" + }, + "type": "library", + "autoload": { + "psr-4": { + "Matrix\\": "classes/src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Mark Baker", + "email": "mark@demon-angel.eu" + } + ], + "description": "PHP Class for working with matrices", + "homepage": "https://github.com/MarkBaker/PHPMatrix", + "keywords": [ + "mathematics", + "matrix", + "vector" + ], + "time": "2022-12-02T22:17:43+00:00" + }, + { + "name": "max/var-dumper", + "version": "0.1.1", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/max/var-dumper/0.1.1/max-var-dumper-0.1.1.zip", + "reference": "603bdeb2135b24c9eb91e559df1f9fc6a95b811d", + "shasum": "" + }, + "require": { + "php": ">=7.4", + "symfony/var-dumper": "*" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "0.1.x-dev" + } + }, + "autoload": { + "files": [ + "src/helpers.php" + ], + "psr-4": { + "Max\\VarDumper\\": "src/" + } + }, + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "chengyao", + "email": "987861463@qq.com" + } + ], + "homepage": "https://github.com/marxphp/var-dumper", + "time": "2023-07-07T12:54:36+00:00" + }, + { + "name": "monolog/monolog", + "version": "2.9.1", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/monolog/monolog/2.9.1/monolog-monolog-2.9.1.zip", + "reference": "f259e2b15fb95494c83f52d3caad003bbf5ffaa1", + "shasum": "" + }, + "require": { + "php": ">=7.2", + "psr/log": "^1.0.1 || ^2.0 || ^3.0" + }, + "provide": { + "psr/log-implementation": "1.0.0 || 2.0.0 || 3.0.0" + }, + "require-dev": { + "aws/aws-sdk-php": "^2.4.9 || ^3.0", + "doctrine/couchdb": "~1.0@dev", + "elasticsearch/elasticsearch": "^7 || ^8", + "ext-json": "*", + "graylog2/gelf-php": "^1.4.2 || ^2@dev", + "guzzlehttp/guzzle": "^7.4", + "guzzlehttp/psr7": "^2.2", + "mongodb/mongodb": "^1.8", + "php-amqplib/php-amqplib": "~2.4 || ^3", + "phpspec/prophecy": "^1.15", + "phpstan/phpstan": "^0.12.91", + "phpunit/phpunit": "^8.5.14", + "predis/predis": "^1.1 || ^2.0", + "rollbar/rollbar": "^1.3 || ^2 || ^3", + "ruflin/elastica": "^7", + "swiftmailer/swiftmailer": "^5.3|^6.0", + "symfony/mailer": "^5.4 || ^6", + "symfony/mime": "^5.4 || ^6" + }, + "suggest": { + "aws/aws-sdk-php": "Allow sending log messages to AWS services like DynamoDB", + "doctrine/couchdb": "Allow sending log messages to a CouchDB server", + "elasticsearch/elasticsearch": "Allow sending log messages to an Elasticsearch server via official client", + "ext-amqp": "Allow sending log messages to an AMQP server (1.0+ required)", + "ext-curl": "Required to send log messages using the IFTTTHandler, the LogglyHandler, the SendGridHandler, the SlackWebhookHandler or the TelegramBotHandler", + "ext-mbstring": "Allow to work properly with unicode symbols", + "ext-mongodb": "Allow sending log messages to a MongoDB server (via driver)", + "ext-openssl": "Required to send log messages using SSL", + "ext-sockets": "Allow sending log messages to a Syslog server (via UDP driver)", + "graylog2/gelf-php": "Allow sending log messages to a GrayLog2 server", + "mongodb/mongodb": "Allow sending log messages to a MongoDB server (via library)", + "php-amqplib/php-amqplib": "Allow sending log messages to an AMQP server using php-amqplib", + "rollbar/rollbar": "Allow sending log messages to Rollbar", + "ruflin/elastica": "Allow sending log messages to an Elastic Search server" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "2.x-dev" + } + }, + "autoload": { + "psr-4": { + "Monolog\\": "src/Monolog" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Jordi Boggiano", + "email": "j.boggiano@seld.be", + "homepage": "https://seld.be" + } + ], + "description": "Sends your logs to files, sockets, inboxes, databases and various web services", + "homepage": "https://github.com/Seldaek/monolog", + "keywords": [ + "log", + "logging", + "psr-3" + ], + "time": "2023-02-06T13:44:46+00:00" + }, + { + "name": "nesbot/carbon", + "version": "2.70.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/nesbot/carbon/2.70.0/nesbot-carbon-2.70.0.zip", + "reference": "d3298b38ea8612e5f77d38d1a99438e42f70341d", + "shasum": "" + }, + "require": { + "ext-json": "*", + "php": "^7.1.8 || ^8.0", + "psr/clock": "^1.0", + "symfony/polyfill-mbstring": "^1.0", + "symfony/polyfill-php80": "^1.16", + "symfony/translation": "^3.4 || ^4.0 || ^5.0 || ^6.0" + }, + "provide": { + "psr/clock-implementation": "1.0" + }, + "require-dev": { + "doctrine/dbal": "^2.0 || ^3.1.4", + "doctrine/orm": "^2.7", + "friendsofphp/php-cs-fixer": "^3.0", + "kylekatarnls/multi-tester": "^2.0", + "ondrejmirtes/better-reflection": "*", + "phpmd/phpmd": "^2.9", + "phpstan/extension-installer": "^1.0", + "phpstan/phpstan": "^0.12.99 || ^1.7.14", + "phpunit/php-file-iterator": "^2.0.5 || ^3.0.6", + "phpunit/phpunit": "^7.5.20 || ^8.5.26 || ^9.5.20", + "squizlabs/php_codesniffer": "^3.4" + }, + "bin": [ + "bin/carbon" + ], + "type": "library", + "extra": { + "branch-alias": { + "dev-3.x": "3.x-dev", + "dev-master": "2.x-dev" + }, + "laravel": { + "providers": [ + "Carbon\\Laravel\\ServiceProvider" + ] + }, + "phpstan": { + "includes": [ + "extension.neon" + ] + } + }, + "autoload": { + "psr-4": { + "Carbon\\": "src/Carbon/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Brian Nesbitt", + "email": "brian@nesbot.com", + "homepage": "https://markido.com" + }, + { + "name": "kylekatarnls", + "homepage": "https://github.com/kylekatarnls" + } + ], + "description": "An API extension for DateTime that supports 281 different languages.", + "homepage": "https://carbon.nesbot.com", + "keywords": [ + "date", + "datetime", + "time" + ], + "time": "2023-09-07T16:43:50+00:00" + }, + { + "name": "nikic/fast-route", + "version": "v1.3.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/nikic/fast-route/v1.3.0/nikic-fast-route-v1.3.0.zip", + "reference": "181d480e08d9476e61381e04a71b34dc0432e812", + "shasum": "" + }, + "require": { + "php": ">=5.4.0" + }, + "require-dev": { + "phpunit/phpunit": "^4.8.35|~5.7" + }, + "type": "library", + "autoload": { + "files": [ + "src/functions.php" + ], + "psr-4": { + "FastRoute\\": "src/" + } + }, + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Nikita Popov", + "email": "nikic@php.net" + } + ], + "description": "Fast request router for PHP", + "keywords": [ + "router", + "routing" + ], + "time": "2018-02-13T20:26:39+00:00" + }, + { + "name": "php-di/invoker", + "version": "2.3.4", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/php-di/invoker/2.3.4/php-di-invoker-2.3.4.zip", + "reference": "33234b32dafa8eb69202f950a1fc92055ed76a86", + "shasum": "" + }, + "require": { + "php": ">=7.3", + "psr/container": "^1.0|^2.0" + }, + "require-dev": { + "athletic/athletic": "~0.1.8", + "mnapoli/hard-mode": "~0.3.0", + "phpunit/phpunit": "^9.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Invoker\\": "src/" + } + }, + "license": [ + "MIT" + ], + "description": "Generic and extensible callable invoker", + "homepage": "https://github.com/PHP-DI/Invoker", + "keywords": [ + "callable", + "dependency", + "dependency-injection", + "injection", + "invoke", + "invoker" + ], + "time": "2023-09-08T09:24:21+00:00" + }, + { + "name": "php-di/php-di", + "version": "6.4.0", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/php-di/php-di/6.4.0/php-di-php-di-6.4.0.zip", + "reference": "ae0f1b3b03d8b29dff81747063cbfd6276246cc4", + "shasum": "" + }, + "require": { + "laravel/serializable-closure": "^1.0", + "php": ">=7.4.0", + "php-di/invoker": "^2.0", + "php-di/phpdoc-reader": "^2.0.1", + "psr/container": "^1.0" + }, + "provide": { + "psr/container-implementation": "^1.0" + }, + "require-dev": { + "doctrine/annotations": "~1.10", + "friendsofphp/php-cs-fixer": "^2.4", + "mnapoli/phpunit-easymock": "^1.2", + "ocramius/proxy-manager": "^2.11.2", + "phpstan/phpstan": "^0.12", + "phpunit/phpunit": "^9.5" + }, + "suggest": { + "doctrine/annotations": "Install it if you want to use annotations (version ~1.2)", + "ocramius/proxy-manager": "Install it if you want to use lazy injection (version ~2.0)" + }, + "type": "library", + "autoload": { + "files": [ + "src/functions.php" + ], + "psr-4": { + "DI\\": "src/" + } + }, + "license": [ + "MIT" + ], + "description": "The dependency injection container for humans", + "homepage": "https://php-di.org/", + "keywords": [ + "PSR-11", + "container", + "container-interop", + "dependency injection", + "di", + "ioc", + "psr11" + ], + "time": "2022-04-09T16:46:38+00:00" + }, + { + "name": "php-di/phpdoc-reader", + "version": "2.2.1", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/php-di/phpdoc-reader/2.2.1/php-di-phpdoc-reader-2.2.1.zip", + "reference": "66daff34cbd2627740ffec9469ffbac9f8c8185c", + "shasum": "" + }, + "require": { + "php": ">=7.2.0" + }, + "require-dev": { + "mnapoli/hard-mode": "~0.3.0", + "phpunit/phpunit": "^8.5|^9.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "PhpDocReader\\": "src/PhpDocReader" + } + }, + "license": [ + "MIT" + ], + "description": "PhpDocReader parses @var and @param values in PHP docblocks (supports namespaced class names with the same resolution rules as PHP)", + "keywords": [ + "phpdoc", + "reflection" + ], + "time": "2020-10-12T12:39:22+00:00" + }, + { + "name": "phpoffice/phpspreadsheet", + "version": "1.29.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/phpoffice/phpspreadsheet/1.29.0/phpoffice-phpspreadsheet-1.29.0.zip", + "reference": "fde2ccf55eaef7e86021ff1acce26479160a0fa0", + "shasum": "" + }, + "require": { + "ext-ctype": "*", + "ext-dom": "*", + "ext-fileinfo": "*", + "ext-gd": "*", + "ext-iconv": "*", + "ext-libxml": "*", + "ext-mbstring": "*", + "ext-simplexml": "*", + "ext-xml": "*", + "ext-xmlreader": "*", + "ext-xmlwriter": "*", + "ext-zip": "*", + "ext-zlib": "*", + "ezyang/htmlpurifier": "^4.15", + "maennchen/zipstream-php": "^2.1 || ^3.0", + "markbaker/complex": "^3.0", + "markbaker/matrix": "^3.0", + "php": "^7.4 || ^8.0", + "psr/http-client": "^1.0", + "psr/http-factory": "^1.0", + "psr/simple-cache": "^1.0 || ^2.0 || ^3.0" + }, + "require-dev": { + "dealerdirect/phpcodesniffer-composer-installer": "dev-main", + "dompdf/dompdf": "^1.0 || ^2.0", + "friendsofphp/php-cs-fixer": "^3.2", + "mitoteam/jpgraph": "^10.3", + "mpdf/mpdf": "^8.1.1", + "phpcompatibility/php-compatibility": "^9.3", + "phpstan/phpstan": "^1.1", + "phpstan/phpstan-phpunit": "^1.0", + "phpunit/phpunit": "^8.5 || ^9.0 || ^10.0", + "squizlabs/php_codesniffer": "^3.7", + "tecnickcom/tcpdf": "^6.5" + }, + "suggest": { + "dompdf/dompdf": "Option for rendering PDF with PDF Writer", + "ext-intl": "PHP Internationalization Functions", + "mitoteam/jpgraph": "Option for rendering charts, or including charts with PDF or HTML Writers", + "mpdf/mpdf": "Option for rendering PDF with PDF Writer", + "tecnickcom/tcpdf": "Option for rendering PDF with PDF Writer" + }, + "type": "library", + "autoload": { + "psr-4": { + "PhpOffice\\PhpSpreadsheet\\": "src/PhpSpreadsheet" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Maarten Balliauw", + "homepage": "https://blog.maartenballiauw.be" + }, + { + "name": "Mark Baker", + "homepage": "https://markbakeruk.net" + }, + { + "name": "Franck Lefevre", + "homepage": "https://rootslabs.net" + }, + { + "name": "Erik Tilt" + }, + { + "name": "Adrien Crivelli" + } + ], + "description": "PHPSpreadsheet - Read, Create and Write Spreadsheet documents in PHP - Spreadsheet engine", + "homepage": "https://github.com/PHPOffice/PhpSpreadsheet", + "keywords": [ + "OpenXML", + "excel", + "gnumeric", + "ods", + "php", + "spreadsheet", + "xls", + "xlsx" + ], + "time": "2023-06-14T22:48:31+00:00" + }, + { + "name": "phpoption/phpoption", + "version": "1.9.1", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/phpoption/phpoption/1.9.1/phpoption-phpoption-1.9.1.zip", + "reference": "dd3a383e599f49777d8b628dadbb90cae435b87e", + "shasum": "" + }, + "require": { + "php": "^7.2.5 || ^8.0" + }, + "require-dev": { + "bamarni/composer-bin-plugin": "^1.8.2", + "phpunit/phpunit": "^8.5.32 || ^9.6.3 || ^10.0.12" + }, + "type": "library", + "extra": { + "bamarni-bin": { + "bin-links": true, + "forward-command": true + }, + "branch-alias": { + "dev-master": "1.9-dev" + } + }, + "autoload": { + "psr-4": { + "PhpOption\\": "src/PhpOption/" + } + }, + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "Johannes M. Schmitt", + "email": "schmittjoh@gmail.com", + "homepage": "https://github.com/schmittjoh" + }, + { + "name": "Graham Campbell", + "email": "hello@gjcampbell.co.uk", + "homepage": "https://github.com/GrahamCampbell" + } + ], + "description": "Option Type for PHP", + "keywords": [ + "language", + "option", + "php", + "type" + ], + "time": "2023-02-25T19:38:58+00:00" + }, + { + "name": "psr/cache", + "version": "1.0.1", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/psr/cache/1.0.1/psr-cache-1.0.1.zip", + "reference": "d11b50ad223250cf17b86e38383413f5a6764bf8", + "shasum": "" + }, + "require": { + "php": ">=5.3.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Cache\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "http://www.php-fig.org/" + } + ], + "description": "Common interface for caching libraries", + "keywords": [ + "cache", + "psr", + "psr-6" + ], + "time": "2016-08-06T20:24:11+00:00" + }, + { + "name": "psr/clock", + "version": "1.0.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/psr/clock/1.0.0/psr-clock-1.0.0.zip", + "reference": "e41a24703d4560fd0acb709162f73b8adfc3aa0d", + "shasum": "" + }, + "require": { + "php": "^7.0 || ^8.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Psr\\Clock\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "Common interface for reading the clock.", + "homepage": "https://github.com/php-fig/clock", + "keywords": [ + "clock", + "now", + "psr", + "psr-20", + "time" + ], + "time": "2022-11-25T14:36:26+00:00" + }, + { + "name": "psr/container", + "version": "1.1.2", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/psr/container/1.1.2/psr-container-1.1.2.zip", + "reference": "513e0666f7216c7459170d56df27dfcefe1689ea", + "shasum": "" + }, + "require": { + "php": ">=7.4.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Psr\\Container\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "Common Container Interface (PHP FIG PSR-11)", + "homepage": "https://github.com/php-fig/container", + "keywords": [ + "PSR-11", + "container", + "container-interface", + "container-interop", + "psr" + ], + "time": "2021-11-05T16:50:12+00:00" + }, + { + "name": "psr/http-client", + "version": "1.0.2", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/psr/http-client/1.0.2/psr-http-client-1.0.2.zip", + "reference": "0955afe48220520692d2d09f7ab7e0f93ffd6a31", + "shasum": "" + }, + "require": { + "php": "^7.0 || ^8.0", + "psr/http-message": "^1.0 || ^2.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Http\\Client\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "Common interface for HTTP clients", + "homepage": "https://github.com/php-fig/http-client", + "keywords": [ + "http", + "http-client", + "psr", + "psr-18" + ], + "time": "2023-04-10T20:12:12+00:00" + }, + { + "name": "psr/http-factory", + "version": "1.0.2", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/psr/http-factory/1.0.2/psr-http-factory-1.0.2.zip", + "reference": "e616d01114759c4c489f93b099585439f795fe35", + "shasum": "" + }, + "require": { + "php": ">=7.0.0", + "psr/http-message": "^1.0 || ^2.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Http\\Message\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "Common interfaces for PSR-7 HTTP message factories", + "keywords": [ + "factory", + "http", + "message", + "psr", + "psr-17", + "psr-7", + "request", + "response" + ], + "time": "2023-04-10T20:10:41+00:00" + }, + { + "name": "psr/http-message", + "version": "2.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/psr/http-message/2.0/psr-http-message-2.0.zip", + "reference": "402d35bcb92c70c026d1a6a9883f06b2ead23d71", + "shasum": "" + }, + "require": { + "php": "^7.2 || ^8.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "2.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Http\\Message\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "Common interface for HTTP messages", + "homepage": "https://github.com/php-fig/http-message", + "keywords": [ + "http", + "http-message", + "psr", + "psr-7", + "request", + "response" + ], + "time": "2023-04-04T09:54:51+00:00" + }, + { + "name": "psr/log", + "version": "3.0.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/psr/log/3.0.0/psr-log-3.0.0.zip", + "reference": "fe5ea303b0887d5caefd3d431c3e61ad47037001", + "shasum": "" + }, + "require": { + "php": ">=8.0.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "3.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Log\\": "src" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "https://www.php-fig.org/" + } + ], + "description": "Common interface for logging libraries", + "homepage": "https://github.com/php-fig/log", + "keywords": [ + "log", + "psr", + "psr-3" + ], + "time": "2021-07-14T16:46:02+00:00" + }, + { + "name": "psr/simple-cache", + "version": "1.0.1", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/psr/simple-cache/1.0.1/psr-simple-cache-1.0.1.zip", + "reference": "408d5eafb83c57f6365a3ca330ff23aa4a5fa39b", + "shasum": "" + }, + "require": { + "php": ">=5.3.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\SimpleCache\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "http://www.php-fig.org/" + } + ], + "description": "Common interfaces for simple caching", + "keywords": [ + "cache", + "caching", + "psr", + "psr-16", + "simple-cache" + ], + "time": "2017-10-23T01:57:42+00:00" + }, + { + "name": "qcloud/cos-sdk-v5", + "version": "v2.6.6", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/qcloud/cos-sdk-v5/v2.6.6/qcloud-cos-sdk-v5-v2.6.6.zip", + "reference": "9d82ccb550fe2dca1adfb53835791d314023a9a8", + "shasum": "" + }, + "require": { + "ext-curl": "*", + "ext-json": "*", + "ext-mbstring": "*", + "ext-simplexml": "*", + "guzzlehttp/guzzle": "^6.2.1 || ^7.0", + "guzzlehttp/guzzle-services": "^1.1", + "guzzlehttp/psr7": "^1.3.1 || ^2.0", + "php": ">=5.6" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "2.4-dev" + } + }, + "autoload": { + "files": [ + "src/Common.php" + ], + "psr-4": { + "Qcloud\\Cos\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "yaozongyou", + "email": "yaozongyou@vip.qq.com" + }, + { + "name": "lewzylu", + "email": "327874225@qq.com" + }, + { + "name": "tuunalai", + "email": "550566181@qq.com" + } + ], + "description": "PHP SDK for QCloud COS", + "keywords": [ + "cos", + "php", + "qcloud" + ], + "time": "2023-08-23T08:14:27+00:00" + }, + { + "name": "qiniu/php-sdk", + "version": "v7.4.0", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/qiniu/php-sdk/v7.4.0/qiniu-php-sdk-v7.4.0.zip", + "reference": "1c6bc89166e524a40ee42bf516fb99ffc6401c82", + "shasum": "" + }, + "require": { + "php": ">=5.3.3" + }, + "require-dev": { + "phpunit/phpunit": "~4.0", + "squizlabs/php_codesniffer": "~3.6" + }, + "type": "library", + "autoload": { + "files": [ + "src/Qiniu/functions.php" + ], + "psr-4": { + "Qiniu\\": "src/Qiniu" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Qiniu", + "email": "sdk@qiniu.com", + "homepage": "http://www.qiniu.com" + } + ], + "description": "Qiniu Resource (Cloud) Storage SDK for PHP", + "homepage": "http://developer.qiniu.com/", + "keywords": [ + "cloud", + "qiniu", + "sdk", + "storage" + ], + "time": "2021-07-19T07:41:36+00:00" + }, + { + "name": "ralouphie/getallheaders", + "version": "3.0.3", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/ralouphie/getallheaders/3.0.3/ralouphie-getallheaders-3.0.3.zip", + "reference": "120b605dfeb996808c31b6477290a714d356e822", + "shasum": "" + }, + "require": { + "php": ">=5.6" + }, + "require-dev": { + "php-coveralls/php-coveralls": "^2.1", + "phpunit/phpunit": "^5 || ^6.5" + }, + "type": "library", + "autoload": { + "files": [ + "src/getallheaders.php" + ] + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Ralph Khattar", + "email": "ralph.khattar@gmail.com" + } + ], + "description": "A polyfill for getallheaders.", + "time": "2019-03-08T08:55:37+00:00" + }, + { + "name": "symfony/cache", + "version": "v5.4.28", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/symfony/cache/v5.4.28/symfony-cache-v5.4.28.zip", + "reference": "62b7ae3bccc5b474a30fadc7ef6bbc362007d3f9", + "shasum": "" + }, + "require": { + "php": ">=7.2.5", + "psr/cache": "^1.0|^2.0", + "psr/log": "^1.1|^2|^3", + "symfony/cache-contracts": "^1.1.7|^2", + "symfony/deprecation-contracts": "^2.1|^3", + "symfony/polyfill-php73": "^1.9", + "symfony/polyfill-php80": "^1.16", + "symfony/service-contracts": "^1.1|^2|^3", + "symfony/var-exporter": "^4.4|^5.0|^6.0" + }, + "conflict": { + "doctrine/dbal": "<2.13.1", + "symfony/dependency-injection": "<4.4", + "symfony/http-kernel": "<4.4", + "symfony/var-dumper": "<4.4" + }, + "provide": { + "psr/cache-implementation": "1.0|2.0", + "psr/simple-cache-implementation": "1.0|2.0", + "symfony/cache-implementation": "1.0|2.0" + }, + "require-dev": { + "cache/integration-tests": "dev-master", + "doctrine/cache": "^1.6|^2.0", + "doctrine/dbal": "^2.13.1|^3.0", + "predis/predis": "^1.1", + "psr/simple-cache": "^1.0|^2.0", + "symfony/config": "^4.4|^5.0|^6.0", + "symfony/dependency-injection": "^4.4|^5.0|^6.0", + "symfony/filesystem": "^4.4|^5.0|^6.0", + "symfony/http-kernel": "^4.4|^5.0|^6.0", + "symfony/messenger": "^4.4|^5.0|^6.0", + "symfony/var-dumper": "^4.4|^5.0|^6.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Symfony\\Component\\Cache\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Provides extended PSR-6, PSR-16 (and tags) implementations", + "homepage": "https://symfony.com", + "keywords": [ + "caching", + "psr6" + ], + "time": "2023-08-05T08:32:42+00:00" + }, + { + "name": "symfony/cache-contracts", + "version": "v2.5.2", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/symfony/cache-contracts/v2.5.2/symfony-cache-contracts-v2.5.2.zip", + "reference": "64be4a7acb83b6f2bf6de9a02cee6dad41277ebc", + "shasum": "" + }, + "require": { + "php": ">=7.2.5", + "psr/cache": "^1.0|^2.0|^3.0" + }, + "suggest": { + "symfony/cache-implementation": "" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "2.5-dev" + }, + "thanks": { + "name": "symfony/contracts", + "url": "https://github.com/symfony/contracts" + } + }, + "autoload": { + "psr-4": { + "Symfony\\Contracts\\Cache\\": "" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Generic abstractions related to caching", + "homepage": "https://symfony.com", + "keywords": [ + "abstractions", + "contracts", + "decoupling", + "interfaces", + "interoperability", + "standards" + ], + "time": "2022-01-02T09:53:40+00:00" + }, + { + "name": "symfony/console", + "version": "v6.3.4", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/symfony/console/v6.3.4/symfony-console-v6.3.4.zip", + "reference": "eca495f2ee845130855ddf1cf18460c38966c8b6", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "symfony/deprecation-contracts": "^2.5|^3", + "symfony/polyfill-mbstring": "~1.0", + "symfony/service-contracts": "^2.5|^3", + "symfony/string": "^5.4|^6.0" + }, + "conflict": { + "symfony/dependency-injection": "<5.4", + "symfony/dotenv": "<5.4", + "symfony/event-dispatcher": "<5.4", + "symfony/lock": "<5.4", + "symfony/process": "<5.4" + }, + "provide": { + "psr/log-implementation": "1.0|2.0|3.0" + }, + "require-dev": { + "psr/log": "^1|^2|^3", + "symfony/config": "^5.4|^6.0", + "symfony/dependency-injection": "^5.4|^6.0", + "symfony/event-dispatcher": "^5.4|^6.0", + "symfony/lock": "^5.4|^6.0", + "symfony/process": "^5.4|^6.0", + "symfony/var-dumper": "^5.4|^6.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Symfony\\Component\\Console\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Fabien Potencier", + "email": "fabien@symfony.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Eases the creation of beautiful and testable command line interfaces", + "homepage": "https://symfony.com", + "keywords": [ + "cli", + "command-line", + "console", + "terminal" + ], + "time": "2023-08-16T10:10:12+00:00" + }, + { + "name": "symfony/deprecation-contracts", + "version": "v3.3.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/symfony/deprecation-contracts/v3.3.0/symfony-deprecation-contracts-v3.3.0.zip", + "reference": "7c3aff79d10325257a001fcf92d991f24fc967cf", + "shasum": "" + }, + "require": { + "php": ">=8.1" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "3.4-dev" + }, + "thanks": { + "name": "symfony/contracts", + "url": "https://github.com/symfony/contracts" + } + }, + "autoload": { + "files": [ + "function.php" + ] + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "A generic function and convention to trigger deprecation notices", + "homepage": "https://symfony.com", + "time": "2023-05-23T14:45:45+00:00" + }, + { + "name": "symfony/polyfill-ctype", + "version": "v1.28.0", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/symfony/polyfill-ctype/v1.28.0/symfony-polyfill-ctype-v1.28.0.zip", + "reference": "ea208ce43cbb04af6867b4fdddb1bdbf84cc28cb", + "shasum": "" + }, + "require": { + "php": ">=7.1" + }, + "provide": { + "ext-ctype": "*" + }, + "suggest": { + "ext-ctype": "For best performance" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.28-dev" + }, + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Ctype\\": "" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Gert de Pagter", + "email": "BackEndTea@gmail.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Symfony polyfill for ctype functions", + "homepage": "https://symfony.com", + "keywords": [ + "compatibility", + "ctype", + "polyfill", + "portable" + ], + "time": "2023-01-26T09:26:14+00:00" + }, + { + "name": "symfony/polyfill-intl-grapheme", + "version": "v1.28.0", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/symfony/polyfill-intl-grapheme/v1.28.0/symfony-polyfill-intl-grapheme-v1.28.0.zip", + "reference": "875e90aeea2777b6f135677f618529449334a612", + "shasum": "" + }, + "require": { + "php": ">=7.1" + }, + "suggest": { + "ext-intl": "For best performance" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.28-dev" + }, + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Intl\\Grapheme\\": "" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Symfony polyfill for intl's grapheme_* functions", + "homepage": "https://symfony.com", + "keywords": [ + "compatibility", + "grapheme", + "intl", + "polyfill", + "portable", + "shim" + ], + "time": "2023-01-26T09:26:14+00:00" + }, + { + "name": "symfony/polyfill-intl-normalizer", + "version": "v1.28.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/symfony/polyfill-intl-normalizer/v1.28.0/symfony-polyfill-intl-normalizer-v1.28.0.zip", + "reference": "8c4ad05dd0120b6a53c1ca374dca2ad0a1c4ed92", + "shasum": "" + }, + "require": { + "php": ">=7.1" + }, + "suggest": { + "ext-intl": "For best performance" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.28-dev" + }, + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Intl\\Normalizer\\": "" + }, + "classmap": [ + "Resources/stubs" + ] + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Symfony polyfill for intl's Normalizer class and related functions", + "homepage": "https://symfony.com", + "keywords": [ + "compatibility", + "intl", + "normalizer", + "polyfill", + "portable", + "shim" + ], + "time": "2023-01-26T09:26:14+00:00" + }, + { + "name": "symfony/polyfill-mbstring", + "version": "v1.28.0", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/symfony/polyfill-mbstring/v1.28.0/symfony-polyfill-mbstring-v1.28.0.zip", + "reference": "42292d99c55abe617799667f454222c54c60e229", + "shasum": "" + }, + "require": { + "php": ">=7.1" + }, + "provide": { + "ext-mbstring": "*" + }, + "suggest": { + "ext-mbstring": "For best performance" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.28-dev" + }, + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Mbstring\\": "" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Symfony polyfill for the Mbstring extension", + "homepage": "https://symfony.com", + "keywords": [ + "compatibility", + "mbstring", + "polyfill", + "portable", + "shim" + ], + "time": "2023-07-28T09:04:16+00:00" + }, + { + "name": "symfony/polyfill-php73", + "version": "v1.28.0", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/symfony/polyfill-php73/v1.28.0/symfony-polyfill-php73-v1.28.0.zip", + "reference": "fe2f306d1d9d346a7fee353d0d5012e401e984b5", + "shasum": "" + }, + "require": { + "php": ">=7.1" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.28-dev" + }, + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Php73\\": "" + }, + "classmap": [ + "Resources/stubs" + ] + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Symfony polyfill backporting some PHP 7.3+ features to lower PHP versions", + "homepage": "https://symfony.com", + "keywords": [ + "compatibility", + "polyfill", + "portable", + "shim" + ], + "time": "2023-01-26T09:26:14+00:00" + }, + { + "name": "symfony/polyfill-php80", + "version": "v1.28.0", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/symfony/polyfill-php80/v1.28.0/symfony-polyfill-php80-v1.28.0.zip", + "reference": "6caa57379c4aec19c0a12a38b59b26487dcfe4b5", + "shasum": "" + }, + "require": { + "php": ">=7.1" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.28-dev" + }, + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Php80\\": "" + }, + "classmap": [ + "Resources/stubs" + ] + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Ion Bazan", + "email": "ion.bazan@gmail.com" + }, + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Symfony polyfill backporting some PHP 8.0+ features to lower PHP versions", + "homepage": "https://symfony.com", + "keywords": [ + "compatibility", + "polyfill", + "portable", + "shim" + ], + "time": "2023-01-26T09:26:14+00:00" + }, + { + "name": "symfony/service-contracts", + "version": "v2.5.2", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/symfony/service-contracts/v2.5.2/symfony-service-contracts-v2.5.2.zip", + "reference": "4b426aac47d6427cc1a1d0f7e2ac724627f5966c", + "shasum": "" + }, + "require": { + "php": ">=7.2.5", + "psr/container": "^1.1", + "symfony/deprecation-contracts": "^2.1|^3" + }, + "conflict": { + "ext-psr": "<1.1|>=2" + }, + "suggest": { + "symfony/service-implementation": "" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "2.5-dev" + }, + "thanks": { + "name": "symfony/contracts", + "url": "https://github.com/symfony/contracts" + } + }, + "autoload": { + "psr-4": { + "Symfony\\Contracts\\Service\\": "" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Generic abstractions related to writing services", + "homepage": "https://symfony.com", + "keywords": [ + "abstractions", + "contracts", + "decoupling", + "interfaces", + "interoperability", + "standards" + ], + "time": "2022-05-30T19:17:29+00:00" + }, + { + "name": "symfony/string", + "version": "v6.3.2", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/symfony/string/v6.3.2/symfony-string-v6.3.2.zip", + "reference": "53d1a83225002635bca3482fcbf963001313fb68", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "symfony/polyfill-ctype": "~1.8", + "symfony/polyfill-intl-grapheme": "~1.0", + "symfony/polyfill-intl-normalizer": "~1.0", + "symfony/polyfill-mbstring": "~1.0" + }, + "conflict": { + "symfony/translation-contracts": "<2.5" + }, + "require-dev": { + "symfony/error-handler": "^5.4|^6.0", + "symfony/http-client": "^5.4|^6.0", + "symfony/intl": "^6.2", + "symfony/translation-contracts": "^2.5|^3.0", + "symfony/var-exporter": "^5.4|^6.0" + }, + "type": "library", + "autoload": { + "files": [ + "Resources/functions.php" + ], + "psr-4": { + "Symfony\\Component\\String\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Provides an object-oriented API to strings and deals with bytes, UTF-8 code points and grapheme clusters in a unified way", + "homepage": "https://symfony.com", + "keywords": [ + "grapheme", + "i18n", + "string", + "unicode", + "utf-8", + "utf8" + ], + "time": "2023-07-05T08:41:27+00:00" + }, + { + "name": "symfony/translation", + "version": "v6.3.3", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/symfony/translation/v6.3.3/symfony-translation-v6.3.3.zip", + "reference": "3ed078c54bc98bbe4414e1e9b2d5e85ed5a5c8bd", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "symfony/deprecation-contracts": "^2.5|^3", + "symfony/polyfill-mbstring": "~1.0", + "symfony/translation-contracts": "^2.5|^3.0" + }, + "conflict": { + "symfony/config": "<5.4", + "symfony/console": "<5.4", + "symfony/dependency-injection": "<5.4", + "symfony/http-client-contracts": "<2.5", + "symfony/http-kernel": "<5.4", + "symfony/service-contracts": "<2.5", + "symfony/twig-bundle": "<5.4", + "symfony/yaml": "<5.4" + }, + "provide": { + "symfony/translation-implementation": "2.3|3.0" + }, + "require-dev": { + "nikic/php-parser": "^4.13", + "psr/log": "^1|^2|^3", + "symfony/config": "^5.4|^6.0", + "symfony/console": "^5.4|^6.0", + "symfony/dependency-injection": "^5.4|^6.0", + "symfony/finder": "^5.4|^6.0", + "symfony/http-client-contracts": "^2.5|^3.0", + "symfony/http-kernel": "^5.4|^6.0", + "symfony/intl": "^5.4|^6.0", + "symfony/polyfill-intl-icu": "^1.21", + "symfony/routing": "^5.4|^6.0", + "symfony/service-contracts": "^2.5|^3", + "symfony/yaml": "^5.4|^6.0" + }, + "type": "library", + "autoload": { + "files": [ + "Resources/functions.php" + ], + "psr-4": { + "Symfony\\Component\\Translation\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Fabien Potencier", + "email": "fabien@symfony.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Provides tools to internationalize your application", + "homepage": "https://symfony.com", + "time": "2023-07-31T07:08:24+00:00" + }, + { + "name": "symfony/translation-contracts", + "version": "v3.3.0", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/symfony/translation-contracts/v3.3.0/symfony-translation-contracts-v3.3.0.zip", + "reference": "02c24deb352fb0d79db5486c0c79905a85e37e86", + "shasum": "" + }, + "require": { + "php": ">=8.1" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "3.4-dev" + }, + "thanks": { + "name": "symfony/contracts", + "url": "https://github.com/symfony/contracts" + } + }, + "autoload": { + "psr-4": { + "Symfony\\Contracts\\Translation\\": "" + }, + "exclude-from-classmap": [ + "/Test/" + ] + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Generic abstractions related to translation", + "homepage": "https://symfony.com", + "keywords": [ + "abstractions", + "contracts", + "decoupling", + "interfaces", + "interoperability", + "standards" + ], + "time": "2023-05-30T17:17:10+00:00" + }, + { + "name": "symfony/var-dumper", + "version": "v6.3.4", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/symfony/var-dumper/v6.3.4/symfony-var-dumper-v6.3.4.zip", + "reference": "2027be14f8ae8eae999ceadebcda5b4909b81d45", + "shasum": "" + }, + "require": { + "php": ">=8.1", + "symfony/deprecation-contracts": "^2.5|^3", + "symfony/polyfill-mbstring": "~1.0" + }, + "conflict": { + "symfony/console": "<5.4" + }, + "require-dev": { + "ext-iconv": "*", + "symfony/console": "^5.4|^6.0", + "symfony/http-kernel": "^5.4|^6.0", + "symfony/process": "^5.4|^6.0", + "symfony/uid": "^5.4|^6.0", + "twig/twig": "^2.13|^3.0.4" + }, + "bin": [ + "Resources/bin/var-dump-server" + ], + "type": "library", + "autoload": { + "files": [ + "Resources/functions/dump.php" + ], + "psr-4": { + "Symfony\\Component\\VarDumper\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Provides mechanisms for walking through any arbitrary PHP variable", + "homepage": "https://symfony.com", + "keywords": [ + "debug", + "dump" + ], + "time": "2023-08-24T14:51:05+00:00" + }, + { + "name": "symfony/var-exporter", + "version": "v6.3.4", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/symfony/var-exporter/v6.3.4/symfony-var-exporter-v6.3.4.zip", + "reference": "df1f8aac5751871b83d30bf3e2c355770f8f0691", + "shasum": "" + }, + "require": { + "php": ">=8.1" + }, + "require-dev": { + "symfony/var-dumper": "^5.4|^6.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Symfony\\Component\\VarExporter\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Nicolas Grekas", + "email": "p@tchwork.com" + }, + { + "name": "Symfony Community", + "homepage": "https://symfony.com/contributors" + } + ], + "description": "Allows exporting any serializable PHP data structure to plain PHP code", + "homepage": "https://symfony.com", + "keywords": [ + "clone", + "construct", + "export", + "hydrate", + "instantiate", + "lazy-loading", + "proxy", + "serialize" + ], + "time": "2023-08-16T18:14:47+00:00" + }, + { + "name": "taoser/webman-validate", + "version": "v1.7.2", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/taoser/webman-validate/v1.7.2/taoser-webman-validate-v1.7.2.zip", + "reference": "f6ceda3700cabf3e48966b845fd980faa292127e", + "shasum": "" + }, + "require": { + "php": ">=7.2.0", + "symfony/translation": "^5.4|^6.0", + "topthink/think-container": "~2.0", + "topthink/think-helper": "^3.1" + }, + "type": "library", + "autoload": { + "files": [ + "src/helper.php" + ], + "psr-4": { + "taoser\\": "src/" + } + }, + "license": [ + "mit" + ], + "authors": [ + { + "name": "taoser", + "email": "changlin_zhao@qq.com" + } + ], + "description": "The webman Validate Package", + "time": "2022-08-27T08:29:08+00:00" + }, + { + "name": "textalk/websocket", + "version": "1.5.8", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/textalk/websocket/1.5.8/textalk-websocket-1.5.8.zip", + "reference": "d05dbaa97500176447ffb1f1800573f23085ab13", + "shasum": "" + }, + "require": { + "php": "^7.2 | ^8.0", + "psr/log": "^1 | ^2 | ^3" + }, + "require-dev": { + "php-coveralls/php-coveralls": "^2.0", + "phpunit/phpunit": "^8.0|^9.0", + "squizlabs/php_codesniffer": "^3.5" + }, + "type": "library", + "autoload": { + "psr-4": { + "WebSocket\\": "lib" + } + }, + "license": [ + "ISC" + ], + "authors": [ + { + "name": "Fredrik Liljegren" + }, + { + "name": "Sören Jensen", + "email": "soren@abicart.se" + } + ], + "description": "WebSocket client and server", + "time": "2022-04-26T06:28:24+00:00" + }, + { + "name": "tinywan/storage", + "version": "v0.3.4", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/tinywan/storage/v0.3.4/tinywan-storage-v0.3.4.zip", + "reference": "0867631bbd1731658ac745481131ef93415cdf62", + "shasum": "" + }, + "require": { + "php": ">=7.2", + "workerman/webman-framework": "^1.2.1" + }, + "require-dev": { + "aliyuncs/oss-sdk-php": "^2.4", + "friendsofphp/php-cs-fixer": "^3.6", + "phpstan/phpstan": "^1.4", + "phpunit/phpunit": "^9.5", + "qcloud/cos-sdk-v5": "^2.5", + "qiniu/php-sdk": "^7.4", + "workerman/webman": "^1.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Tinywan\\Storage\\": "src" + } + }, + "license": [ + "MIT" + ], + "description": "webman storage plugin", + "time": "2022-08-03T12:12:34+00:00" + }, + { + "name": "topthink/think-container", + "version": "v2.0.5", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/topthink/think-container/v2.0.5/topthink-think-container-v2.0.5.zip", + "reference": "2189b39e42af2c14203ed4372b92e38989e9dabb", + "shasum": "" + }, + "require": { + "php": ">=7.2.0", + "psr/container": "^1.0|^2.0", + "topthink/think-helper": "^3.1" + }, + "require-dev": { + "phpunit/phpunit": "^7.0|^8.0" + }, + "type": "library", + "autoload": { + "files": [], + "psr-4": { + "think\\": "src" + } + }, + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "liu21st", + "email": "liu21st@gmail.com" + } + ], + "description": "PHP Container & Facade Manager", + "time": "2022-05-23T06:24:54+00:00" + }, + { + "name": "topthink/think-helper", + "version": "v3.1.6", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/topthink/think-helper/v3.1.6/topthink-think-helper-v3.1.6.zip", + "reference": "769acbe50a4274327162f9c68ec2e89a38eb2aff", + "shasum": "" + }, + "require": { + "php": ">=7.1.0" + }, + "require-dev": { + "phpunit/phpunit": "^9.5" + }, + "type": "library", + "autoload": { + "files": [ + "src/helper.php" + ], + "psr-4": { + "think\\": "src" + } + }, + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "yunwuxin", + "email": "448901948@qq.com" + } + ], + "description": "The ThinkPHP6 Helper Package", + "time": "2021-12-15T04:27:55+00:00" + }, + { + "name": "topthink/think-orm", + "version": "v3.0.13", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/topthink/think-orm/v3.0.13/topthink-think-orm-v3.0.13.zip", + "reference": "e676e172d54055266005986d05685bdac5ac66b7", + "shasum": "" + }, + "require": { + "ext-json": "*", + "ext-pdo": "*", + "php": ">=8.0.0", + "psr/log": ">=1.0", + "psr/simple-cache": ">=1.0", + "topthink/think-helper": "^3.1" + }, + "require-dev": { + "phpunit/phpunit": "^8|^9.5|^10" + }, + "type": "library", + "autoload": { + "files": [ + "stubs/load_stubs.php" + ], + "psr-4": { + "think\\": "src" + } + }, + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "liu21st", + "email": "liu21st@gmail.com" + } + ], + "description": "the PHP Database&ORM Framework", + "keywords": [ + "database", + "orm" + ], + "time": "2023-09-01T09:08:49+00:00" + }, + { + "name": "vlucas/phpdotenv", + "version": "v5.5.0", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/vlucas/phpdotenv/v5.5.0/vlucas-phpdotenv-v5.5.0.zip", + "reference": "1a7ea2afc49c3ee6d87061f5a233e3a035d0eae7", + "shasum": "" + }, + "require": { + "ext-pcre": "*", + "graham-campbell/result-type": "^1.0.2", + "php": "^7.1.3 || ^8.0", + "phpoption/phpoption": "^1.8", + "symfony/polyfill-ctype": "^1.23", + "symfony/polyfill-mbstring": "^1.23.1", + "symfony/polyfill-php80": "^1.23.1" + }, + "require-dev": { + "bamarni/composer-bin-plugin": "^1.4.1", + "ext-filter": "*", + "phpunit/phpunit": "^7.5.20 || ^8.5.30 || ^9.5.25" + }, + "suggest": { + "ext-filter": "Required to use the boolean validator." + }, + "type": "library", + "extra": { + "bamarni-bin": { + "bin-links": true, + "forward-command": true + }, + "branch-alias": { + "dev-master": "5.5-dev" + } + }, + "autoload": { + "psr-4": { + "Dotenv\\": "src/" + } + }, + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Graham Campbell", + "email": "hello@gjcampbell.co.uk", + "homepage": "https://github.com/GrahamCampbell" + }, + { + "name": "Vance Lucas", + "email": "vance@vancelucas.com", + "homepage": "https://github.com/vlucas" + } + ], + "description": "Loads environment variables from `.env` to `getenv()`, `$_ENV` and `$_SERVER` automagically.", + "keywords": [ + "dotenv", + "env", + "environment" + ], + "time": "2022-10-16T01:01:54+00:00" + }, + { + "name": "voku/portable-ascii", + "version": "2.0.1", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/voku/portable-ascii/2.0.1/voku-portable-ascii-2.0.1.zip", + "reference": "b56450eed252f6801410d810c8e1727224ae0743", + "shasum": "" + }, + "require": { + "php": ">=7.0.0" + }, + "require-dev": { + "phpunit/phpunit": "~6.0 || ~7.0 || ~9.0" + }, + "suggest": { + "ext-intl": "Use Intl for transliterator_transliterate() support" + }, + "type": "library", + "autoload": { + "psr-4": { + "voku\\": "src/voku/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Lars Moelleken", + "homepage": "http://www.moelleken.org/" + } + ], + "description": "Portable ASCII library - performance optimized (ascii) string functions for php.", + "homepage": "https://github.com/voku/portable-ascii", + "keywords": [ + "ascii", + "clean", + "php" + ], + "time": "2022-03-08T17:03:00+00:00" + }, + { + "name": "webman/console", + "version": "v1.2.38", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/webman/console/v1.2.38/webman-console-v1.2.38.zip", + "reference": "2a48e1d196e67f4deba2dc9d624fb09682bb034d", + "shasum": "" + }, + "require": { + "doctrine/inflector": "^2.0", + "symfony/console": ">=5.0" + }, + "require-dev": { + "workerman/webman": "^1.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Webman\\Console\\": "src" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "walkor", + "email": "walkor@workerman.net", + "homepage": "http://www.workerman.net", + "role": "Developer" + } + ], + "description": "Webman console", + "homepage": "http://www.workerman.net", + "keywords": [ + "webman console" + ], + "time": "2023-08-31T06:36:57+00:00" + }, + { + "name": "webman/log", + "version": "v1.1.5", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/webman/log/v1.1.5/webman-log-v1.1.5.zip", + "reference": "28f722778ef722a78c9be2565d537eea7805cc01", + "shasum": "" + }, + "type": "library", + "autoload": { + "psr-4": { + "Webman\\Log\\": "src" + } + }, + "license": [ + "MIT" + ], + "description": "Webman plugin webman/log", + "time": "2022-12-28T08:12:41+00:00" + }, + { + "name": "webman/think-orm", + "version": "v1.1.1", + "dist": { + "type": "zip", + "url": "https://mirrors.cloud.tencent.com/repository/composer/webman/think-orm/v1.1.1/webman-think-orm-v1.1.1.zip", + "reference": "9f1e525c5c4b5a2e1eee6a4f82ef5d23c69139a2", + "shasum": "" + }, + "require": { + "topthink/think-orm": "^2.0.53 || ^3.0.0", + "workerman/webman-framework": "^1.2.1" + }, + "type": "library", + "autoload": { + "psr-4": { + "Webman\\ThinkOrm\\": "src" + } + }, + "license": [ + "MIT" + ], + "time": "2023-04-23T14:40:18+00:00" + }, + { + "name": "webmozart/assert", + "version": "1.11.0", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/webmozart/assert/1.11.0/webmozart-assert-1.11.0.zip", + "reference": "11cb2199493b2f8a3b53e7f19068fc6aac760991", + "shasum": "" + }, + "require": { + "ext-ctype": "*", + "php": "^7.2 || ^8.0" + }, + "conflict": { + "phpstan/phpstan": "<0.12.20", + "vimeo/psalm": "<4.6.1 || 4.6.2" + }, + "require-dev": { + "phpunit/phpunit": "^8.5.13" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.10-dev" + } + }, + "autoload": { + "psr-4": { + "Webmozart\\Assert\\": "src/" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Bernhard Schussek", + "email": "bschussek@gmail.com" + } + ], + "description": "Assertions to validate method input/output with nice error messages.", + "keywords": [ + "assert", + "check", + "validate" + ], + "time": "2022-06-03T18:03:27+00:00" + }, + { + "name": "workerman/webman-framework", + "version": "v1.5.8", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/workerman/webman-framework/v1.5.8/workerman-webman-framework-v1.5.8.zip", + "reference": "1cf2d1c8231dd4acaf88a491170552747f325732", + "shasum": "" + }, + "require": { + "ext-json": "*", + "nikic/fast-route": "^1.3", + "php": ">=7.2", + "psr/container": ">=1.0", + "workerman/workerman": "^4.0.4 || ^5.0.0" + }, + "suggest": { + "ext-event": "For better performance. " + }, + "type": "library", + "autoload": { + "psr-4": { + "Webman\\": "./src", + "Support\\": "./src/support", + "support\\": "./src/support", + "Support\\View\\": "./src/support/view", + "Support\\Bootstrap\\": "./src/support/bootstrap", + "Support\\Exception\\": "./src/support/exception" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "walkor", + "email": "walkor@workerman.net", + "homepage": "https://www.workerman.net", + "role": "Developer" + } + ], + "description": "High performance HTTP Service Framework.", + "homepage": "https://www.workerman.net", + "keywords": [ + "High Performance", + "http service" + ], + "time": "2023-07-21T09:39:36+00:00" + }, + { + "name": "workerman/workerman", + "version": "v4.1.13", + "dist": { + "type": "zip", + "url": "https://mirrors.tencent.com/repository/composer/workerman/workerman/v4.1.13/workerman-workerman-v4.1.13.zip", + "reference": "807780ff672775fcd08f89e573a2824e939021ce", + "shasum": "" + }, + "require": { + "php": ">=7.0" + }, + "suggest": { + "ext-event": "For better performance. " + }, + "type": "library", + "autoload": { + "psr-4": { + "Workerman\\": "./" + } + }, + "license": [ + "MIT" + ], + "authors": [ + { + "name": "walkor", + "email": "walkor@workerman.net", + "homepage": "http://www.workerman.net", + "role": "Developer" + } + ], + "description": "An asynchronous event driven PHP framework for easily building fast, scalable network applications.", + "homepage": "http://www.workerman.net", + "keywords": [ + "asynchronous", + "event-loop" + ], + "time": "2023-07-31T05:57:25+00:00" + } + ], + "packages-dev": [], + "aliases": [], + "minimum-stability": "stable", + "stability-flags": [], + "prefer-stable": false, + "prefer-lowest": false, + "platform": { + "php": ">=7.2", + "ext-json": "*" + }, + "platform-dev": [], + "plugin-api-version": "2.3.0" +} diff --git a/composer.phar b/composer.phar new file mode 100755 index 0000000..d39c3e6 Binary files /dev/null and b/composer.phar differ diff --git a/config/app.php b/config/app.php new file mode 100644 index 0000000..7e0f531 --- /dev/null +++ b/config/app.php @@ -0,0 +1,27 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use support\Request; + +return [ + 'debug' => true, + 'error_reporting' => E_ALL, + 'default_timezone' => 'Asia/Shanghai', + 'request_class' => Request::class, + 'public_path' => base_path() . DIRECTORY_SEPARATOR . 'public', + 'runtime_path' => base_path(false) . DIRECTORY_SEPARATOR . 'runtime', + 'controller_suffix' => 'Controller', + 'controller_reuse' => false, + 'support_php_files' => true, +]; diff --git a/config/autoload.php b/config/autoload.php new file mode 100644 index 0000000..69a8135 --- /dev/null +++ b/config/autoload.php @@ -0,0 +1,21 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return [ + 'files' => [ + base_path() . '/app/functions.php', + base_path() . '/support/Request.php', + base_path() . '/support/Response.php', + ] +]; diff --git a/config/bootstrap.php b/config/bootstrap.php new file mode 100644 index 0000000..3257ec5 --- /dev/null +++ b/config/bootstrap.php @@ -0,0 +1,19 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return [ + support\bootstrap\Session::class, + support\bootstrap\LaravelDb::class, + Webman\ThinkOrm\ThinkOrm::class, +]; diff --git a/config/container.php b/config/container.php new file mode 100644 index 0000000..106b7b4 --- /dev/null +++ b/config/container.php @@ -0,0 +1,15 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return new Webman\Container; \ No newline at end of file diff --git a/config/database.php b/config/database.php new file mode 100644 index 0000000..48402ad --- /dev/null +++ b/config/database.php @@ -0,0 +1,24 @@ + 'mysql', + // 各种数据库配置 + 'connections' => [ + + 'mysql' => [ + 'driver' => 'mysql', + 'host' => env('DB_HOST'), + 'port' => 3306, + 'database' => env('DB_DATABASE'), + 'username' => env('DB_USERNAME'), + 'password' => env('DB_PASSWORD'), + 'unix_socket' => '', + 'prefix' => 'la_', + 'strict' => true, + 'engine' => null, + ], + ], + ]; \ No newline at end of file diff --git a/config/dependence.php b/config/dependence.php new file mode 100644 index 0000000..8e964ed --- /dev/null +++ b/config/dependence.php @@ -0,0 +1,15 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return []; \ No newline at end of file diff --git a/config/exception.php b/config/exception.php new file mode 100644 index 0000000..df5efd9 --- /dev/null +++ b/config/exception.php @@ -0,0 +1,18 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return [ +// '' => support\exception\Handler::class, + '' => app\common\exception\Handler::class, +]; \ No newline at end of file diff --git a/config/log.php b/config/log.php new file mode 100644 index 0000000..7f05de5 --- /dev/null +++ b/config/log.php @@ -0,0 +1,32 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return [ + 'default' => [ + 'handlers' => [ + [ + 'class' => Monolog\Handler\RotatingFileHandler::class, + 'constructor' => [ + runtime_path() . '/logs/webman.log', + 7, //$maxFiles + Monolog\Logger::DEBUG, + ], + 'formatter' => [ + 'class' => Monolog\Formatter\LineFormatter::class, + 'constructor' => [null, 'Y-m-d H:i:s', true], + ], + ] + ], + ], +]; diff --git a/config/middleware.php b/config/middleware.php new file mode 100644 index 0000000..78cbaee --- /dev/null +++ b/config/middleware.php @@ -0,0 +1,26 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return [ + 'admin'=>[ + // 跨域中间件 + app\common\http\middleware\AdminAllowMiddleware::class, + // 初始化 + app\admin\middleware\InitMiddleware::class, + // 登录验证 + app\admin\middleware\LoginMiddleware::class, + // 权限认证 + app\admin\middleware\AuthMiddleware::class, + ] +]; \ No newline at end of file diff --git a/config/plugin/tinywan/storage/app.php b/config/plugin/tinywan/storage/app.php new file mode 100644 index 0000000..70969be --- /dev/null +++ b/config/plugin/tinywan/storage/app.php @@ -0,0 +1,62 @@ + true, + 'storage' => [ + 'default' => 'local', // local:本地 oss:阿里云 cos:腾讯云 qos:七牛云 + 'single_limit' => 1024 * 1024 * 200, // 单个文件的大小限制,默认200M 1024 * 1024 * 200 + 'total_limit' => 1024 * 1024 * 200, // 所有文件的大小限制,默认200M 1024 * 1024 * 200 + 'nums' => 10, // 文件数量限制,默认10 + 'include' => [], // 被允许的文件类型列表 + 'exclude' => [], // 不被允许的文件类型列表 + // 本地对象存储 + 'local' => [ + 'adapter' => \Tinywan\Storage\Adapter\LocalAdapter::class, + 'root' => runtime_path().'/storage', + 'dirname' => function () { + return date('Ymd'); + }, + 'domain' => 'http://127.0.0.1:8787', + 'uri' => '/runtime', // 如果 domain + uri 不在 public 目录下,请做好软链接,否则生成的url无法访问 + 'algo' => 'sha1', + ], + // 阿里云对象存储 + 'oss' => [ + 'adapter' => \Tinywan\Storage\Adapter\OssAdapter::class, + 'accessKeyId' => 'xxxxxxxxxxxx', + 'accessKeySecret' => 'xxxxxxxxxxxx', + 'bucket' => 'resty-webman', + 'dirname' => function () { + return 'storage'; + }, + 'domain' => 'http://webman.oss.tinywan.com', + 'endpoint' => 'oss-cn-hangzhou.aliyuncs.com', + 'algo' => 'sha1', + ], + // 腾讯云对象存储 + 'cos' => [ + 'adapter' => \Tinywan\Storage\Adapter\CosAdapter::class, + 'secretId' => 'xxxxxxxxxxxxx', + 'secretKey' => 'xxxxxxxxxxxx', + 'bucket' => 'resty-webman-xxxxxxxxx', + 'dirname' => 'storage', + 'domain' => 'http://webman.oss.tinywan.com', + 'region' => 'ap-shanghai', + ], + // 七牛云对象存储 + 'qiniu' => [ + 'adapter' => \Tinywan\Storage\Adapter\QiniuAdapter::class, + 'accessKey' => 'xxxxxxxxxxxxx', + 'secretKey' => 'xxxxxxxxxxxxx', + 'bucket' => 'resty-webman', + 'dirname' => 'storage', + 'domain' => 'http://webman.oss.tinywan.com', + ], + ], +]; diff --git a/config/plugin/webman/console/app.php b/config/plugin/webman/console/app.php new file mode 100644 index 0000000..b6cef57 --- /dev/null +++ b/config/plugin/webman/console/app.php @@ -0,0 +1,18 @@ + true, + + 'phar_file_output_dir' => BASE_PATH . DIRECTORY_SEPARATOR . 'build', + + 'phar_filename' => 'webman.phar', + + 'signature_algorithm'=> Phar::SHA256, //set the signature algorithm for a phar and apply it. The signature algorithm must be one of Phar::MD5, Phar::SHA1, Phar::SHA256, Phar::SHA512, or Phar::OPENSSL. + + 'private_key_file' => '', // The file path for certificate or OpenSSL private key file. + + //'exclude_pattern' => '#^(?!.*(config/plugin/webman/console/app.php|webman/console/src/Commands/(PharPackCommand.php|ReloadCommand.php)|LICENSE|composer.json|.github|.idea|doc|docs|.git|.setting|runtime|test|test_old|tests|Tests|vendor-bin|.md))(.*)$#', + + 'exclude_files' => [ + '.env', 'LICENSE', 'composer.json', 'composer.lock','start.php' + ] +]; diff --git a/config/plugin/webman/log/app.php b/config/plugin/webman/log/app.php new file mode 100644 index 0000000..391150e --- /dev/null +++ b/config/plugin/webman/log/app.php @@ -0,0 +1,12 @@ + true, + 'exception' => [ + // 是否记录异常到日志 + 'enable' => true, + // 不会记录到日志的异常类 + 'dontReport' => [ + support\exception\BusinessException::class + ] + ] +]; diff --git a/config/plugin/webman/log/middleware.php b/config/plugin/webman/log/middleware.php new file mode 100644 index 0000000..acf9763 --- /dev/null +++ b/config/plugin/webman/log/middleware.php @@ -0,0 +1,21 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use Webman\Log\Middleware; + +return [ + '' => [ + Middleware::class + ] +]; \ No newline at end of file diff --git a/config/process.php b/config/process.php new file mode 100644 index 0000000..f22f488 --- /dev/null +++ b/config/process.php @@ -0,0 +1,42 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use Workerman\Worker; + +return [ + // File update detection and automatic reload + 'monitor' => [ + 'handler' => process\Monitor::class, + 'reloadable' => false, + 'constructor' => [ + // Monitor these directories + 'monitor_dir' => array_merge([ + app_path(), + config_path(), + base_path() . '/process', + base_path() . '/support', + base_path() . '/resource', + base_path() . '/.env', + ], glob(base_path() . '/plugin/*/app'), glob(base_path() . '/plugin/*/config'), glob(base_path() . '/plugin/*/api')), + // Files with these suffixes will be monitored + 'monitor_extensions' => [ + 'php', 'html', 'htm', 'env' + ], + 'options' => [ + 'enable_file_monitor' => !Worker::$daemonize && DIRECTORY_SEPARATOR === '/', + 'enable_memory_monitor' => DIRECTORY_SEPARATOR === '/', + ] + ] + ] +]; diff --git a/config/project.php b/config/project.php new file mode 100755 index 0000000..e835d30 --- /dev/null +++ b/config/project.php @@ -0,0 +1,99 @@ + '1.5.0', + + // 官网 + 'website' => [ + 'name' => getenv('project.web_name')?:'likeadmin', // 网站名称 + 'url' => getenv('project.web_url')?:'www.likeadmin.cn/', // 网站地址 + 'login_image' => 'image/admin/login_image.png', + 'web_logo' => 'image/admin/web_logo.jpg', // 网站logo + 'web_favicon' => 'image/admin/web_favicon.ico', // 网站图标 + 'shop_name' => 'likeadmin', // 商城名称 + 'shop_logo' => 'image/admin/shop_logo.png', // 商城图标 + 'pc_logo' => 'image/admin/pc_logo.png', // pc_logo + 'pc_ico' => 'image/admin/web_favicon.ico', // pc_ico + 'pc_title' => 'likeadmin', // PC网站标题 + ], + + // 后台登录 + 'admin_login' => [ + // 管理后台登录限制 0-不限制 1-需要限制 + 'login_restrictions' => 1, + // 限制密码错误次数 + 'password_error_times' => 5, + // 限制禁止多少分钟不能登录 + 'limit_login_time' => 30, + ], + + // 唯一标识,密码盐、路径加密等 + 'unique_identification' => getenv('UNIQUE_IDENTIFICATION', 'likeadmin'), + + // 后台管理员token(登录令牌)配置 + 'admin_token' => [ + 'expire_duration' => 3600 * 8,//管理后台token过期时长(单位秒) + 'be_expire_duration' => 3600,//管理后台token临时过期前时长,自动续期 + ], + + // 商城用户token(登录令牌)配置 + 'user_token' => [ + 'expire_duration' => 3600 * 8,//用户token过期时长(单位秒) + 'be_expire_duration' => 3600,//用户token临时过期前时长,自动续期 + ], + + // 列表页 + 'lists' => [ + 'page_size_max' => 25000,//列表页查询数量限制(列表页每页数量、导出每页数量) + 'page_size' => 25, //默认每页数量 + ], + + // 各种默认图片 + 'default_image' => [ + 'admin_avatar' => 'image/admin/avatar.png', + 'user_avatar' => 'image/admin/default_avatar.png', + 'qq_group' => 'image/admin/qq_group.png', // qq群 + 'customer_service' => 'image/admin/customer_service.jpg', // 客服 + 'menu_admin' => '/image/admin/menu_admin.png',// 首页快捷菜单-管理员 + 'menu_role' => 'image/admin/menu_role.png', // 首页快捷菜单-角色 + 'menu_dept' => 'image/admin/menu_dept.png',// 首页快捷菜单-部门 + 'menu_dict' => 'image/admin/menu_dict.png',// 首页快捷菜单-字典 + 'menu_generator' => 'image/admin/menu_generator.png',// 首页快捷菜单-代码生成器 + 'menu_auth' => 'image/admin/menu_auth.png',// 首页快捷菜单-菜单权限 + 'menu_web' => 'image/admin/menu_web.png',// 首页快捷菜单-网站信息 + 'menu_file' => 'image/admin/menu_file.png',// 首页快捷菜单-素材中心 + ], + + // 文件上传限制 (图片) + 'file_image' => [ + 'jpg', 'png', 'gif', 'jpeg', 'webp' + ], + + // 文件上传限制 (视频) + 'file_video' => [ + 'wmv', 'avi', 'mpg', 'mpeg', '3gp', 'mov', 'mp4', 'flv', 'f4v', 'rmvb', 'mkv' + ], + + // 登录设置 + 'login' => [ + // 登录方式:1-账号密码登录;2-手机短信验证码登录 + 'login_way' => ['1', '2'], + // 注册强制绑定手机 0-关闭 1-开启 + 'coerce_mobile' => 1, + // 第三方授权登录 0-关闭 1-开启 + 'third_auth' => 1, + // 微信授权登录 0-关闭 1-开启 + 'wechat_auth' => 1, + // qq授权登录 0-关闭 1-开启 + 'qq_auth' => 0, + // 登录政策协议 0-关闭 1-开启 + 'login_agreement' => 1, + ], + + // 后台装修 + 'decorate' => [ + // 底部导航栏样式设置 + 'tabbar_style' => ['default_color' => '#999999', 'selected_color' => '#4173ff'], + ] + +]; diff --git a/config/redis.php b/config/redis.php new file mode 100644 index 0000000..2f9757a --- /dev/null +++ b/config/redis.php @@ -0,0 +1,22 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return [ + 'default' => [ + 'host' => '127.0.0.1', + 'password' => null, + 'port' => 6379, + 'database' => 0, + ], +]; diff --git a/config/route.php b/config/route.php new file mode 100644 index 0000000..2e7b79f --- /dev/null +++ b/config/route.php @@ -0,0 +1,26 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use Webman\Route; +Route::group('/admin',function (){ + //获取网站信息 + Route::get('/get_web_site',[app\admin\controller\setting\web\WebSettingController::class,'getWebsite']); + //设置网站信息 + Route::post('/set_web_site',[app\admin\controller\setting\web\WebSettingController::class,'setWebsite']); +}); + + + + + diff --git a/config/server.php b/config/server.php new file mode 100644 index 0000000..84ea0ef --- /dev/null +++ b/config/server.php @@ -0,0 +1,31 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return [ + 'listen' => 'http://0.0.0.0:8787', + 'transport' => 'tcp', + 'context' => [], + 'name' => 'webman', + 'count' => cpu_count() * 4, + 'user' => '', + 'group' => '', + 'reusePort' => false, + 'event_loop' => '', + 'stop_timeout' => 2, + 'pid_file' => runtime_path() . '/webman.pid', + 'status_file' => runtime_path() . '/webman.status', + 'stdout_file' => runtime_path() . '/logs/stdout.log', + 'log_file' => runtime_path() . '/logs/workerman.log', + 'max_package_size' => 10 * 1024 * 1024 +]; diff --git a/config/session.php b/config/session.php new file mode 100644 index 0000000..043f8c4 --- /dev/null +++ b/config/session.php @@ -0,0 +1,65 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use Webman\Session\FileSessionHandler; +use Webman\Session\RedisSessionHandler; +use Webman\Session\RedisClusterSessionHandler; + +return [ + + 'type' => 'file', // or redis or redis_cluster + + 'handler' => FileSessionHandler::class, + + 'config' => [ + 'file' => [ + 'save_path' => runtime_path() . '/sessions', + ], + 'redis' => [ + 'host' => '127.0.0.1', + 'port' => 6379, + 'auth' => '', + 'timeout' => 2, + 'database' => '', + 'prefix' => 'redis_session_', + ], + 'redis_cluster' => [ + 'host' => ['127.0.0.1:7000', '127.0.0.1:7001', '127.0.0.1:7001'], + 'timeout' => 2, + 'auth' => '', + 'prefix' => 'redis_session_', + ] + ], + + 'session_name' => 'PHPSID', + + 'auto_update_timestamp' => false, + + 'lifetime' => 7*24*60*60, + + 'cookie_lifetime' => 365*24*60*60, + + 'cookie_path' => '/', + + 'domain' => '', + + 'http_only' => true, + + 'secure' => false, + + 'same_site' => '', + + 'gc_probability' => [1, 1000], + +]; diff --git a/config/static.php b/config/static.php new file mode 100644 index 0000000..6313679 --- /dev/null +++ b/config/static.php @@ -0,0 +1,23 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +/** + * Static file settings + */ +return [ + 'enable' => true, + 'middleware' => [ // Static file Middleware + //app\middleware\StaticFile::class, + ], +]; \ No newline at end of file diff --git a/config/thinkorm.php b/config/thinkorm.php new file mode 100644 index 0000000..77eb1a6 --- /dev/null +++ b/config/thinkorm.php @@ -0,0 +1,31 @@ + 'mysql', + 'connections' => [ + 'mysql' => [ + // 数据库类型 + 'type' => 'mysql', + // 服务器地址 + 'hostname' => getenv('DB_HOST'), + // 数据库名 + 'database' => getenv('DB_DATABASE'), + // 数据库用户名 + 'username' => getenv('DB_USERNAME'), + // 数据库密码 + 'password' => getenv('DB_PASSWORD'), + // 数据库连接端口 + 'hostport' => getenv('DB_PORT'), + // 数据库编码默认采用utf8 + 'charset' => 'utf8', + // 数据库表前缀 + 'prefix' => 'la_', + // 断线重连 + 'break_reconnect' => true, + // 关闭SQL监听日志 + 'trigger_sql' => true, + // 自定义分页类 + 'bootstrap' => '' + ], + ], +]; diff --git a/config/translation.php b/config/translation.php new file mode 100644 index 0000000..96589b2 --- /dev/null +++ b/config/translation.php @@ -0,0 +1,25 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +/** + * Multilingual configuration + */ +return [ + // Default language + 'locale' => 'zh_CN', + // Fallback language + 'fallback_locale' => ['zh_CN', 'en'], + // Folder where language files are stored + 'path' => base_path() . '/resource/translations', +]; \ No newline at end of file diff --git a/config/view.php b/config/view.php new file mode 100644 index 0000000..e3a7b85 --- /dev/null +++ b/config/view.php @@ -0,0 +1,22 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use support\view\Raw; +use support\view\Twig; +use support\view\Blade; +use support\view\ThinkPHP; + +return [ + 'handler' => Raw::class +]; diff --git a/process/Monitor.php b/process/Monitor.php new file mode 100644 index 0000000..5649e71 --- /dev/null +++ b/process/Monitor.php @@ -0,0 +1,240 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +namespace process; + +use Workerman\Timer; +use Workerman\Worker; + +/** + * Class FileMonitor + * @package process + */ +class Monitor +{ + /** + * @var array + */ + protected $_paths = []; + + /** + * @var array + */ + protected $_extensions = []; + + /** + * @var string + */ + public static $lockFile = __DIR__ . '/../runtime/monitor.lock'; + + /** + * Pause monitor + * @return void + */ + public static function pause() + { + file_put_contents(static::$lockFile, time()); + } + + /** + * Resume monitor + * @return void + */ + public static function resume() + { + clearstatcache(); + if (is_file(static::$lockFile)) { + unlink(static::$lockFile); + } + } + + /** + * Whether monitor is paused + * @return bool + */ + public static function isPaused(): bool + { + clearstatcache(); + return file_exists(static::$lockFile); + } + + /** + * FileMonitor constructor. + * @param $monitor_dir + * @param $monitor_extensions + * @param array $options + */ + public function __construct($monitor_dir, $monitor_extensions, array $options = []) + { + static::resume(); + $this->_paths = (array)$monitor_dir; + $this->_extensions = $monitor_extensions; + if (!Worker::getAllWorkers()) { + return; + } + $disable_functions = explode(',', ini_get('disable_functions')); + if (in_array('exec', $disable_functions, true)) { + echo "\nMonitor file change turned off because exec() has been disabled by disable_functions setting in " . PHP_CONFIG_FILE_PATH . "/php.ini\n"; + } else { + if ($options['enable_file_monitor'] ?? true) { + Timer::add(1, function () { + $this->checkAllFilesChange(); + }); + } + } + + $memory_limit = $this->getMemoryLimit($options['memory_limit'] ?? null); + if ($options['enable_memory_monitor'] ?? $memory_limit) { + Timer::add(60, [$this, 'checkMemory'], [$memory_limit]); + } + } + + /** + * @param $monitor_dir + * @return bool + */ + public function checkFilesChange($monitor_dir): bool + { + static $last_mtime, $too_many_files_check; + if (!$last_mtime) { + $last_mtime = time(); + } + clearstatcache(); + if (!is_dir($monitor_dir)) { + if (!is_file($monitor_dir)) { + return false; + } + $iterator = [new \SplFileInfo($monitor_dir)]; + } else { + // recursive traversal directory + $dir_iterator = new \RecursiveDirectoryIterator($monitor_dir, \FilesystemIterator::SKIP_DOTS | \FilesystemIterator::FOLLOW_SYMLINKS); + $iterator = new \RecursiveIteratorIterator($dir_iterator); + } + $count = 0; + foreach ($iterator as $file) { + $count ++; + /** var SplFileInfo $file */ + if (is_dir($file->getRealPath())) { + continue; + } + // check mtime + if ($last_mtime < $file->getMTime() && in_array($file->getExtension(), $this->_extensions, true)) { + $var = 0; + exec('"'.PHP_BINARY . '" -l ' . $file, $out, $var); + if ($var) { + $last_mtime = $file->getMTime(); + continue; + } + $last_mtime = $file->getMTime(); + echo $file . " update and reload\n"; + // send SIGUSR1 signal to master process for reload + if (DIRECTORY_SEPARATOR === '/') { + posix_kill(posix_getppid(), SIGUSR1); + } else { + return true; + } + break; + } + } + if (!$too_many_files_check && $count > 1000) { + echo "Monitor: There are too many files ($count files) in $monitor_dir which makes file monitoring very slow\n"; + $too_many_files_check = 1; + } + return false; + } + + /** + * @return bool + */ + public function checkAllFilesChange(): bool + { + if (static::isPaused()) { + return false; + } + foreach ($this->_paths as $path) { + if ($this->checkFilesChange($path)) { + return true; + } + } + return false; + } + + /** + * @param $memory_limit + * @return void + */ + public function checkMemory($memory_limit) + { + if (static::isPaused()) { + return; + } + $ppid = posix_getppid(); + $children_file = "/proc/$ppid/task/$ppid/children"; + if (!is_file($children_file) || !($children = file_get_contents($children_file))) { + return; + } + foreach (explode(' ', $children) as $pid) { + $pid = (int)$pid; + $status_file = "/proc/$pid/status"; + if (!is_file($status_file) || !($status = file_get_contents($status_file))) { + continue; + } + $mem = 0; + if (preg_match('/VmRSS\s*?:\s*?(\d+?)\s*?kB/', $status, $match)) { + $mem = $match[1]; + } + $mem = (int)($mem / 1024); + if ($mem >= $memory_limit) { + posix_kill($pid, SIGINT); + } + } + } + + /** + * Get memory limit + * @return float + */ + protected function getMemoryLimit($memory_limit) + { + if ($memory_limit === 0) { + return 0; + } + $use_php_ini = false; + if (!$memory_limit) { + $memory_limit = ini_get('memory_limit'); + $use_php_ini = true; + } + + if ($memory_limit == -1) { + return 0; + } + $unit = $memory_limit[strlen($memory_limit) - 1]; + if ($unit == 'G') { + $memory_limit = 1024 * (int)$memory_limit; + } else if ($unit == 'M') { + $memory_limit = (int)$memory_limit; + } else if ($unit == 'K') { + $memory_limit = (int)($memory_limit / 1024); + } else { + $memory_limit = (int)($memory_limit / (1024 * 1024)); + } + if ($memory_limit < 30) { + $memory_limit = 30; + } + if ($use_php_ini) { + $memory_limit = (int)(0.8 * $memory_limit); + } + return $memory_limit; + } +} diff --git a/public/404.html b/public/404.html new file mode 100755 index 0000000..da9eace --- /dev/null +++ b/public/404.html @@ -0,0 +1,12 @@ + + + 404 Not Found - webman + + +
+

404 Not Found

+
+
+
webman
+ + diff --git a/public/favicon.ico b/public/favicon.ico new file mode 100755 index 0000000..b9f722e Binary files /dev/null and b/public/favicon.ico differ diff --git a/public/image/admin/ad01.jpg b/public/image/admin/ad01.jpg new file mode 100755 index 0000000..3ac345d Binary files /dev/null and b/public/image/admin/ad01.jpg differ diff --git a/public/image/admin/article01.png b/public/image/admin/article01.png new file mode 100755 index 0000000..ad176e7 Binary files /dev/null and b/public/image/admin/article01.png differ diff --git a/public/image/admin/article02.jpeg b/public/image/admin/article02.jpeg new file mode 100755 index 0000000..3cc6f49 Binary files /dev/null and b/public/image/admin/article02.jpeg differ diff --git a/public/image/admin/article03.png b/public/image/admin/article03.png new file mode 100755 index 0000000..3c62d41 Binary files /dev/null and b/public/image/admin/article03.png differ diff --git a/public/image/admin/avatar.png b/public/image/admin/avatar.png new file mode 100755 index 0000000..2051719 Binary files /dev/null and b/public/image/admin/avatar.png differ diff --git a/public/image/admin/backend_avatar.png b/public/image/admin/backend_avatar.png new file mode 100755 index 0000000..2051719 Binary files /dev/null and b/public/image/admin/backend_avatar.png differ diff --git a/public/image/admin/backend_backdrop.png b/public/image/admin/backend_backdrop.png new file mode 100755 index 0000000..a302011 Binary files /dev/null and b/public/image/admin/backend_backdrop.png differ diff --git a/public/image/admin/backend_favicon.ico b/public/image/admin/backend_favicon.ico new file mode 100755 index 0000000..7c00219 Binary files /dev/null and b/public/image/admin/backend_favicon.ico differ diff --git a/public/image/admin/backend_logo.png b/public/image/admin/backend_logo.png new file mode 100755 index 0000000..cb2f6d5 Binary files /dev/null and b/public/image/admin/backend_logo.png differ diff --git a/public/image/admin/banner01.png b/public/image/admin/banner01.png new file mode 100755 index 0000000..5894291 Binary files /dev/null and b/public/image/admin/banner01.png differ diff --git a/public/image/admin/banner02.png b/public/image/admin/banner02.png new file mode 100755 index 0000000..0c2d334 Binary files /dev/null and b/public/image/admin/banner02.png differ diff --git a/public/image/admin/customer_service.jpg b/public/image/admin/customer_service.jpg new file mode 100755 index 0000000..301009d Binary files /dev/null and b/public/image/admin/customer_service.jpg differ diff --git a/public/image/admin/default_avatar.png b/public/image/admin/default_avatar.png new file mode 100755 index 0000000..64d50e7 Binary files /dev/null and b/public/image/admin/default_avatar.png differ diff --git a/public/image/admin/login_image.png b/public/image/admin/login_image.png new file mode 100755 index 0000000..a302011 Binary files /dev/null and b/public/image/admin/login_image.png differ diff --git a/public/image/admin/menu_admin.png b/public/image/admin/menu_admin.png new file mode 100755 index 0000000..8cbb023 Binary files /dev/null and b/public/image/admin/menu_admin.png differ diff --git a/public/image/admin/menu_auth.png b/public/image/admin/menu_auth.png new file mode 100755 index 0000000..6dfe1f4 Binary files /dev/null and b/public/image/admin/menu_auth.png differ diff --git a/public/image/admin/menu_dept.png b/public/image/admin/menu_dept.png new file mode 100755 index 0000000..0d198e4 Binary files /dev/null and b/public/image/admin/menu_dept.png differ diff --git a/public/image/admin/menu_dict.png b/public/image/admin/menu_dict.png new file mode 100755 index 0000000..30ac7be Binary files /dev/null and b/public/image/admin/menu_dict.png differ diff --git a/public/image/admin/menu_file.png b/public/image/admin/menu_file.png new file mode 100755 index 0000000..8107b90 Binary files /dev/null and b/public/image/admin/menu_file.png differ diff --git a/public/image/admin/menu_generator.png b/public/image/admin/menu_generator.png new file mode 100755 index 0000000..eb023ad Binary files /dev/null and b/public/image/admin/menu_generator.png differ diff --git a/public/image/admin/menu_role.png b/public/image/admin/menu_role.png new file mode 100755 index 0000000..ba011d1 Binary files /dev/null and b/public/image/admin/menu_role.png differ diff --git a/public/image/admin/menu_web.png b/public/image/admin/menu_web.png new file mode 100755 index 0000000..2e3d5e8 Binary files /dev/null and b/public/image/admin/menu_web.png differ diff --git a/public/image/admin/nav01.png b/public/image/admin/nav01.png new file mode 100755 index 0000000..c198426 Binary files /dev/null and b/public/image/admin/nav01.png differ diff --git a/public/image/admin/nav02.png b/public/image/admin/nav02.png new file mode 100755 index 0000000..5c3f810 Binary files /dev/null and b/public/image/admin/nav02.png differ diff --git a/public/image/admin/nav03.png b/public/image/admin/nav03.png new file mode 100755 index 0000000..a9d0a21 Binary files /dev/null and b/public/image/admin/nav03.png differ diff --git a/public/image/admin/nav04.png b/public/image/admin/nav04.png new file mode 100755 index 0000000..6eab45b Binary files /dev/null and b/public/image/admin/nav04.png differ diff --git a/public/image/admin/nav05.png b/public/image/admin/nav05.png new file mode 100755 index 0000000..744b038 Binary files /dev/null and b/public/image/admin/nav05.png differ diff --git a/public/image/admin/pc_logo.png b/public/image/admin/pc_logo.png new file mode 100755 index 0000000..b08e70a Binary files /dev/null and b/public/image/admin/pc_logo.png differ diff --git a/public/image/admin/qq_group.png b/public/image/admin/qq_group.png new file mode 100755 index 0000000..2bc75c1 Binary files /dev/null and b/public/image/admin/qq_group.png differ diff --git a/public/image/admin/shop_logo.png b/public/image/admin/shop_logo.png new file mode 100755 index 0000000..58f7800 Binary files /dev/null and b/public/image/admin/shop_logo.png differ diff --git a/public/image/admin/tabbar_home.png b/public/image/admin/tabbar_home.png new file mode 100755 index 0000000..5c58118 Binary files /dev/null and b/public/image/admin/tabbar_home.png differ diff --git a/public/image/admin/tabbar_home_sel.png b/public/image/admin/tabbar_home_sel.png new file mode 100755 index 0000000..c0538a7 Binary files /dev/null and b/public/image/admin/tabbar_home_sel.png differ diff --git a/public/image/admin/tabbar_me.png b/public/image/admin/tabbar_me.png new file mode 100755 index 0000000..a6d01c4 Binary files /dev/null and b/public/image/admin/tabbar_me.png differ diff --git a/public/image/admin/tabbar_me_sel.png b/public/image/admin/tabbar_me_sel.png new file mode 100755 index 0000000..6994a49 Binary files /dev/null and b/public/image/admin/tabbar_me_sel.png differ diff --git a/public/image/admin/tabbar_text.png b/public/image/admin/tabbar_text.png new file mode 100755 index 0000000..5731d46 Binary files /dev/null and b/public/image/admin/tabbar_text.png differ diff --git a/public/image/admin/tabbar_text_sel.png b/public/image/admin/tabbar_text_sel.png new file mode 100755 index 0000000..6663475 Binary files /dev/null and b/public/image/admin/tabbar_text_sel.png differ diff --git a/public/image/admin/user_about.png b/public/image/admin/user_about.png new file mode 100755 index 0000000..733d570 Binary files /dev/null and b/public/image/admin/user_about.png differ diff --git a/public/image/admin/user_collect.png b/public/image/admin/user_collect.png new file mode 100755 index 0000000..c5484f4 Binary files /dev/null and b/public/image/admin/user_collect.png differ diff --git a/public/image/admin/user_kefu.png b/public/image/admin/user_kefu.png new file mode 100755 index 0000000..f8370fc Binary files /dev/null and b/public/image/admin/user_kefu.png differ diff --git a/public/image/admin/user_setting.png b/public/image/admin/user_setting.png new file mode 100755 index 0000000..c4bcf5a Binary files /dev/null and b/public/image/admin/user_setting.png differ diff --git a/public/image/admin/web_favicon.ico b/public/image/admin/web_favicon.ico new file mode 100755 index 0000000..2434b86 Binary files /dev/null and b/public/image/admin/web_favicon.ico differ diff --git a/public/image/admin/web_logo.jpg b/public/image/admin/web_logo.jpg new file mode 100755 index 0000000..f4073c9 Binary files /dev/null and b/public/image/admin/web_logo.jpg differ diff --git a/public/image/admin/web_logo.png b/public/image/admin/web_logo.png new file mode 100755 index 0000000..402e12e Binary files /dev/null and b/public/image/admin/web_logo.png differ diff --git a/public/uploads/images/20230228/20230228133711986810973.jpg b/public/uploads/images/20230228/20230228133711986810973.jpg new file mode 100644 index 0000000..fcf737e Binary files /dev/null and b/public/uploads/images/20230228/20230228133711986810973.jpg differ diff --git a/resource/translations/en/validate.php b/resource/translations/en/validate.php new file mode 100644 index 0000000..d3f26f3 --- /dev/null +++ b/resource/translations/en/validate.php @@ -0,0 +1,147 @@ + +// +---------------------------------------------------------------------- + +// 核心中文语言包 +return [ + // 系统错误提示 + 'Undefined variable' => '未定义变量', + 'Undefined index' => '未定义数组索引', + 'Undefined offset' => '未定义数组下标', + 'Parse error' => '语法解析错误', + 'Type error' => '类型错误', + 'Fatal error' => '致命错误', + 'syntax error' => '语法错误', + + // 框架核心错误提示 + 'dispatch type not support' => '不支持的调度类型', + 'method param miss' => '方法参数错误', + 'method not exists' => '方法不存在', + 'function not exists' => '函数不存在', + 'app not exists' => '应用不存在', + 'controller not exists' => '控制器不存在', + 'class not exists' => '类不存在', + 'property not exists' => '类的属性不存在', + 'template not exists' => '模板文件不存在', + 'illegal controller name' => '非法的控制器名称', + 'illegal action name' => '非法的操作名称', + 'url suffix deny' => '禁止的URL后缀访问', + 'Undefined cache config' => '缓存配置未定义', + 'Route Not Found' => '当前访问路由未定义或不匹配', + 'Undefined db config' => '数据库配置未定义', + 'Undefined log config' => '日志配置未定义', + 'Undefined db type' => '未定义数据库类型', + 'variable type error' => '变量类型错误', + 'PSR-4 error' => 'PSR-4 规范错误', + 'not support type' => '不支持的分页索引字段类型', + 'not support total' => '简洁模式下不能获取数据总数', + 'not support last' => '简洁模式下不能获取最后一页', + 'error session handler' => '错误的SESSION处理器类', + 'not allow php tag' => '模板不允许使用PHP语法', + 'not support' => '不支持', + 'database config error' => '数据库配置信息错误', + 'redisd master' => 'Redisd 主服务器错误', + 'redisd slave' => 'Redisd 从服务器错误', + 'must run at sae' => '必须在SAE运行', + 'memcache init error' => '未开通Memcache服务,请在SAE管理平台初始化Memcache服务', + 'KVDB init error' => '没有初始化KVDB,请在SAE管理平台初始化KVDB服务', + 'fields not exists' => '数据表字段不存在', + 'where express error' => '查询表达式错误', + 'no data to update' => '没有任何数据需要更新', + 'miss data to insert' => '缺少需要写入的数据', + 'miss complex primary data' => '缺少复合主键数据', + 'miss update condition' => '缺少更新条件', + 'model data Not Found' => '模型数据不存在', + 'table data not Found' => '表数据不存在', + 'delete without condition' => '没有条件不会执行删除操作', + 'miss relation data' => '缺少关联表数据', + 'tag attr must' => '模板标签属性必须', + 'tag error' => '模板标签错误', + 'cache write error' => '缓存写入失败', + 'sae mc write error' => 'SAE mc 写入错误', + 'route name not exists' => '路由标识不存在(或参数不够)', + 'invalid request' => '非法请求', + 'bind attr has exists' => '模型的属性已经存在', + 'relation data not exists' => '关联数据不存在', + 'relation not support' => '关联不支持', + 'chunk not support order' => 'Chunk不支持调用order方法', + 'route pattern error' => '路由变量规则定义错误', + 'route behavior will not support' => '路由行为废弃(使用中间件替代)', + 'closure not support cache(true)' => '使用闭包查询不支持cache(true),请指定缓存Key', + + // 上传错误信息 + 'unknown upload error' => '未知上传错误!', + 'file write error' => '文件写入失败!', + 'upload temp dir not found' => '找不到临时文件夹!', + 'no file to uploaded' => '没有文件被上传!', + 'only the portion of file is uploaded' => '文件只有部分被上传!', + 'upload File size exceeds the maximum value' => '上传文件大小超过了最大值!', + 'upload write error' => '文件上传保存错误!', + 'has the same filename: {:filename}' => '存在同名文件:{:filename}', + 'upload illegal files' => '非法上传文件', + 'illegal image files' => '非法图片文件', + 'extensions to upload is not allowed' => '上传文件后缀不允许', + 'mimetype to upload is not allowed' => '上传文件MIME类型不允许!', + 'filesize not match' => '上传文件大小不符!', + 'directory {:path} creation failed' => '目录 {:path} 创建失败!', + + 'The middleware must return Response instance' => '中间件方法必须返回Response对象实例', + 'The queue was exhausted, with no response returned' => '中间件队列为空', + // Validate Error Message + ':attribute require' => ':attribute不能为空', + ':attribute must' => ':attribute必须', + ':attribute must be numeric' => ':attribute必须是数字', + ':attribute must be integer' => ':attribute必须是整数', + ':attribute must be float' => ':attribute必须是浮点数', + ':attribute must be bool' => ':attribute必须是布尔值', + ':attribute not a valid email address' => ':attribute格式不符', + ':attribute not a valid mobile' => ':attribute格式不符', + ':attribute must be a array' => ':attribute必须是数组', + ':attribute must be yes,on or 1' => ':attribute必须是yes、on或者1', + ':attribute not a valid datetime' => ':attribute不是一个有效的日期或时间格式', + ':attribute not a valid file' => ':attribute不是有效的上传文件', + ':attribute not a valid image' => ':attribute不是有效的图像文件', + ':attribute must be alpha' => ':attribute只能是字母', + ':attribute must be alpha-numeric' => ':attribute只能是字母和数字', + ':attribute must be alpha-numeric, dash, underscore' => ':attribute只能是字母、数字和下划线_及破折号-', + ':attribute not a valid domain or ip' => ':attribute不是有效的域名或者IP', + ':attribute must be chinese' => ':attribute只能是汉字', + ':attribute must be chinese or alpha' => ':attribute只能是汉字、字母', + ':attribute must be chinese,alpha-numeric' => ':attribute只能是汉字、字母和数字', + ':attribute must be chinese,alpha-numeric,underscore, dash' => ':attribute只能是汉字、字母、数字和下划线_及破折号-', + ':attribute not a valid url' => ':attribute不是有效的URL地址', + ':attribute not a valid ip' => ':attribute不是有效的IP地址', + ':attribute must be dateFormat of :rule' => ':attribute必须使用日期格式 :rule', + ':attribute must be in :rule' => ':attribute必须在 :rule 范围内', + ':attribute be notin :rule' => ':attribute不能在 :rule 范围内', + ':attribute must between :1 - :2' => ':attribute只能在 :1 - :2 之间', + ':attribute not between :1 - :2' => ':attribute不能在 :1 - :2 之间', + 'size of :attribute must be :rule' => ':attribute长度不符合要求 :rule', + 'max size of :attribute must be :rule' => ':attribute长度不能超过 :rule', + 'min size of :attribute must be :rule' => ':attribute长度不能小于 :rule', + ':attribute cannot be less than :rule' => ':attribute日期不能小于 :rule', + ':attribute cannot exceed :rule' => ':attribute日期不能超过 :rule', + ':attribute not within :rule' => '不在有效期内 :rule', + 'access IP is not allowed' => '不允许的IP访问', + 'access IP denied' => '禁止的IP访问', + ':attribute out of accord with :2' => ':attribute和确认字段:2不一致', + ':attribute cannot be same with :2' => ':attribute和比较字段:2不能相同', + ':attribute must greater than or equal :rule' => ':attribute必须大于等于 :rule', + ':attribute must greater than :rule' => ':attribute必须大于 :rule', + ':attribute must less than or equal :rule' => ':attribute必须小于等于 :rule', + ':attribute must less than :rule' => ':attribute必须小于 :rule', + ':attribute must equal :rule' => ':attribute必须等于 :rule', + ':attribute has exists' => ':attribute已存在', + ':attribute not conform to the rules' => ':attribute不符合指定规则', + 'invalid Request method' => '无效的请求类型', + 'invalid token' => '令牌数据无效', + 'not conform to the rules' => '规则错误', + 'record has update' => '记录已经被更新了', +]; diff --git a/resource/translations/zh_CN/validate.php b/resource/translations/zh_CN/validate.php new file mode 100644 index 0000000..d3f26f3 --- /dev/null +++ b/resource/translations/zh_CN/validate.php @@ -0,0 +1,147 @@ + +// +---------------------------------------------------------------------- + +// 核心中文语言包 +return [ + // 系统错误提示 + 'Undefined variable' => '未定义变量', + 'Undefined index' => '未定义数组索引', + 'Undefined offset' => '未定义数组下标', + 'Parse error' => '语法解析错误', + 'Type error' => '类型错误', + 'Fatal error' => '致命错误', + 'syntax error' => '语法错误', + + // 框架核心错误提示 + 'dispatch type not support' => '不支持的调度类型', + 'method param miss' => '方法参数错误', + 'method not exists' => '方法不存在', + 'function not exists' => '函数不存在', + 'app not exists' => '应用不存在', + 'controller not exists' => '控制器不存在', + 'class not exists' => '类不存在', + 'property not exists' => '类的属性不存在', + 'template not exists' => '模板文件不存在', + 'illegal controller name' => '非法的控制器名称', + 'illegal action name' => '非法的操作名称', + 'url suffix deny' => '禁止的URL后缀访问', + 'Undefined cache config' => '缓存配置未定义', + 'Route Not Found' => '当前访问路由未定义或不匹配', + 'Undefined db config' => '数据库配置未定义', + 'Undefined log config' => '日志配置未定义', + 'Undefined db type' => '未定义数据库类型', + 'variable type error' => '变量类型错误', + 'PSR-4 error' => 'PSR-4 规范错误', + 'not support type' => '不支持的分页索引字段类型', + 'not support total' => '简洁模式下不能获取数据总数', + 'not support last' => '简洁模式下不能获取最后一页', + 'error session handler' => '错误的SESSION处理器类', + 'not allow php tag' => '模板不允许使用PHP语法', + 'not support' => '不支持', + 'database config error' => '数据库配置信息错误', + 'redisd master' => 'Redisd 主服务器错误', + 'redisd slave' => 'Redisd 从服务器错误', + 'must run at sae' => '必须在SAE运行', + 'memcache init error' => '未开通Memcache服务,请在SAE管理平台初始化Memcache服务', + 'KVDB init error' => '没有初始化KVDB,请在SAE管理平台初始化KVDB服务', + 'fields not exists' => '数据表字段不存在', + 'where express error' => '查询表达式错误', + 'no data to update' => '没有任何数据需要更新', + 'miss data to insert' => '缺少需要写入的数据', + 'miss complex primary data' => '缺少复合主键数据', + 'miss update condition' => '缺少更新条件', + 'model data Not Found' => '模型数据不存在', + 'table data not Found' => '表数据不存在', + 'delete without condition' => '没有条件不会执行删除操作', + 'miss relation data' => '缺少关联表数据', + 'tag attr must' => '模板标签属性必须', + 'tag error' => '模板标签错误', + 'cache write error' => '缓存写入失败', + 'sae mc write error' => 'SAE mc 写入错误', + 'route name not exists' => '路由标识不存在(或参数不够)', + 'invalid request' => '非法请求', + 'bind attr has exists' => '模型的属性已经存在', + 'relation data not exists' => '关联数据不存在', + 'relation not support' => '关联不支持', + 'chunk not support order' => 'Chunk不支持调用order方法', + 'route pattern error' => '路由变量规则定义错误', + 'route behavior will not support' => '路由行为废弃(使用中间件替代)', + 'closure not support cache(true)' => '使用闭包查询不支持cache(true),请指定缓存Key', + + // 上传错误信息 + 'unknown upload error' => '未知上传错误!', + 'file write error' => '文件写入失败!', + 'upload temp dir not found' => '找不到临时文件夹!', + 'no file to uploaded' => '没有文件被上传!', + 'only the portion of file is uploaded' => '文件只有部分被上传!', + 'upload File size exceeds the maximum value' => '上传文件大小超过了最大值!', + 'upload write error' => '文件上传保存错误!', + 'has the same filename: {:filename}' => '存在同名文件:{:filename}', + 'upload illegal files' => '非法上传文件', + 'illegal image files' => '非法图片文件', + 'extensions to upload is not allowed' => '上传文件后缀不允许', + 'mimetype to upload is not allowed' => '上传文件MIME类型不允许!', + 'filesize not match' => '上传文件大小不符!', + 'directory {:path} creation failed' => '目录 {:path} 创建失败!', + + 'The middleware must return Response instance' => '中间件方法必须返回Response对象实例', + 'The queue was exhausted, with no response returned' => '中间件队列为空', + // Validate Error Message + ':attribute require' => ':attribute不能为空', + ':attribute must' => ':attribute必须', + ':attribute must be numeric' => ':attribute必须是数字', + ':attribute must be integer' => ':attribute必须是整数', + ':attribute must be float' => ':attribute必须是浮点数', + ':attribute must be bool' => ':attribute必须是布尔值', + ':attribute not a valid email address' => ':attribute格式不符', + ':attribute not a valid mobile' => ':attribute格式不符', + ':attribute must be a array' => ':attribute必须是数组', + ':attribute must be yes,on or 1' => ':attribute必须是yes、on或者1', + ':attribute not a valid datetime' => ':attribute不是一个有效的日期或时间格式', + ':attribute not a valid file' => ':attribute不是有效的上传文件', + ':attribute not a valid image' => ':attribute不是有效的图像文件', + ':attribute must be alpha' => ':attribute只能是字母', + ':attribute must be alpha-numeric' => ':attribute只能是字母和数字', + ':attribute must be alpha-numeric, dash, underscore' => ':attribute只能是字母、数字和下划线_及破折号-', + ':attribute not a valid domain or ip' => ':attribute不是有效的域名或者IP', + ':attribute must be chinese' => ':attribute只能是汉字', + ':attribute must be chinese or alpha' => ':attribute只能是汉字、字母', + ':attribute must be chinese,alpha-numeric' => ':attribute只能是汉字、字母和数字', + ':attribute must be chinese,alpha-numeric,underscore, dash' => ':attribute只能是汉字、字母、数字和下划线_及破折号-', + ':attribute not a valid url' => ':attribute不是有效的URL地址', + ':attribute not a valid ip' => ':attribute不是有效的IP地址', + ':attribute must be dateFormat of :rule' => ':attribute必须使用日期格式 :rule', + ':attribute must be in :rule' => ':attribute必须在 :rule 范围内', + ':attribute be notin :rule' => ':attribute不能在 :rule 范围内', + ':attribute must between :1 - :2' => ':attribute只能在 :1 - :2 之间', + ':attribute not between :1 - :2' => ':attribute不能在 :1 - :2 之间', + 'size of :attribute must be :rule' => ':attribute长度不符合要求 :rule', + 'max size of :attribute must be :rule' => ':attribute长度不能超过 :rule', + 'min size of :attribute must be :rule' => ':attribute长度不能小于 :rule', + ':attribute cannot be less than :rule' => ':attribute日期不能小于 :rule', + ':attribute cannot exceed :rule' => ':attribute日期不能超过 :rule', + ':attribute not within :rule' => '不在有效期内 :rule', + 'access IP is not allowed' => '不允许的IP访问', + 'access IP denied' => '禁止的IP访问', + ':attribute out of accord with :2' => ':attribute和确认字段:2不一致', + ':attribute cannot be same with :2' => ':attribute和比较字段:2不能相同', + ':attribute must greater than or equal :rule' => ':attribute必须大于等于 :rule', + ':attribute must greater than :rule' => ':attribute必须大于 :rule', + ':attribute must less than or equal :rule' => ':attribute必须小于等于 :rule', + ':attribute must less than :rule' => ':attribute必须小于 :rule', + ':attribute must equal :rule' => ':attribute必须等于 :rule', + ':attribute has exists' => ':attribute已存在', + ':attribute not conform to the rules' => ':attribute不符合指定规则', + 'invalid Request method' => '无效的请求类型', + 'invalid token' => '令牌数据无效', + 'not conform to the rules' => '规则错误', + 'record has update' => '记录已经被更新了', +]; diff --git a/runtime/.gitignore b/runtime/.gitignore new file mode 100644 index 0000000..d337465 --- /dev/null +++ b/runtime/.gitignore @@ -0,0 +1,4 @@ +* +!logs +!views +!.gitignore diff --git a/runtime/logs/.gitignore b/runtime/logs/.gitignore new file mode 100644 index 0000000..c96a04f --- /dev/null +++ b/runtime/logs/.gitignore @@ -0,0 +1,2 @@ +* +!.gitignore \ No newline at end of file diff --git a/runtime/views/.gitignore b/runtime/views/.gitignore new file mode 100644 index 0000000..d6b7ef3 --- /dev/null +++ b/runtime/views/.gitignore @@ -0,0 +1,2 @@ +* +!.gitignore diff --git a/start.php b/start.php new file mode 100755 index 0000000..489e447 --- /dev/null +++ b/start.php @@ -0,0 +1,4 @@ +#!/usr/bin/env php + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +namespace support; + +/** + * Class Request + * @package support + */ +class Request extends \Webman\Http\Request +{ + +} \ No newline at end of file diff --git a/support/Response.php b/support/Response.php new file mode 100644 index 0000000..9bc4e1e --- /dev/null +++ b/support/Response.php @@ -0,0 +1,24 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +namespace support; + +/** + * Class Response + * @package support + */ +class Response extends \Webman\Http\Response +{ + +} \ No newline at end of file diff --git a/support/bootstrap.php b/support/bootstrap.php new file mode 100644 index 0000000..d9471e6 --- /dev/null +++ b/support/bootstrap.php @@ -0,0 +1,133 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use Dotenv\Dotenv; +use support\Log; +use Webman\Bootstrap; +use Webman\Config; +use Webman\Middleware; +use Webman\Route; +use Webman\Util; + +$worker = $worker ?? null; + +set_error_handler(function ($level, $message, $file = '', $line = 0) { + if (error_reporting() & $level) { + throw new ErrorException($message, 0, $level, $file, $line); + } +}); + +if ($worker) { + register_shutdown_function(function ($startTime) { + if (time() - $startTime <= 0.1) { + sleep(1); + } + }, time()); +} + +if (class_exists('Dotenv\Dotenv') && file_exists(base_path(false) . '/.env')) { + if (method_exists('Dotenv\Dotenv', 'createUnsafeMutable')) { + Dotenv::createUnsafeMutable(base_path(false))->load(); + } else { + Dotenv::createMutable(base_path(false))->load(); + } +} + +Config::clear(); +support\App::loadAllConfig(['route']); +if ($timezone = config('app.default_timezone')) { + date_default_timezone_set($timezone); +} + +foreach (config('autoload.files', []) as $file) { + include_once $file; +} +foreach (config('plugin', []) as $firm => $projects) { + foreach ($projects as $name => $project) { + if (!is_array($project)) { + continue; + } + foreach ($project['autoload']['files'] ?? [] as $file) { + include_once $file; + } + } + foreach ($projects['autoload']['files'] ?? [] as $file) { + include_once $file; + } +} + +Middleware::load(config('middleware', [])); +foreach (config('plugin', []) as $firm => $projects) { + foreach ($projects as $name => $project) { + if (!is_array($project) || $name === 'static') { + continue; + } + Middleware::load($project['middleware'] ?? []); + } + Middleware::load($projects['middleware'] ?? [], $firm); + if ($staticMiddlewares = config("plugin.$firm.static.middleware")) { + Middleware::load(['__static__' => $staticMiddlewares], $firm); + } +} +Middleware::load(['__static__' => config('static.middleware', [])]); + +foreach (config('bootstrap', []) as $className) { + if (!class_exists($className)) { + $log = "Warning: Class $className setting in config/bootstrap.php not found\r\n"; + echo $log; + Log::error($log); + continue; + } + /** @var Bootstrap $className */ + $className::start($worker); +} + +foreach (config('plugin', []) as $firm => $projects) { + foreach ($projects as $name => $project) { + if (!is_array($project)) { + continue; + } + foreach ($project['bootstrap'] ?? [] as $className) { + if (!class_exists($className)) { + $log = "Warning: Class $className setting in config/plugin/$firm/$name/bootstrap.php not found\r\n"; + echo $log; + Log::error($log); + continue; + } + /** @var Bootstrap $className */ + $className::start($worker); + } + } + foreach ($projects['bootstrap'] ?? [] as $className) { + /** @var string $className */ + if (!class_exists($className)) { + $log = "Warning: Class $className setting in plugin/$firm/config/bootstrap.php not found\r\n"; + echo $log; + Log::error($log); + continue; + } + /** @var Bootstrap $className */ + $className::start($worker); + } +} + +$directory = base_path() . '/plugin'; +$paths = [config_path()]; +foreach (Util::scanDir($directory) as $path) { + if (is_dir($path = "$path/config")) { + $paths[] = $path; + } +} +Route::load($paths); + diff --git a/support/helpers.php b/support/helpers.php new file mode 100644 index 0000000..f6d25b5 --- /dev/null +++ b/support/helpers.php @@ -0,0 +1,516 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +use support\Container; +use support\Request; +use support\Response; +use support\Translation; +use support\view\Blade; +use support\view\Raw; +use support\view\ThinkPHP; +use support\view\Twig; +use Twig\Error\LoaderError; +use Twig\Error\RuntimeError; +use Twig\Error\SyntaxError; +use Webman\App; +use Webman\Config; +use Webman\Route; +use Workerman\Worker; + +// Project base path +define('BASE_PATH', dirname(__DIR__)); + +/** + * return the program execute directory + * @param string $path + * @return string + */ +function run_path(string $path = ''): string +{ + static $runPath = ''; + if (!$runPath) { + $runPath = is_phar() ? dirname(Phar::running(false)) : BASE_PATH; + } + return path_combine($runPath, $path); +} + +/** + * if the param $path equal false,will return this program current execute directory + * @param string|false $path + * @return string + */ +function base_path($path = ''): string +{ + if (false === $path) { + return run_path(); + } + return path_combine(BASE_PATH, $path); +} + +/** + * App path + * @param string $path + * @return string + */ +function app_path(string $path = ''): string +{ + return path_combine(BASE_PATH . DIRECTORY_SEPARATOR . 'app', $path); +} + +/** + * Public path + * @param string $path + * @return string + */ +function public_path(string $path = ''): string +{ + static $publicPath = ''; + if (!$publicPath) { + $publicPath = \config('app.public_path') ?: run_path('public'); + } + return path_combine($publicPath, $path); +} + +/** + * Config path + * @param string $path + * @return string + */ +function config_path(string $path = ''): string +{ + return path_combine(BASE_PATH . DIRECTORY_SEPARATOR . 'config', $path); +} + +/** + * Runtime path + * @param string $path + * @return string + */ +function runtime_path(string $path = ''): string +{ + static $runtimePath = ''; + if (!$runtimePath) { + $runtimePath = \config('app.runtime_path') ?: run_path('runtime'); + } + return path_combine($runtimePath, $path); +} + +/** + * Generate paths based on given information + * @param string $front + * @param string $back + * @return string + */ +function path_combine(string $front, string $back): string +{ + return $front . ($back ? (DIRECTORY_SEPARATOR . ltrim($back, DIRECTORY_SEPARATOR)) : $back); +} + +/** + * Response + * @param int $status + * @param array $headers + * @param string $body + * @return Response + */ +function response(string $body = '', int $status = 200, array $headers = []): Response +{ + return new Response($status, $headers, $body); +} + +/** + * Json response + * @param $data + * @param int $options + * @return Response + */ +function json($data, int $options = JSON_UNESCAPED_UNICODE): Response +{ + return new Response(200, ['Content-Type' => 'application/json'], json_encode($data, $options)); +} + +/** + * Xml response + * @param $xml + * @return Response + */ +function xml($xml): Response +{ + if ($xml instanceof SimpleXMLElement) { + $xml = $xml->asXML(); + } + return new Response(200, ['Content-Type' => 'text/xml'], $xml); +} + +/** + * Jsonp response + * @param $data + * @param string $callbackName + * @return Response + */ +function jsonp($data, string $callbackName = 'callback'): Response +{ + if (!is_scalar($data) && null !== $data) { + $data = json_encode($data); + } + return new Response(200, [], "$callbackName($data)"); +} + +/** + * Redirect response + * @param string $location + * @param int $status + * @param array $headers + * @return Response + */ +function redirect(string $location, int $status = 302, array $headers = []): Response +{ + $response = new Response($status, ['Location' => $location]); + if (!empty($headers)) { + $response->withHeaders($headers); + } + return $response; +} + +/** + * View response + * @param string $template + * @param array $vars + * @param string|null $app + * @param string|null $plugin + * @return Response + */ +function view(string $template, array $vars = [], string $app = null, string $plugin = null): Response +{ + $request = \request(); + $plugin = $plugin === null ? ($request->plugin ?? '') : $plugin; + $handler = \config($plugin ? "plugin.$plugin.view.handler" : 'view.handler'); + return new Response(200, [], $handler::render($template, $vars, $app, $plugin)); +} + +/** + * Raw view response + * @param string $template + * @param array $vars + * @param string|null $app + * @return Response + * @throws Throwable + */ +function raw_view(string $template, array $vars = [], string $app = null): Response +{ + return new Response(200, [], Raw::render($template, $vars, $app)); +} + +/** + * Blade view response + * @param string $template + * @param array $vars + * @param string|null $app + * @return Response + */ +function blade_view(string $template, array $vars = [], string $app = null): Response +{ + return new Response(200, [], Blade::render($template, $vars, $app)); +} + +/** + * Think view response + * @param string $template + * @param array $vars + * @param string|null $app + * @return Response + */ +function think_view(string $template, array $vars = [], string $app = null): Response +{ + return new Response(200, [], ThinkPHP::render($template, $vars, $app)); +} + +/** + * Twig view response + * @param string $template + * @param array $vars + * @param string|null $app + * @return Response + * @throws LoaderError + * @throws RuntimeError + * @throws SyntaxError + */ +function twig_view(string $template, array $vars = [], string $app = null): Response +{ + return new Response(200, [], Twig::render($template, $vars, $app)); +} + +/** + * Get request + * @return \Webman\Http\Request|Request|null + */ +function request() +{ + return App::request(); +} + +/** + * Get config + * @param string|null $key + * @param $default + * @return array|mixed|null + */ +function config(string $key = null, $default = null) +{ + return Config::get($key, $default); +} + +/** + * Create url + * @param string $name + * @param ...$parameters + * @return string + */ +function route(string $name, ...$parameters): string +{ + $route = Route::getByName($name); + if (!$route) { + return ''; + } + + if (!$parameters) { + return $route->url(); + } + + if (is_array(current($parameters))) { + $parameters = current($parameters); + } + + return $route->url($parameters); +} + +/** + * Session + * @param mixed $key + * @param mixed $default + * @return mixed + */ +function session($key = null, $default = null) +{ + $session = \request()->session(); + if (null === $key) { + return $session; + } + if (is_array($key)) { + $session->put($key); + return null; + } + if (strpos($key, '.')) { + $keyArray = explode('.', $key); + $value = $session->all(); + foreach ($keyArray as $index) { + if (!isset($value[$index])) { + return $default; + } + $value = $value[$index]; + } + return $value; + } + return $session->get($key, $default); +} + +/** + * Translation + * @param string $id + * @param array $parameters + * @param string|null $domain + * @param string|null $locale + * @return string + */ +function trans(string $id, array $parameters = [], string $domain = null, string $locale = null): string +{ + $res = Translation::trans($id, $parameters, $domain, $locale); + return $res === '' ? $id : $res; +} + +/** + * Locale + * @param string|null $locale + * @return string + */ +function locale(string $locale = null): string +{ + if (!$locale) { + return Translation::getLocale(); + } + Translation::setLocale($locale); + return $locale; +} + +/** + * 404 not found + * @return Response + */ +function not_found(): Response +{ + return new Response(404, [], file_get_contents(public_path() . '/404.html')); +} + +/** + * Copy dir + * @param string $source + * @param string $dest + * @param bool $overwrite + * @return void + */ +function copy_dir(string $source, string $dest, bool $overwrite = false) +{ + if (is_dir($source)) { + if (!is_dir($dest)) { + mkdir($dest); + } + $files = scandir($source); + foreach ($files as $file) { + if ($file !== "." && $file !== "..") { + copy_dir("$source/$file", "$dest/$file"); + } + } + } else if (file_exists($source) && ($overwrite || !file_exists($dest))) { + copy($source, $dest); + } +} + +/** + * Remove dir + * @param string $dir + * @return bool + */ +function remove_dir(string $dir): bool +{ + if (is_link($dir) || is_file($dir)) { + return unlink($dir); + } + $files = array_diff(scandir($dir), array('.', '..')); + foreach ($files as $file) { + (is_dir("$dir/$file") && !is_link($dir)) ? remove_dir("$dir/$file") : unlink("$dir/$file"); + } + return rmdir($dir); +} + +/** + * Bind worker + * @param $worker + * @param $class + */ +function worker_bind($worker, $class) +{ + $callbackMap = [ + 'onConnect', + 'onMessage', + 'onClose', + 'onError', + 'onBufferFull', + 'onBufferDrain', + 'onWorkerStop', + 'onWebSocketConnect', + 'onWorkerReload' + ]; + foreach ($callbackMap as $name) { + if (method_exists($class, $name)) { + $worker->$name = [$class, $name]; + } + } + if (method_exists($class, 'onWorkerStart')) { + call_user_func([$class, 'onWorkerStart'], $worker); + } +} + +/** + * Start worker + * @param $processName + * @param $config + * @return void + */ +function worker_start($processName, $config) +{ + $worker = new Worker($config['listen'] ?? null, $config['context'] ?? []); + $propertyMap = [ + 'count', + 'user', + 'group', + 'reloadable', + 'reusePort', + 'transport', + 'protocol', + ]; + $worker->name = $processName; + foreach ($propertyMap as $property) { + if (isset($config[$property])) { + $worker->$property = $config[$property]; + } + } + + $worker->onWorkerStart = function ($worker) use ($config) { + require_once base_path('/support/bootstrap.php'); + if (isset($config['handler'])) { + if (!class_exists($config['handler'])) { + echo "process error: class {$config['handler']} not exists\r\n"; + return; + } + + $instance = Container::make($config['handler'], $config['constructor'] ?? []); + worker_bind($worker, $instance); + } + }; +} + +/** + * Get realpath + * @param string $filePath + * @return string + */ +function get_realpath(string $filePath): string +{ + if (strpos($filePath, 'phar://') === 0) { + return $filePath; + } else { + return realpath($filePath); + } +} + +/** + * Is phar + * @return bool + */ +function is_phar(): bool +{ + return class_exists(Phar::class, false) && Phar::running(); +} + +/** + * Get cpu count + * @return int + */ +function cpu_count(): int +{ + // Windows does not support the number of processes setting. + if (DIRECTORY_SEPARATOR === '\\') { + return 1; + } + $count = 4; + if (is_callable('shell_exec')) { + if (strtolower(PHP_OS) === 'darwin') { + $count = (int)shell_exec('sysctl -n machdep.cpu.core_count'); + } else { + $count = (int)shell_exec('nproc'); + } + } + return $count > 0 ? $count : 4; +} diff --git a/webman b/webman new file mode 100755 index 0000000..00520d1 --- /dev/null +++ b/webman @@ -0,0 +1,45 @@ +#!/usr/bin/env php +setName('webman cli'); +$cli->installInternalCommands(); +if (is_dir($command_path = Util::guessPath(app_path(), '/command', true))) { + $cli->installCommands($command_path); +} + +foreach (config('plugin', []) as $firm => $projects) { + if (isset($projects['app'])) { + if ($command_str = Util::guessPath(base_path() . "/plugin/$firm", 'command')) { + $command_path = base_path() . "/plugin/$firm/$command_str"; + $cli->installCommands($command_path, "plugin\\$firm\\$command_str"); + } + } + foreach ($projects as $name => $project) { + if (!is_array($project)) { + continue; + } + foreach ($project['command'] ?? [] as $command) { + $cli->add(Container::get($command)); + } + } +} + +$cli->run(); diff --git a/windows.bat b/windows.bat new file mode 100644 index 0000000..f07ce53 --- /dev/null +++ b/windows.bat @@ -0,0 +1,3 @@ +CHCP 65001 +php windows.php +pause \ No newline at end of file diff --git a/windows.php b/windows.php new file mode 100644 index 0000000..d1975d6 --- /dev/null +++ b/windows.php @@ -0,0 +1,116 @@ +load(); + } else { + Dotenv::createMutable(base_path())->load(); + } +} + +App::loadAllConfig(['route']); + +$errorReporting = config('app.error_reporting'); +if (isset($errorReporting)) { + error_reporting($errorReporting); +} + +$runtimeProcessPath = runtime_path() . DIRECTORY_SEPARATOR . '/windows'; +if (!is_dir($runtimeProcessPath)) { + mkdir($runtimeProcessPath); +} +$processFiles = [ + __DIR__ . DIRECTORY_SEPARATOR . 'start.php' +]; +foreach (config('process', []) as $processName => $config) { + $processFiles[] = write_process_file($runtimeProcessPath, $processName, ''); +} + +foreach (config('plugin', []) as $firm => $projects) { + foreach ($projects as $name => $project) { + if (!is_array($project)) { + continue; + } + foreach ($project['process'] ?? [] as $processName => $config) { + $processFiles[] = write_process_file($runtimeProcessPath, $processName, "$firm.$name"); + } + } + foreach ($projects['process'] ?? [] as $processName => $config) { + $processFiles[] = write_process_file($runtimeProcessPath, $processName, $firm); + } +} + +function write_process_file($runtimeProcessPath, $processName, $firm): string +{ + $processParam = $firm ? "plugin.$firm.$processName" : $processName; + $configParam = $firm ? "config('plugin.$firm.process')['$processName']" : "config('process')['$processName']"; + $fileContent = << true]); + if (!$resource) { + exit("Can not execute $cmd\r\n"); + } + return $resource; +} + +$resource = popen_processes($processFiles); +echo "\r\n"; +while (1) { + sleep(1); + if (!empty($monitor) && $monitor->checkAllFilesChange()) { + $status = proc_get_status($resource); + $pid = $status['pid']; + shell_exec("taskkill /F /T /PID $pid"); + proc_close($resource); + $resource = popen_processes($processFiles); + } +}