commit a1037a21b51e14b73d45f962660483ae3cab49c4 Author: 醉挽清风 <1550969027@qq.com> Date: Tue Feb 28 15:13:10 2023 +0800 first 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..61fda03 --- /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..170e9e1 --- /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 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..0b9a3e4 --- /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..ae669c3 --- /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..5a4c833 --- /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..0cef20f --- /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..0918aee --- /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..7b8e6a1 --- /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..5e6cc89 --- /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..bc716d8 --- /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..e59f796 --- /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..33a66f0 --- /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..8614e03 --- /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..486a5fd --- /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..9b5d965 --- /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..8ade0c0 --- /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..aed2648 --- /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..7995525 --- /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..936c06a --- /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..56c2354 --- /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..ea99bed --- /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..8ef03a1 --- /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..660b4a6 --- /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/user/UserController.php b/app/admin/controller/user/UserController.php new file mode 100644 index 0000000..dac6599 --- /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..d661db1 --- /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..756eb08 --- /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..da54d52 --- /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..f491e0a --- /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..d3a5bd6 --- /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..4d4af82 --- /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..3ca9430 --- /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..1d28a87 --- /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/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..1321de8 --- /dev/null +++ b/app/admin/logic/LoginLogic.php @@ -0,0 +1,84 @@ +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..103f538 --- /dev/null +++ b/app/admin/logic/WorkbenchLogic.php @@ -0,0 +1,208 @@ + 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_dict')), + 'url' => '/dev_tools/dict' + ], + [ + 'name' => '代码生成器', + 'image' => FileService::getFileUrl(config('project.default_image.menu_generator')), + 'url' => '/dev_tools/code' + ], + [ + '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://www.likeadmin.cn', + 'gitee' => 'https://gitee.com/likeadmin/likeadmin_php', + ] + ]; + } + + + /** + * @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..f426f07 --- /dev/null +++ b/app/admin/logic/auth/AdminLogic.php @@ -0,0 +1,337 @@ + $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'])) { + $passwordSalt = Config::get('project.unique_identification'); + $data['password'] = create_password($params['password'], $passwordSalt); + } + + // 禁用或更换角色后.设置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'])) { + $passwordSalt = Config::get('project.unique_identification'); + $data['password'] = create_password($params['password'], $passwordSalt); + } + + 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..8fc26e6 --- /dev/null +++ b/app/admin/logic/setting/system/SystemLogic.php @@ -0,0 +1,65 @@ + '服务器操作系统', 'value' => PHP_OS], + ['param' => 'web服务器环境', 'value' => $_SERVER['SERVER_SOFTWARE']], + ['param' => 'PHP版本', 'value' => PHP_VERSION], + ]; + + $env = [ + [ 'option' => 'PHP版本', + 'require' => '8.0版本以上', + 'status' => (int)compare_php('8.0.0'), + '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..a0e43eb --- /dev/null +++ b/app/admin/logic/tools/GeneratorLogic.php @@ -0,0 +1,505 @@ +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 = app()->make(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 app()->make(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('curd_file_name' . $fileName); + if (empty($cacheFileName)) { + self::$error = '请重新生成代码'; + return false; + } + + $path = root_path() . 'runtime/generate/' . $fileName; + if (!file_exists($path)) { + self::$error = '下载失败'; + return false; + } + + cache('curd_file_name' . $fileName, null); + 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..bae9c1d --- /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..71366f3 --- /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..e92548f --- /dev/null +++ b/app/admin/validate/LoginValidate.php @@ -0,0 +1,86 @@ + '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..9987d0f --- /dev/null +++ b/app/admin/validate/auth/editSelfValidate.php @@ -0,0 +1,74 @@ + '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']); + $passwordSalt = Config::get('project.unique_identification'); + $oldPassword = create_password($data['password_old'], $passwordSalt); + + if ($admin['password'] != $oldPassword) { + 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..690f772 --- /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..ead810b --- /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..694f132 --- /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..7d6c919 --- /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..16e0534 --- /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..bb82bd5 --- /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/user/UserValidate.php b/app/admin/validate/user/UserValidate.php new file mode 100644 index 0000000..d995d7c --- /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/common/cache/AdminAccountSafeCache.php b/app/common/cache/AdminAccountSafeCache.php new file mode 100644 index 0000000..2b11aba --- /dev/null +++ b/app/common/cache/AdminAccountSafeCache.php @@ -0,0 +1,62 @@ +getLocalIp(); + $this->key = $this->tagName . $ip; + } + + /** + * @notes 记录登录错误次数 + * @author 令狐冲 + * @date 2021/6/30 01:51 + */ + public function record() + { + if (Cache::get($this->key)) { + //缓存存在,记录错误次数 + Cache::inc($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..74e3630 --- /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..434633c --- /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, new \DateTime(Date('Y-m-d H:i:s', $adminSession->expire_time))); + 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..5e06677 --- /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::store()->tag($this->tagName)->set($key, $value, $ttl); + } + + + /** + * @notes 清除缓存类所有缓存 + * @return bool + * @author 乔峰 + * @date 2021/12/27 14:16 + */ + public function deleteTag(): bool + { + return Cache::tag($this->tagName)->clear(); + } +} \ 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..8296edc --- /dev/null +++ b/app/common/cache/ExportCache.php @@ -0,0 +1,72 @@ +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..4ed303d --- /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::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缓存并返回下载地址 + $vars = ['file' => $exportCache->setFile($this->fileName)]; + return (string)(url('admin/download/export', $vars, true, true)); + } + /** + * @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 @@ +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..e933859 --- /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 = request()->domain(); + 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..12ea10f --- /dev/null +++ b/app/common/service/JsonService.php @@ -0,0 +1,117 @@ +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..c3ba5a9 --- /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/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..727dbad --- /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..b3fbd92 --- /dev/null +++ b/app/functions.php @@ -0,0 +1,96 @@ +=7.2", + "workerman/webman-framework": "^1.4.7", + "monolog/monolog": "^2.0", + "webman/think-orm": "^1.0", + "topthink/think-validate": "^2.0", + "vlucas/phpdotenv": "^5.4", + "psr/container": "^1.1.1", + "webman/think-cache": "^1.0", + "ext-json": "*", + "phpoffice/phpspreadsheet": "^1.19", + "aliyuncs/oss-sdk-php": "^2.6", + "tencentcloud/tencentcloud-sdk-php": "^3.0", + "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" + }, + "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..62f6a49 --- /dev/null +++ b/composer.lock @@ -0,0 +1,3939 @@ +{ + "_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": "64507c05c6f5e4cf1e21794755059863", + "packages": [ + { + "name": "aliyuncs/oss-sdk-php", + "version": "v2.6.0", + "source": { + "type": "git", + "url": "https://github.com/aliyun/aliyun-oss-php-sdk.git", + "reference": "572d0f8e099e8630ae7139ed3fdedb926c7a760f" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/aliyun/aliyun-oss-php-sdk/zipball/572d0f8e099e8630ae7139ed3fdedb926c7a760f", + "reference": "572d0f8e099e8630ae7139ed3fdedb926c7a760f", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=5.3" + }, + "require-dev": { + "phpunit/phpunit": "*", + "satooshi/php-coveralls": "*" + }, + "type": "library", + "autoload": { + "psr-4": { + "OSS\\": "src/OSS" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Aliyuncs", + "homepage": "http://www.aliyun.com" + } + ], + "description": "Aliyun OSS SDK for PHP", + "homepage": "http://www.aliyun.com/product/oss/", + "support": { + "issues": "https://github.com/aliyun/aliyun-oss-php-sdk/issues", + "source": "https://github.com/aliyun/aliyun-oss-php-sdk/tree/v2.6.0" + }, + "time": "2022-08-03T08:06:01+00:00" + }, + { + "name": "doctrine/inflector", + "version": "2.0.4", + "source": { + "type": "git", + "url": "https://github.com/doctrine/inflector.git", + "reference": "8b7ff3e4b7de6b2c84da85637b59fd2880ecaa89" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/doctrine/inflector/zipball/8b7ff3e4b7de6b2c84da85637b59fd2880ecaa89", + "reference": "8b7ff3e4b7de6b2c84da85637b59fd2880ecaa89", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": "^7.2 || ^8.0" + }, + "require-dev": { + "doctrine/coding-standard": "^8.2", + "phpstan/phpstan": "^0.12", + "phpstan/phpstan-phpunit": "^0.12", + "phpstan/phpstan-strict-rules": "^0.12", + "phpunit/phpunit": "^7.0 || ^8.0 || ^9.0", + "vimeo/psalm": "^4.10" + }, + "type": "library", + "autoload": { + "psr-4": { + "Doctrine\\Inflector\\": "lib/Doctrine/Inflector" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/doctrine/inflector/issues", + "source": "https://github.com/doctrine/inflector/tree/2.0.4" + }, + "funding": [ + { + "url": "https://www.doctrine-project.org/sponsorship.html", + "type": "custom" + }, + { + "url": "https://www.patreon.com/phpdoctrine", + "type": "patreon" + }, + { + "url": "https://tidelift.com/funding/github/packagist/doctrine%2Finflector", + "type": "tidelift" + } + ], + "time": "2021-10-22T20:16:43+00:00" + }, + { + "name": "dragonmantank/cron-expression", + "version": "v3.3.1", + "source": { + "type": "git", + "url": "https://github.com/dragonmantank/cron-expression.git", + "reference": "be85b3f05b46c39bbc0d95f6c071ddff669510fa" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/dragonmantank/cron-expression/zipball/be85b3f05b46c39bbc0d95f6c071ddff669510fa", + "reference": "be85b3f05b46c39bbc0d95f6c071ddff669510fa", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "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/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/dragonmantank/cron-expression/issues", + "source": "https://github.com/dragonmantank/cron-expression/tree/v3.3.1" + }, + "funding": [ + { + "url": "https://github.com/dragonmantank", + "type": "github" + } + ], + "time": "2022-01-18T15:43:28+00:00" + }, + { + "name": "ezyang/htmlpurifier", + "version": "v4.16.0", + "source": { + "type": "git", + "url": "https://github.com/ezyang/htmlpurifier.git", + "reference": "523407fb06eb9e5f3d59889b3978d5bfe94299c8" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/ezyang/htmlpurifier/zipball/523407fb06eb9e5f3d59889b3978d5bfe94299c8", + "reference": "523407fb06eb9e5f3d59889b3978d5bfe94299c8", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "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/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/ezyang/htmlpurifier/issues", + "source": "https://github.com/ezyang/htmlpurifier/tree/v4.16.0" + }, + "time": "2022-09-18T07:06:19+00:00" + }, + { + "name": "graham-campbell/result-type", + "version": "v1.0.4", + "source": { + "type": "git", + "url": "https://github.com/GrahamCampbell/Result-Type.git", + "reference": "0690bde05318336c7221785f2a932467f98b64ca" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/GrahamCampbell/Result-Type/zipball/0690bde05318336c7221785f2a932467f98b64ca", + "reference": "0690bde05318336c7221785f2a932467f98b64ca", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": "^7.0 || ^8.0", + "phpoption/phpoption": "^1.8" + }, + "require-dev": { + "phpunit/phpunit": "^6.5.14 || ^7.5.20 || ^8.5.19 || ^9.5.8" + }, + "type": "library", + "autoload": { + "psr-4": { + "GrahamCampbell\\ResultType\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/GrahamCampbell/Result-Type/issues", + "source": "https://github.com/GrahamCampbell/Result-Type/tree/v1.0.4" + }, + "funding": [ + { + "url": "https://github.com/GrahamCampbell", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/graham-campbell/result-type", + "type": "tidelift" + } + ], + "time": "2021-11-21T21:41:47+00:00" + }, + { + "name": "guzzlehttp/command", + "version": "1.2.2", + "source": { + "type": "git", + "url": "https://github.com/guzzle/command.git", + "reference": "7883359e0ecab8a8f7c43aad2fc36360a35d21e8" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/guzzle/command/zipball/7883359e0ecab8a8f7c43aad2fc36360a35d21e8", + "reference": "7883359e0ecab8a8f7c43aad2fc36360a35d21e8", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "guzzlehttp/guzzle": "^7.4.1", + "guzzlehttp/promises": "^1.5.1", + "guzzlehttp/psr7": "^1.8.3 || ^2.1", + "php": "^7.2.5 || ^8.0" + }, + "require-dev": { + "phpunit/phpunit": "^8.5.19" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.2-dev" + } + }, + "autoload": { + "psr-4": { + "GuzzleHttp\\Command\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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", + "support": { + "issues": "https://github.com/guzzle/command/issues", + "source": "https://github.com/guzzle/command/tree/1.2.2" + }, + "funding": [ + { + "url": "https://github.com/GrahamCampbell", + "type": "github" + }, + { + "url": "https://github.com/Nyholm", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/guzzlehttp/command", + "type": "tidelift" + } + ], + "time": "2022-02-08T10:21:14+00:00" + }, + { + "name": "guzzlehttp/guzzle", + "version": "7.5.0", + "source": { + "type": "git", + "url": "https://github.com/guzzle/guzzle.git", + "reference": "b50a2a1251152e43f6a37f0fa053e730a67d25ba" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/guzzle/guzzle/zipball/b50a2a1251152e43f6a37f0fa053e730a67d25ba", + "reference": "b50a2a1251152e43f6a37f0fa053e730a67d25ba", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "ext-json": "*", + "guzzlehttp/promises": "^1.5", + "guzzlehttp/psr7": "^1.9 || ^2.4", + "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": "^3.0", + "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 + }, + "branch-alias": { + "dev-master": "7.5-dev" + } + }, + "autoload": { + "files": [ + "src/functions_include.php" + ], + "psr-4": { + "GuzzleHttp\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/guzzle/guzzle/issues", + "source": "https://github.com/guzzle/guzzle/tree/7.5.0" + }, + "funding": [ + { + "url": "https://github.com/GrahamCampbell", + "type": "github" + }, + { + "url": "https://github.com/Nyholm", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/guzzlehttp/guzzle", + "type": "tidelift" + } + ], + "time": "2022-08-28T15:39:27+00:00" + }, + { + "name": "guzzlehttp/guzzle-services", + "version": "1.3.2", + "source": { + "type": "git", + "url": "https://github.com/guzzle/guzzle-services.git", + "reference": "4989d902dd4e0411b320e851c46f3c94d652d891" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/guzzle/guzzle-services/zipball/4989d902dd4e0411b320e851c46f3c94d652d891", + "reference": "4989d902dd4e0411b320e851c46f3c94d652d891", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "guzzlehttp/command": "^1.2.2", + "guzzlehttp/guzzle": "^7.4.1", + "guzzlehttp/psr7": "^1.8.3 || ^2.1", + "guzzlehttp/uri-template": "^1.0.1", + "php": "^7.2.5 || ^8.0" + }, + "require-dev": { + "phpunit/phpunit": "^8.5.19 || ^9.5.8" + }, + "suggest": { + "gimler/guzzle-description-loader": "^0.0.4" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.3-dev" + } + }, + "autoload": { + "psr-4": { + "GuzzleHttp\\Command\\Guzzle\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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.", + "support": { + "issues": "https://github.com/guzzle/guzzle-services/issues", + "source": "https://github.com/guzzle/guzzle-services/tree/1.3.2" + }, + "funding": [ + { + "url": "https://github.com/GrahamCampbell", + "type": "github" + }, + { + "url": "https://github.com/Nyholm", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/guzzlehttp/guzzle-services", + "type": "tidelift" + } + ], + "time": "2022-03-03T11:21:34+00:00" + }, + { + "name": "guzzlehttp/promises", + "version": "1.5.2", + "source": { + "type": "git", + "url": "https://github.com/guzzle/promises.git", + "reference": "b94b2807d85443f9719887892882d0329d1e2598" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/guzzle/promises/zipball/b94b2807d85443f9719887892882d0329d1e2598", + "reference": "b94b2807d85443f9719887892882d0329d1e2598", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=5.5" + }, + "require-dev": { + "symfony/phpunit-bridge": "^4.4 || ^5.1" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.5-dev" + } + }, + "autoload": { + "files": [ + "src/functions_include.php" + ], + "psr-4": { + "GuzzleHttp\\Promise\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/guzzle/promises/issues", + "source": "https://github.com/guzzle/promises/tree/1.5.2" + }, + "funding": [ + { + "url": "https://github.com/GrahamCampbell", + "type": "github" + }, + { + "url": "https://github.com/Nyholm", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/guzzlehttp/promises", + "type": "tidelift" + } + ], + "time": "2022-08-28T14:55:35+00:00" + }, + { + "name": "guzzlehttp/psr7", + "version": "2.4.3", + "source": { + "type": "git", + "url": "https://github.com/guzzle/psr7.git", + "reference": "67c26b443f348a51926030c83481b85718457d3d" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/guzzle/psr7/zipball/67c26b443f348a51926030c83481b85718457d3d", + "reference": "67c26b443f348a51926030c83481b85718457d3d", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": "^7.2.5 || ^8.0", + "psr/http-factory": "^1.0", + "psr/http-message": "^1.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 + }, + "branch-alias": { + "dev-master": "2.4-dev" + } + }, + "autoload": { + "psr-4": { + "GuzzleHttp\\Psr7\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/guzzle/psr7/issues", + "source": "https://github.com/guzzle/psr7/tree/2.4.3" + }, + "funding": [ + { + "url": "https://github.com/GrahamCampbell", + "type": "github" + }, + { + "url": "https://github.com/Nyholm", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/guzzlehttp/psr7", + "type": "tidelift" + } + ], + "time": "2022-10-26T14:07:24+00:00" + }, + { + "name": "guzzlehttp/uri-template", + "version": "v1.0.1", + "source": { + "type": "git", + "url": "https://github.com/guzzle/uri-template.git", + "reference": "b945d74a55a25a949158444f09ec0d3c120d69e2" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/guzzle/uri-template/zipball/b945d74a55a25a949158444f09ec0d3c120d69e2", + "reference": "b945d74a55a25a949158444f09ec0d3c120d69e2", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": "^7.2.5 || ^8.0", + "symfony/polyfill-php80": "^1.17" + }, + "require-dev": { + "phpunit/phpunit": "^8.5.19 || ^9.5.8", + "uri-template/tests": "1.0.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0-dev" + } + }, + "autoload": { + "psr-4": { + "GuzzleHttp\\UriTemplate\\": "src" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/guzzle/uri-template/issues", + "source": "https://github.com/guzzle/uri-template/tree/v1.0.1" + }, + "funding": [ + { + "url": "https://github.com/GrahamCampbell", + "type": "github" + }, + { + "url": "https://github.com/Nyholm", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/guzzlehttp/uri-template", + "type": "tidelift" + } + ], + "time": "2021-10-07T12:57:01+00:00" + }, + { + "name": "maennchen/zipstream-php", + "version": "2.1.0", + "source": { + "type": "git", + "url": "https://github.com/maennchen/ZipStream-PHP.git", + "reference": "c4c5803cc1f93df3d2448478ef79394a5981cc58" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/maennchen/ZipStream-PHP/zipball/c4c5803cc1f93df3d2448478ef79394a5981cc58", + "reference": "c4c5803cc1f93df3d2448478ef79394a5981cc58", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "myclabs/php-enum": "^1.5", + "php": ">= 7.1", + "psr/http-message": "^1.0", + "symfony/polyfill-mbstring": "^1.0" + }, + "require-dev": { + "ext-zip": "*", + "guzzlehttp/guzzle": ">= 6.3", + "mikey179/vfsstream": "^1.6", + "phpunit/phpunit": ">= 7.5" + }, + "type": "library", + "autoload": { + "psr-4": { + "ZipStream\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/maennchen/ZipStream-PHP/issues", + "source": "https://github.com/maennchen/ZipStream-PHP/tree/2.1.0" + }, + "funding": [ + { + "url": "https://github.com/maennchen", + "type": "github" + }, + { + "url": "https://opencollective.com/zipstream", + "type": "open_collective" + } + ], + "time": "2020-05-30T13:11:16+00:00" + }, + { + "name": "markbaker/complex", + "version": "3.0.2", + "source": { + "type": "git", + "url": "https://github.com/MarkBaker/PHPComplex.git", + "reference": "95c56caa1cf5c766ad6d65b6344b807c1e8405b9" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/MarkBaker/PHPComplex/zipball/95c56caa1cf5c766ad6d65b6344b807c1e8405b9", + "reference": "95c56caa1cf5c766ad6d65b6344b807c1e8405b9", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "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/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/MarkBaker/PHPComplex/issues", + "source": "https://github.com/MarkBaker/PHPComplex/tree/3.0.2" + }, + "time": "2022-12-06T16:21:08+00:00" + }, + { + "name": "markbaker/matrix", + "version": "3.0.1", + "source": { + "type": "git", + "url": "https://github.com/MarkBaker/PHPMatrix.git", + "reference": "728434227fe21be27ff6d86621a1b13107a2562c" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/MarkBaker/PHPMatrix/zipball/728434227fe21be27ff6d86621a1b13107a2562c", + "reference": "728434227fe21be27ff6d86621a1b13107a2562c", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "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/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/MarkBaker/PHPMatrix/issues", + "source": "https://github.com/MarkBaker/PHPMatrix/tree/3.0.1" + }, + "time": "2022-12-02T22:17:43+00:00" + }, + { + "name": "monolog/monolog", + "version": "2.8.0", + "source": { + "type": "git", + "url": "https://github.com/Seldaek/monolog.git", + "reference": "720488632c590286b88b80e62aa3d3d551ad4a50" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/Seldaek/monolog/zipball/720488632c590286b88b80e62aa3d3d551ad4a50", + "reference": "720488632c590286b88b80e62aa3d3d551ad4a50", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "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", + "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" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/Seldaek/monolog/issues", + "source": "https://github.com/Seldaek/monolog/tree/2.8.0" + }, + "funding": [ + { + "url": "https://github.com/Seldaek", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/monolog/monolog", + "type": "tidelift" + } + ], + "time": "2022-07-24T11:55:47+00:00" + }, + { + "name": "myclabs/php-enum", + "version": "1.7.7", + "source": { + "type": "git", + "url": "https://github.com/myclabs/php-enum.git", + "reference": "d178027d1e679832db9f38248fcc7200647dc2b7" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/myclabs/php-enum/zipball/d178027d1e679832db9f38248fcc7200647dc2b7", + "reference": "d178027d1e679832db9f38248fcc7200647dc2b7", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "ext-json": "*", + "php": ">=7.1" + }, + "require-dev": { + "phpunit/phpunit": "^7", + "squizlabs/php_codesniffer": "1.*", + "vimeo/psalm": "^3.8" + }, + "type": "library", + "autoload": { + "psr-4": { + "MyCLabs\\Enum\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP Enum contributors", + "homepage": "https://github.com/myclabs/php-enum/graphs/contributors" + } + ], + "description": "PHP Enum implementation", + "homepage": "http://github.com/myclabs/php-enum", + "keywords": [ + "enum" + ], + "support": { + "issues": "https://github.com/myclabs/php-enum/issues", + "source": "https://github.com/myclabs/php-enum/tree/1.7.7" + }, + "funding": [ + { + "url": "https://github.com/mnapoli", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/myclabs/php-enum", + "type": "tidelift" + } + ], + "time": "2020-11-14T18:14:52+00:00" + }, + { + "name": "nikic/fast-route", + "version": "v1.3.0", + "source": { + "type": "git", + "url": "https://github.com/nikic/FastRoute.git", + "reference": "181d480e08d9476e61381e04a71b34dc0432e812" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/nikic/FastRoute/zipball/181d480e08d9476e61381e04a71b34dc0432e812", + "reference": "181d480e08d9476e61381e04a71b34dc0432e812", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "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/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "BSD-3-Clause" + ], + "authors": [ + { + "name": "Nikita Popov", + "email": "nikic@php.net" + } + ], + "description": "Fast request router for PHP", + "keywords": [ + "router", + "routing" + ], + "support": { + "issues": "https://github.com/nikic/FastRoute/issues", + "source": "https://github.com/nikic/FastRoute/tree/master" + }, + "time": "2018-02-13T20:26:39+00:00" + }, + { + "name": "opis/closure", + "version": "3.6.3", + "source": { + "type": "git", + "url": "https://github.com/opis/closure.git", + "reference": "3d81e4309d2a927abbe66df935f4bb60082805ad" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/opis/closure/zipball/3d81e4309d2a927abbe66df935f4bb60082805ad", + "reference": "3d81e4309d2a927abbe66df935f4bb60082805ad", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": "^5.4 || ^7.0 || ^8.0" + }, + "require-dev": { + "jeremeamia/superclosure": "^2.0", + "phpunit/phpunit": "^4.0 || ^5.0 || ^6.0 || ^7.0 || ^8.0 || ^9.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "3.6.x-dev" + } + }, + "autoload": { + "files": [ + "functions.php" + ], + "psr-4": { + "Opis\\Closure\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Marius Sarca", + "email": "marius.sarca@gmail.com" + }, + { + "name": "Sorin Sarca", + "email": "sarca_sorin@hotmail.com" + } + ], + "description": "A library that can be used to serialize closures (anonymous functions) and arbitrary objects.", + "homepage": "https://opis.io/closure", + "keywords": [ + "anonymous functions", + "closure", + "function", + "serializable", + "serialization", + "serialize" + ], + "support": { + "issues": "https://github.com/opis/closure/issues", + "source": "https://github.com/opis/closure/tree/3.6.3" + }, + "time": "2022-01-27T09:35:39+00:00" + }, + { + "name": "phpoffice/phpspreadsheet", + "version": "1.19.0", + "source": { + "type": "git", + "url": "https://github.com/PHPOffice/PhpSpreadsheet.git", + "reference": "a9ab55bfae02eecffb3df669a2e19ba0e2f04bbf" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/PHPOffice/PhpSpreadsheet/zipball/a9ab55bfae02eecffb3df669a2e19ba0e2f04bbf", + "reference": "a9ab55bfae02eecffb3df669a2e19ba0e2f04bbf", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "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.13", + "maennchen/zipstream-php": "^2.1", + "markbaker/complex": "^3.0", + "markbaker/matrix": "^3.0", + "php": "^7.2 || ^8.0", + "psr/http-client": "^1.0", + "psr/http-factory": "^1.0", + "psr/simple-cache": "^1.0" + }, + "require-dev": { + "dealerdirect/phpcodesniffer-composer-installer": "dev-master", + "dompdf/dompdf": "^1.0", + "friendsofphp/php-cs-fixer": "^2.18", + "jpgraph/jpgraph": "^4.0", + "mpdf/mpdf": "^8.0", + "phpcompatibility/php-compatibility": "^9.3", + "phpstan/phpstan": "^0.12.82", + "phpstan/phpstan-phpunit": "^0.12.18", + "phpunit/phpunit": "^8.5", + "squizlabs/php_codesniffer": "^3.5", + "tecnickcom/tcpdf": "^6.3" + }, + "suggest": { + "dompdf/dompdf": "Option for rendering PDF with PDF Writer (doesn't yet support PHP8)", + "jpgraph/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 (doesn't yet support PHP8)" + }, + "type": "library", + "autoload": { + "psr-4": { + "PhpOffice\\PhpSpreadsheet\\": "src/PhpSpreadsheet" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/PHPOffice/PhpSpreadsheet/issues", + "source": "https://github.com/PHPOffice/PhpSpreadsheet/tree/1.19.0" + }, + "time": "2021-10-31T15:09:20+00:00" + }, + { + "name": "phpoption/phpoption", + "version": "1.8.1", + "source": { + "type": "git", + "url": "https://github.com/schmittjoh/php-option.git", + "reference": "eab7a0df01fe2344d172bff4cd6dbd3f8b84ad15" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/schmittjoh/php-option/zipball/eab7a0df01fe2344d172bff4cd6dbd3f8b84ad15", + "reference": "eab7a0df01fe2344d172bff4cd6dbd3f8b84ad15", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": "^7.0 || ^8.0" + }, + "require-dev": { + "bamarni/composer-bin-plugin": "^1.4.1", + "phpunit/phpunit": "^6.5.14 || ^7.5.20 || ^8.5.19 || ^9.5.8" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.8-dev" + } + }, + "autoload": { + "psr-4": { + "PhpOption\\": "src/PhpOption/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/schmittjoh/php-option/issues", + "source": "https://github.com/schmittjoh/php-option/tree/1.8.1" + }, + "funding": [ + { + "url": "https://github.com/GrahamCampbell", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/phpoption/phpoption", + "type": "tidelift" + } + ], + "time": "2021-12-04T23:24:31+00:00" + }, + { + "name": "psr/cache", + "version": "1.0.1", + "source": { + "type": "git", + "url": "https://github.com/php-fig/cache.git", + "reference": "d11b50ad223250cf17b86e38383413f5a6764bf8" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-fig/cache/zipball/d11b50ad223250cf17b86e38383413f5a6764bf8", + "reference": "d11b50ad223250cf17b86e38383413f5a6764bf8", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=5.3.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Cache\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "http://www.php-fig.org/" + } + ], + "description": "Common interface for caching libraries", + "keywords": [ + "cache", + "psr", + "psr-6" + ], + "support": { + "source": "https://github.com/php-fig/cache/tree/master" + }, + "time": "2016-08-06T20:24:11+00:00" + }, + { + "name": "psr/container", + "version": "1.1.1", + "source": { + "type": "git", + "url": "https://github.com/php-fig/container.git", + "reference": "8622567409010282b7aeebe4bb841fe98b58dcaf" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-fig/container/zipball/8622567409010282b7aeebe4bb841fe98b58dcaf", + "reference": "8622567409010282b7aeebe4bb841fe98b58dcaf", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.2.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Psr\\Container\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/php-fig/container/issues", + "source": "https://github.com/php-fig/container/tree/1.1.1" + }, + "time": "2021-03-05T17:36:06+00:00" + }, + { + "name": "psr/http-client", + "version": "1.0.1", + "source": { + "type": "git", + "url": "https://github.com/php-fig/http-client.git", + "reference": "2dfb5f6c5eff0e91e20e913f8c5452ed95b86621" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-fig/http-client/zipball/2dfb5f6c5eff0e91e20e913f8c5452ed95b86621", + "reference": "2dfb5f6c5eff0e91e20e913f8c5452ed95b86621", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": "^7.0 || ^8.0", + "psr/http-message": "^1.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Http\\Client\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "http://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" + ], + "support": { + "source": "https://github.com/php-fig/http-client/tree/master" + }, + "time": "2020-06-29T06:28:15+00:00" + }, + { + "name": "psr/http-factory", + "version": "1.0.1", + "source": { + "type": "git", + "url": "https://github.com/php-fig/http-factory.git", + "reference": "12ac7fcd07e5b077433f5f2bee95b3a771bf61be" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-fig/http-factory/zipball/12ac7fcd07e5b077433f5f2bee95b3a771bf61be", + "reference": "12ac7fcd07e5b077433f5f2bee95b3a771bf61be", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.0.0", + "psr/http-message": "^1.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Http\\Message\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "http://www.php-fig.org/" + } + ], + "description": "Common interfaces for PSR-7 HTTP message factories", + "keywords": [ + "factory", + "http", + "message", + "psr", + "psr-17", + "psr-7", + "request", + "response" + ], + "support": { + "source": "https://github.com/php-fig/http-factory/tree/master" + }, + "time": "2019-04-30T12:38:16+00:00" + }, + { + "name": "psr/http-message", + "version": "1.0.1", + "source": { + "type": "git", + "url": "https://github.com/php-fig/http-message.git", + "reference": "f6561bf28d520154e4b0ec72be95418abe6d9363" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-fig/http-message/zipball/f6561bf28d520154e4b0ec72be95418abe6d9363", + "reference": "f6561bf28d520154e4b0ec72be95418abe6d9363", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=5.3.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Http\\Message\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "PHP-FIG", + "homepage": "http://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" + ], + "support": { + "source": "https://github.com/php-fig/http-message/tree/master" + }, + "time": "2016-08-06T14:39:51+00:00" + }, + { + "name": "psr/log", + "version": "1.1.4", + "source": { + "type": "git", + "url": "https://github.com/php-fig/log.git", + "reference": "d49695b909c3b7628b6289db5479a1c204601f11" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-fig/log/zipball/d49695b909c3b7628b6289db5479a1c204601f11", + "reference": "d49695b909c3b7628b6289db5479a1c204601f11", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=5.3.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.1.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\Log\\": "Psr/Log/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "source": "https://github.com/php-fig/log/tree/1.1.4" + }, + "time": "2021-05-03T11:20:27+00:00" + }, + { + "name": "psr/simple-cache", + "version": "1.0.1", + "source": { + "type": "git", + "url": "https://github.com/php-fig/simple-cache.git", + "reference": "408d5eafb83c57f6365a3ca330ff23aa4a5fa39b" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/php-fig/simple-cache/zipball/408d5eafb83c57f6365a3ca330ff23aa4a5fa39b", + "reference": "408d5eafb83c57f6365a3ca330ff23aa4a5fa39b", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=5.3.0" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "1.0.x-dev" + } + }, + "autoload": { + "psr-4": { + "Psr\\SimpleCache\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "source": "https://github.com/php-fig/simple-cache/tree/master" + }, + "time": "2017-10-23T01:57:42+00:00" + }, + { + "name": "qcloud/cos-sdk-v5", + "version": "v2.6.0", + "source": { + "type": "git", + "url": "https://github.com/tencentyun/cos-php-sdk-v5.git", + "reference": "bb9f0f1a72922413a3d8eae0f02fd7f78e7d847b" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/tencentyun/cos-php-sdk-v5/zipball/bb9f0f1a72922413a3d8eae0f02fd7f78e7d847b", + "reference": "bb9f0f1a72922413a3d8eae0f02fd7f78e7d847b", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "ext-curl": "*", + "ext-json": "*", + "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/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/tencentyun/cos-php-sdk-v5/issues", + "source": "https://github.com/tencentyun/cos-php-sdk-v5/tree/v2.6.0" + }, + "time": "2022-11-14T11:12:33+00:00" + }, + { + "name": "qiniu/php-sdk", + "version": "v7.4.0", + "source": { + "type": "git", + "url": "https://github.com/qiniu/php-sdk.git", + "reference": "1c6bc89166e524a40ee42bf516fb99ffc6401c82" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/qiniu/php-sdk/zipball/1c6bc89166e524a40ee42bf516fb99ffc6401c82", + "reference": "1c6bc89166e524a40ee42bf516fb99ffc6401c82", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "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" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/qiniu/php-sdk/issues", + "source": "https://github.com/qiniu/php-sdk/tree/v7.4.0" + }, + "time": "2021-07-19T07:41:36+00:00" + }, + { + "name": "ralouphie/getallheaders", + "version": "3.0.3", + "source": { + "type": "git", + "url": "https://github.com/ralouphie/getallheaders.git", + "reference": "120b605dfeb996808c31b6477290a714d356e822" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/ralouphie/getallheaders/zipball/120b605dfeb996808c31b6477290a714d356e822", + "reference": "120b605dfeb996808c31b6477290a714d356e822", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=5.6" + }, + "require-dev": { + "php-coveralls/php-coveralls": "^2.1", + "phpunit/phpunit": "^5 || ^6.5" + }, + "type": "library", + "autoload": { + "files": [ + "src/getallheaders.php" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "authors": [ + { + "name": "Ralph Khattar", + "email": "ralph.khattar@gmail.com" + } + ], + "description": "A polyfill for getallheaders.", + "support": { + "issues": "https://github.com/ralouphie/getallheaders/issues", + "source": "https://github.com/ralouphie/getallheaders/tree/develop" + }, + "time": "2019-03-08T08:55:37+00:00" + }, + { + "name": "symfony/console", + "version": "v5.4.17", + "source": { + "type": "git", + "url": "https://github.com/symfony/console.git", + "reference": "58422fdcb0e715ed05b385f70d3e8b5ed4bbd45f" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/console/zipball/58422fdcb0e715ed05b385f70d3e8b5ed4bbd45f", + "reference": "58422fdcb0e715ed05b385f70d3e8b5ed4bbd45f", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.2.5", + "symfony/deprecation-contracts": "^2.1|^3", + "symfony/polyfill-mbstring": "~1.0", + "symfony/polyfill-php73": "^1.9", + "symfony/polyfill-php80": "^1.16", + "symfony/service-contracts": "^1.1|^2|^3", + "symfony/string": "^5.1|^6.0" + }, + "conflict": { + "psr/log": ">=3", + "symfony/dependency-injection": "<4.4", + "symfony/dotenv": "<5.1", + "symfony/event-dispatcher": "<4.4", + "symfony/lock": "<4.4", + "symfony/process": "<4.4" + }, + "provide": { + "psr/log-implementation": "1.0|2.0" + }, + "require-dev": { + "psr/log": "^1|^2", + "symfony/config": "^4.4|^5.0|^6.0", + "symfony/dependency-injection": "^4.4|^5.0|^6.0", + "symfony/event-dispatcher": "^4.4|^5.0|^6.0", + "symfony/lock": "^4.4|^5.0|^6.0", + "symfony/process": "^4.4|^5.0|^6.0", + "symfony/var-dumper": "^4.4|^5.0|^6.0" + }, + "suggest": { + "psr/log": "For using the console logger", + "symfony/event-dispatcher": "", + "symfony/lock": "", + "symfony/process": "" + }, + "type": "library", + "autoload": { + "psr-4": { + "Symfony\\Component\\Console\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "source": "https://github.com/symfony/console/tree/v5.4.17" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2022-12-28T14:15:31+00:00" + }, + { + "name": "symfony/deprecation-contracts", + "version": "v2.5.2", + "source": { + "type": "git", + "url": "https://github.com/symfony/deprecation-contracts.git", + "reference": "e8b495ea28c1d97b5e0c121748d6f9b53d075c66" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/deprecation-contracts/zipball/e8b495ea28c1d97b5e0c121748d6f9b53d075c66", + "reference": "e8b495ea28c1d97b5e0c121748d6f9b53d075c66", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.1" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "2.5-dev" + }, + "thanks": { + "name": "symfony/contracts", + "url": "https://github.com/symfony/contracts" + } + }, + "autoload": { + "files": [ + "function.php" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "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", + "support": { + "source": "https://github.com/symfony/deprecation-contracts/tree/v2.5.2" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2022-01-02T09:53:40+00:00" + }, + { + "name": "symfony/polyfill-ctype", + "version": "v1.27.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/polyfill-ctype.git", + "reference": "5bbc823adecdae860bb64756d639ecfec17b050a" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/polyfill-ctype/zipball/5bbc823adecdae860bb64756d639ecfec17b050a", + "reference": "5bbc823adecdae860bb64756d639ecfec17b050a", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.1" + }, + "provide": { + "ext-ctype": "*" + }, + "suggest": { + "ext-ctype": "For best performance" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.27-dev" + }, + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Ctype\\": "" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "source": "https://github.com/symfony/polyfill-ctype/tree/v1.27.0" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2022-11-03T14:55:06+00:00" + }, + { + "name": "symfony/polyfill-intl-grapheme", + "version": "v1.27.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/polyfill-intl-grapheme.git", + "reference": "511a08c03c1960e08a883f4cffcacd219b758354" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/polyfill-intl-grapheme/zipball/511a08c03c1960e08a883f4cffcacd219b758354", + "reference": "511a08c03c1960e08a883f4cffcacd219b758354", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.1" + }, + "suggest": { + "ext-intl": "For best performance" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.27-dev" + }, + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Intl\\Grapheme\\": "" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "source": "https://github.com/symfony/polyfill-intl-grapheme/tree/v1.27.0" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2022-11-03T14:55:06+00:00" + }, + { + "name": "symfony/polyfill-intl-normalizer", + "version": "v1.27.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/polyfill-intl-normalizer.git", + "reference": "19bd1e4fcd5b91116f14d8533c57831ed00571b6" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/polyfill-intl-normalizer/zipball/19bd1e4fcd5b91116f14d8533c57831ed00571b6", + "reference": "19bd1e4fcd5b91116f14d8533c57831ed00571b6", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.1" + }, + "suggest": { + "ext-intl": "For best performance" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.27-dev" + }, + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Intl\\Normalizer\\": "" + }, + "classmap": [ + "Resources/stubs" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "source": "https://github.com/symfony/polyfill-intl-normalizer/tree/v1.27.0" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2022-11-03T14:55:06+00:00" + }, + { + "name": "symfony/polyfill-mbstring", + "version": "v1.27.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/polyfill-mbstring.git", + "reference": "8ad114f6b39e2c98a8b0e3bd907732c207c2b534" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/polyfill-mbstring/zipball/8ad114f6b39e2c98a8b0e3bd907732c207c2b534", + "reference": "8ad114f6b39e2c98a8b0e3bd907732c207c2b534", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.1" + }, + "provide": { + "ext-mbstring": "*" + }, + "suggest": { + "ext-mbstring": "For best performance" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.27-dev" + }, + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Mbstring\\": "" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "source": "https://github.com/symfony/polyfill-mbstring/tree/v1.27.0" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2022-11-03T14:55:06+00:00" + }, + { + "name": "symfony/polyfill-php73", + "version": "v1.27.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/polyfill-php73.git", + "reference": "9e8ecb5f92152187c4799efd3c96b78ccab18ff9" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/polyfill-php73/zipball/9e8ecb5f92152187c4799efd3c96b78ccab18ff9", + "reference": "9e8ecb5f92152187c4799efd3c96b78ccab18ff9", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.1" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.27-dev" + }, + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Php73\\": "" + }, + "classmap": [ + "Resources/stubs" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "source": "https://github.com/symfony/polyfill-php73/tree/v1.27.0" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2022-11-03T14:55:06+00:00" + }, + { + "name": "symfony/polyfill-php80", + "version": "v1.27.0", + "source": { + "type": "git", + "url": "https://github.com/symfony/polyfill-php80.git", + "reference": "7a6ff3f1959bb01aefccb463a0f2cd3d3d2fd936" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/polyfill-php80/zipball/7a6ff3f1959bb01aefccb463a0f2cd3d3d2fd936", + "reference": "7a6ff3f1959bb01aefccb463a0f2cd3d3d2fd936", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.1" + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-main": "1.27-dev" + }, + "thanks": { + "name": "symfony/polyfill", + "url": "https://github.com/symfony/polyfill" + } + }, + "autoload": { + "files": [ + "bootstrap.php" + ], + "psr-4": { + "Symfony\\Polyfill\\Php80\\": "" + }, + "classmap": [ + "Resources/stubs" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "source": "https://github.com/symfony/polyfill-php80/tree/v1.27.0" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2022-11-03T14:55:06+00:00" + }, + { + "name": "symfony/service-contracts", + "version": "v2.5.2", + "source": { + "type": "git", + "url": "https://github.com/symfony/service-contracts.git", + "reference": "4b426aac47d6427cc1a1d0f7e2ac724627f5966c" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/service-contracts/zipball/4b426aac47d6427cc1a1d0f7e2ac724627f5966c", + "reference": "4b426aac47d6427cc1a1d0f7e2ac724627f5966c", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "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\\": "" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "source": "https://github.com/symfony/service-contracts/tree/v2.5.2" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2022-05-30T19:17:29+00:00" + }, + { + "name": "symfony/string", + "version": "v5.4.17", + "source": { + "type": "git", + "url": "https://github.com/symfony/string.git", + "reference": "55733a8664b8853b003e70251c58bc8cb2d82a6b" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/symfony/string/zipball/55733a8664b8853b003e70251c58bc8cb2d82a6b", + "reference": "55733a8664b8853b003e70251c58bc8cb2d82a6b", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.2.5", + "symfony/polyfill-ctype": "~1.8", + "symfony/polyfill-intl-grapheme": "~1.0", + "symfony/polyfill-intl-normalizer": "~1.0", + "symfony/polyfill-mbstring": "~1.0", + "symfony/polyfill-php80": "~1.15" + }, + "conflict": { + "symfony/translation-contracts": ">=3.0" + }, + "require-dev": { + "symfony/error-handler": "^4.4|^5.0|^6.0", + "symfony/http-client": "^4.4|^5.0|^6.0", + "symfony/translation-contracts": "^1.1|^2", + "symfony/var-exporter": "^4.4|^5.0|^6.0" + }, + "type": "library", + "autoload": { + "files": [ + "Resources/functions.php" + ], + "psr-4": { + "Symfony\\Component\\String\\": "" + }, + "exclude-from-classmap": [ + "/Tests/" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "source": "https://github.com/symfony/string/tree/v5.4.17" + }, + "funding": [ + { + "url": "https://symfony.com/sponsor", + "type": "custom" + }, + { + "url": "https://github.com/fabpot", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/symfony/symfony", + "type": "tidelift" + } + ], + "time": "2022-12-12T15:54:21+00:00" + }, + { + "name": "tencentcloud/tencentcloud-sdk-php", + "version": "3.0.807", + "source": { + "type": "git", + "url": "https://github.com/TencentCloud/tencentcloud-sdk-php.git", + "reference": "d7c58728eee8842b0559866aea94b6d136f3a49b" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/TencentCloud/tencentcloud-sdk-php/zipball/d7c58728eee8842b0559866aea94b6d136f3a49b", + "reference": "d7c58728eee8842b0559866aea94b6d136f3a49b", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "guzzlehttp/guzzle": "^6.3 || ^7.0", + "php": ">=5.6.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "TencentCloud\\": "./src/TencentCloud" + }, + "classmap": [ + "src/QcloudApi/QcloudApi.php" + ] + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "coolli", + "email": "tencentcloudapi@tencent.com", + "homepage": "https://cloud.tencent.com/document/sdk/PHP", + "role": "Developer" + } + ], + "description": "TencentCloudApi php sdk", + "homepage": "https://github.com/TencentCloud/tencentcloud-sdk-php", + "support": { + "issues": "https://github.com/TencentCloud/tencentcloud-sdk-php/issues", + "source": "https://github.com/TencentCloud/tencentcloud-sdk-php/tree/3.0.807" + }, + "time": "2023-01-12T00:07:10+00:00" + }, + { + "name": "tinywan/storage", + "version": "v0.3.4", + "source": { + "type": "git", + "url": "https://github.com/Tinywan/webman-storage.git", + "reference": "0867631bbd1731658ac745481131ef93415cdf62" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/Tinywan/webman-storage/zipball/0867631bbd1731658ac745481131ef93415cdf62", + "reference": "0867631bbd1731658ac745481131ef93415cdf62", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "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" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "description": "webman storage plugin", + "support": { + "issues": "https://github.com/Tinywan/webman-storage/issues", + "source": "https://github.com/Tinywan/webman-storage/tree/v0.3.4" + }, + "time": "2022-08-03T12:12:34+00:00" + }, + { + "name": "topthink/think-cache", + "version": "v2.0.6", + "source": { + "type": "git", + "url": "https://github.com/top-think/think-cache.git", + "reference": "75a56b24affc65b51688fd89ada48c102757fd74" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/top-think/think-cache/zipball/75a56b24affc65b51688fd89ada48c102757fd74", + "reference": "75a56b24affc65b51688fd89ada48c102757fd74", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "opis/closure": "^3.1", + "php": ">=7.1.0", + "psr/cache": "~1.0", + "psr/simple-cache": "^1.0", + "topthink/think-container": "~2.0" + }, + "type": "library", + "autoload": { + "files": [], + "psr-4": { + "think\\": "src" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "liu21st", + "email": "liu21st@gmail.com" + } + ], + "description": "Cache Manager", + "support": { + "issues": "https://github.com/top-think/think-cache/issues", + "source": "https://github.com/top-think/think-cache/tree/v2.0.6" + }, + "time": "2019-07-07T14:34:35+00:00" + }, + { + "name": "topthink/think-container", + "version": "v2.0.5", + "source": { + "type": "git", + "url": "https://github.com/top-think/think-container.git", + "reference": "2189b39e42af2c14203ed4372b92e38989e9dabb" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/top-think/think-container/zipball/2189b39e42af2c14203ed4372b92e38989e9dabb", + "reference": "2189b39e42af2c14203ed4372b92e38989e9dabb", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "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" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "liu21st", + "email": "liu21st@gmail.com" + } + ], + "description": "PHP Container & Facade Manager", + "support": { + "issues": "https://github.com/top-think/think-container/issues", + "source": "https://github.com/top-think/think-container/tree/v2.0.5" + }, + "time": "2022-05-23T06:24:54+00:00" + }, + { + "name": "topthink/think-helper", + "version": "v3.1.6", + "source": { + "type": "git", + "url": "https://github.com/top-think/think-helper.git", + "reference": "769acbe50a4274327162f9c68ec2e89a38eb2aff" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/top-think/think-helper/zipball/769acbe50a4274327162f9c68ec2e89a38eb2aff", + "reference": "769acbe50a4274327162f9c68ec2e89a38eb2aff", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.1.0" + }, + "require-dev": { + "phpunit/phpunit": "^9.5" + }, + "type": "library", + "autoload": { + "files": [ + "src/helper.php" + ], + "psr-4": { + "think\\": "src" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "yunwuxin", + "email": "448901948@qq.com" + } + ], + "description": "The ThinkPHP6 Helper Package", + "support": { + "issues": "https://github.com/top-think/think-helper/issues", + "source": "https://github.com/top-think/think-helper/tree/v3.1.6" + }, + "time": "2021-12-15T04:27:55+00:00" + }, + { + "name": "topthink/think-orm", + "version": "v2.0.56", + "source": { + "type": "git", + "url": "https://github.com/top-think/think-orm.git", + "reference": "75b8512736daaa056d511f42c15bed87c9f3605a" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/top-think/think-orm/zipball/75b8512736daaa056d511f42c15bed87c9f3605a", + "reference": "75b8512736daaa056d511f42c15bed87c9f3605a", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "ext-json": "*", + "ext-pdo": "*", + "php": ">=7.1.0", + "psr/log": "^1.0|^2.0", + "psr/simple-cache": "^1.0|^2.0", + "topthink/think-helper": "^3.1" + }, + "require-dev": { + "phpunit/phpunit": "^7|^8|^9.5" + }, + "type": "library", + "autoload": { + "files": [ + "stubs/load_stubs.php" + ], + "psr-4": { + "think\\": "src" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "liu21st", + "email": "liu21st@gmail.com" + } + ], + "description": "think orm", + "keywords": [ + "database", + "orm" + ], + "support": { + "issues": "https://github.com/top-think/think-orm/issues", + "source": "https://github.com/top-think/think-orm/tree/v2.0.56" + }, + "time": "2022-12-15T02:52:53+00:00" + }, + { + "name": "topthink/think-validate", + "version": "v2.0.2", + "source": { + "type": "git", + "url": "https://github.com/top-think/think-validate.git", + "reference": "857f9bffc1a09a41e3969a19726cb04315848f0f" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/top-think/think-validate/zipball/857f9bffc1a09a41e3969a19726cb04315848f0f", + "reference": "857f9bffc1a09a41e3969a19726cb04315848f0f", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.1.0", + "topthink/think-container": "~2.0" + }, + "type": "library", + "autoload": { + "files": [], + "psr-4": { + "think\\": "src" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "Apache-2.0" + ], + "authors": [ + { + "name": "liu21st", + "email": "liu21st@gmail.com" + } + ], + "description": "think validate", + "support": { + "issues": "https://github.com/top-think/think-validate/issues", + "source": "https://github.com/top-think/think-validate/tree/2.0" + }, + "time": "2019-05-15T06:58:23+00:00" + }, + { + "name": "vlucas/phpdotenv", + "version": "v5.4.1", + "source": { + "type": "git", + "url": "https://github.com/vlucas/phpdotenv.git", + "reference": "264dce589e7ce37a7ba99cb901eed8249fbec92f" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/vlucas/phpdotenv/zipball/264dce589e7ce37a7ba99cb901eed8249fbec92f", + "reference": "264dce589e7ce37a7ba99cb901eed8249fbec92f", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "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.21 || ^9.5.10" + }, + "suggest": { + "ext-filter": "Required to use the boolean validator." + }, + "type": "library", + "extra": { + "branch-alias": { + "dev-master": "5.4-dev" + } + }, + "autoload": { + "psr-4": { + "Dotenv\\": "src/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/vlucas/phpdotenv/issues", + "source": "https://github.com/vlucas/phpdotenv/tree/v5.4.1" + }, + "funding": [ + { + "url": "https://github.com/GrahamCampbell", + "type": "github" + }, + { + "url": "https://tidelift.com/funding/github/packagist/vlucas/phpdotenv", + "type": "tidelift" + } + ], + "time": "2021-12-12T23:22:04+00:00" + }, + { + "name": "webman/console", + "version": "v1.2.18", + "source": { + "type": "git", + "url": "https://github.com/webman-php/console.git", + "reference": "96c5d57b36a9c46c2fbb7dc6df251e30aeb48c63" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/webman-php/console/zipball/96c5d57b36a9c46c2fbb7dc6df251e30aeb48c63", + "reference": "96c5d57b36a9c46c2fbb7dc6df251e30aeb48c63", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "doctrine/inflector": "^2.0", + "symfony/console": ">=5.0" + }, + "require-dev": { + "workerman/webman": "^1.0" + }, + "type": "library", + "autoload": { + "psr-4": { + "Webman\\Console\\": "src" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "email": "walkor@workerman.net", + "forum": "http://www.workerman.net/questions", + "issues": "https://github.com/webman-php/console/issues", + "source": "https://github.com/webman-php/console", + "wiki": "http://www.workerman.net/doc/webman" + }, + "time": "2022-11-26T15:15:21+00:00" + }, + { + "name": "webman/think-cache", + "version": "v1.0.1", + "source": { + "type": "git", + "url": "https://github.com/webman-php/think-cache.git", + "reference": "25bd103d7fc9347aca680e677282db761cc90a43" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/webman-php/think-cache/zipball/25bd103d7fc9347aca680e677282db761cc90a43", + "reference": "25bd103d7fc9347aca680e677282db761cc90a43", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "topthink/think-cache": "^2.0.6" + }, + "type": "library", + "autoload": { + "psr-4": { + "Webman\\ThinkCache\\": "src" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "support": { + "issues": "https://github.com/webman-php/think-cache/issues", + "source": "https://github.com/webman-php/think-cache/tree/v1.0.1" + }, + "time": "2022-03-30T03:27:46+00:00" + }, + { + "name": "webman/think-orm", + "version": "v1.0.12", + "source": { + "type": "git", + "url": "https://github.com/webman-php/think-orm.git", + "reference": "6eabf395a7afa507bdb17e0ccd3bb4df9dda8d16" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/webman-php/think-orm/zipball/6eabf395a7afa507bdb17e0ccd3bb4df9dda8d16", + "reference": "6eabf395a7afa507bdb17e0ccd3bb4df9dda8d16", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "topthink/think-orm": "^2.0.53", + "workerman/webman-framework": "^1.2.1" + }, + "type": "library", + "autoload": { + "psr-4": { + "Webman\\ThinkOrm\\": "src" + } + }, + "notification-url": "https://packagist.org/downloads/", + "license": [ + "MIT" + ], + "support": { + "issues": "https://github.com/webman-php/think-orm/issues", + "source": "https://github.com/webman-php/think-orm/tree/v1.0.12" + }, + "time": "2022-12-19T01:52:53+00:00" + }, + { + "name": "webmozart/assert", + "version": "1.11.0", + "source": { + "type": "git", + "url": "https://github.com/webmozarts/assert.git", + "reference": "11cb2199493b2f8a3b53e7f19068fc6aac760991" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/webmozarts/assert/zipball/11cb2199493b2f8a3b53e7f19068fc6aac760991", + "reference": "11cb2199493b2f8a3b53e7f19068fc6aac760991", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "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/" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "issues": "https://github.com/webmozarts/assert/issues", + "source": "https://github.com/webmozarts/assert/tree/1.11.0" + }, + "time": "2022-06-03T18:03:27+00:00" + }, + { + "name": "workerman/webman-framework", + "version": "v1.4.10", + "source": { + "type": "git", + "url": "https://github.com/walkor/webman-framework.git", + "reference": "d9d6a5317f1f11486e37bf5613aa0d1601b83edd" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/walkor/webman-framework/zipball/d9d6a5317f1f11486e37bf5613aa0d1601b83edd", + "reference": "d9d6a5317f1f11486e37bf5613aa0d1601b83edd", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "nikic/fast-route": "^1.3", + "php": ">=7.2", + "psr/container": ">=1.0", + "workerman/workerman": "^4.0.4" + }, + "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" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "email": "walkor@workerman.net", + "forum": "https://wenda.workerman.net/", + "issues": "https://github.com/walkor/webman/issues", + "source": "https://github.com/walkor/webman-framework", + "wiki": "https://doc.workerman.net/" + }, + "time": "2022-12-12T07:54:21+00:00" + }, + { + "name": "workerman/workerman", + "version": "v4.1.5", + "source": { + "type": "git", + "url": "https://github.com/walkor/workerman.git", + "reference": "16bcfc2c7574feea46cdadaaa8ae73f14d464b21" + }, + "dist": { + "type": "zip", + "url": "https://api.github.com/repos/walkor/workerman/zipball/16bcfc2c7574feea46cdadaaa8ae73f14d464b21", + "reference": "16bcfc2c7574feea46cdadaaa8ae73f14d464b21", + "shasum": "", + "mirrors": [ + { + "url": "https://mirrors.aliyun.com/composer/dists/%package%/%reference%.%type%", + "preferred": true + } + ] + }, + "require": { + "php": ">=7.0" + }, + "suggest": { + "ext-event": "For better performance. " + }, + "type": "library", + "autoload": { + "psr-4": { + "Workerman\\": "./" + } + }, + "notification-url": "https://packagist.org/downloads/", + "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" + ], + "support": { + "email": "walkor@workerman.net", + "forum": "http://wenda.workerman.net/", + "issues": "https://github.com/walkor/workerman/issues", + "source": "https://github.com/walkor/workerman", + "wiki": "http://doc.workerman.net/" + }, + "funding": [ + { + "url": "https://opencollective.com/workerman", + "type": "open_collective" + }, + { + "url": "https://www.patreon.com/walkor", + "type": "patreon" + } + ], + "time": "2022-12-14T11:58:06+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/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..6bcb734 --- /dev/null +++ b/config/bootstrap.php @@ -0,0 +1,20 @@ + + * @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, + Webman\ThinkCache\ThinkCache::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..7dc463a --- /dev/null +++ b/config/database.php @@ -0,0 +1,15 @@ + + * @copyright walkor + * @link http://www.workerman.net/ + * @license http://www.opensource.org/licenses/mit-license.php MIT License + */ + +return []; 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/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/thinkcache.php b/config/thinkcache.php new file mode 100644 index 0000000..487f18f --- /dev/null +++ b/config/thinkcache.php @@ -0,0 +1,22 @@ + 'file', + 'stores' => [ + 'file' => [ + 'type' => 'File', + // 缓存保存目录 + 'path' => runtime_path() . '/cache/', + // 缓存前缀 + 'prefix' => '', + // 缓存有效期 0表示永久缓存 + 'expire' => 0, + ], + 'redis' => [ + 'type' => 'redis', + 'host' => '127.0.0.1', + 'port' => 6379, + 'prefix' => '', + 'expire' => 0, + ], + ], +]; \ No newline at end of file diff --git a/config/thinkorm.php b/config/thinkorm.php new file mode 100644 index 0000000..9be215f --- /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' => false, + // 自定义分页类 + '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/export/2023-01/4217088ef73f98b57248170952a5f491/用户列表-2023-01-13-103646.xlsx b/public/export/2023-01/4217088ef73f98b57248170952a5f491/用户列表-2023-01-13-103646.xlsx new file mode 100755 index 0000000..03d0e8f Binary files /dev/null and b/public/export/2023-01/4217088ef73f98b57248170952a5f491/用户列表-2023-01-13-103646.xlsx differ diff --git a/public/export/2023-01/906da811046f469b2c445aeaf5da06bd/用户列表-2023-01-13-105658.xlsx b/public/export/2023-01/906da811046f469b2c445aeaf5da06bd/用户列表-2023-01-13-105658.xlsx new file mode 100755 index 0000000..215ae8b Binary files /dev/null and b/public/export/2023-01/906da811046f469b2c445aeaf5da06bd/用户列表-2023-01-13-105658.xlsx differ 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/test.php b/public/test.php new file mode 100644 index 0000000..c9b8382 --- /dev/null +++ b/public/test.php @@ -0,0 +1,2 @@ + + * @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..7a08896 --- /dev/null +++ b/support/bootstrap.php @@ -0,0 +1,132 @@ + + * @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\Route; +use Webman\Middleware; +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 ($start_time) { + if (time() - $start_time <= 1) { + sleep(1); + } + }, time()); +} + +if (class_exists('Dotenv\Dotenv') && file_exists(base_path() . '/.env')) { + if (method_exists('Dotenv\Dotenv', 'createUnsafeImmutable')) { + Dotenv::createUnsafeImmutable(base_path())->load(); + } else { + Dotenv::createMutable(base_path())->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 ($static_middlewares = config("plugin.$firm.static.middleware")) { + Middleware::load(['__static__' => $static_middlewares], $firm); + } +} +Middleware::load(['__static__' => config('static.middleware', [])], ''); + +foreach (config('bootstrap', []) as $class_name) { + if (!class_exists($class_name)) { + $log = "Warning: Class $class_name setting in config/bootstrap.php not found\r\n"; + echo $log; + Log::error($log); + continue; + } + /** @var Bootstrap $class_name */ + $class_name::start($worker); +} + +foreach (config('plugin', []) as $firm => $projects) { + foreach ($projects as $name => $project) { + if (!is_array($project)) { + continue; + } + foreach ($project['bootstrap'] ?? [] as $class_name) { + if (!class_exists($class_name)) { + $log = "Warning: Class $class_name setting in config/plugin/$firm/$name/bootstrap.php not found\r\n"; + echo $log; + Log::error($log); + continue; + } + /** @var Bootstrap $class_name */ + $class_name::start($worker); + } + } + foreach ($projects['bootstrap'] ?? [] as $class_name) { + if (!class_exists($class_name)) { + $log = "Warning: Class $class_name setting in plugin/$firm/config/bootstrap.php not found\r\n"; + echo $log; + Log::error($log); + continue; + } + /** @var Bootstrap $class_name */ + $class_name::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..6302810 --- /dev/null +++ b/support/helpers.php @@ -0,0 +1,525 @@ + + * @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\Raw; +use support\view\Blade; +use support\view\ThinkPHP; +use support\view\Twig; +use Workerman\Worker; +use Webman\App; +use Webman\Config; +use Webman\Route; + +// Webman version +const WEBMAN_VERSION = '1.4'; + +// 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 $run_path = ''; + if (!$run_path) { + $run_path = \is_phar() ? \dirname(\Phar::running(false)) : BASE_PATH; + } + return \path_combine($run_path, $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 $public_path = ''; + if (!$public_path) { + $public_path = \config('app.public_path') ? : \run_path('public'); + } + return \path_combine($public_path, $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 $runtime_path = ''; + if (!$runtime_path) { + $runtime_path = \config('app.runtime_path') ? : \run_path('runtime'); + } + return \path_combine($runtime_path, $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 $callback_name + * @return Response + */ +function jsonp($data, string $callback_name = 'callback'): Response +{ + if (!\is_scalar($data) && null !== $data) { + $data = \json_encode($data); + } + return new Response(200, [], "$callback_name($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 + * @return Response + */ +function view(string $template, array $vars = [], string $app = null): Response +{ + $request = \request(); + $plugin = $request->plugin ?? ''; + $handler = \config($plugin ? "plugin.$plugin.view.handler" : 'view.handler'); + return new Response(200, [], $handler::render($template, $vars, $app)); +} + +/** + * 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 + */ +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, '.')) { + $key_array = \explode('.', $key); + $value = $session->all(); + foreach ($key_array 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 void + */ +function locale(string $locale = null): string +{ + if (!$locale) { + return Translation::getLocale(); + } + Translation::setLocale($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) +{ + $callback_map = [ + 'onConnect', + 'onMessage', + 'onClose', + 'onError', + 'onBufferFull', + 'onBufferDrain', + 'onWorkerStop', + 'onWebSocketConnect' + ]; + foreach ($callback_map 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 $process_name + * @param $config + * @return void + */ +function worker_start($process_name, $config) +{ + $worker = new Worker($config['listen'] ?? null, $config['context'] ?? []); + $property_map = [ + 'count', + 'user', + 'group', + 'reloadable', + 'reusePort', + 'transport', + 'protocol', + ]; + $worker->name = $process_name; + foreach ($property_map as $property) { + if (isset($config[$property])) { + $worker->$property = $config[$property]; + } + } + + $worker->onWorkerStart = function ($worker) use ($config) { + require_once \base_path() . '/support/bootstrap.php'; + + foreach ($config['services'] ?? [] as $server) { + if (!\class_exists($server['handler'])) { + echo "process error: class {$server['handler']} not exists\r\n"; + continue; + } + $listen = new Worker($server['listen'] ?? null, $server['context'] ?? []); + if (isset($server['listen'])) { + echo "listen: {$server['listen']}\n"; + } + $instance = Container::make($server['handler'], $server['constructor'] ?? []); + \worker_bind($listen, $instance); + $listen->listen(); + } + + 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 $file_path + * @return string + */ +function get_realpath(string $file_path): string +{ + if (\strpos($file_path, 'phar://') === 0) { + return $file_path; + } else { + return \realpath($file_path); + } +} + +/** + * 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..08cd4ec --- /dev/null +++ b/windows.php @@ -0,0 +1,116 @@ +load(); + } else { + Dotenv::createMutable(base_path())->load(); + } +} + +App::loadAllConfig(['route']); + +$error_reporting = config('app.error_reporting'); +if (isset($error_reporting)) { + error_reporting($error_reporting); +} + +$runtime_process_path = runtime_path() . DIRECTORY_SEPARATOR . '/windows'; +if (!is_dir($runtime_process_path)) { + mkdir($runtime_process_path); +} +$process_files = [ + __DIR__ . DIRECTORY_SEPARATOR . 'start.php' +]; +foreach (config('process', []) as $process_name => $config) { + $process_files[] = write_process_file($runtime_process_path, $process_name, ''); +} + +foreach (config('plugin', []) as $firm => $projects) { + foreach ($projects as $name => $project) { + if (!is_array($project)) { + continue; + } + foreach ($project['process'] ?? [] as $process_name => $config) { + $process_files[] = write_process_file($runtime_process_path, $process_name, "$firm.$name"); + } + } + foreach ($projects['process'] ?? [] as $process_name => $config) { + $process_files[] = write_process_file($runtime_process_path, $process_name, $firm); + } +} + +function write_process_file($runtime_process_path, $process_name, $firm) +{ + $process_param = $firm ? "plugin.$firm.$process_name" : $process_name; + $config_param = $firm ? "config('plugin.$firm.process')['$process_name']" : "config('process')['$process_name']"; + $file_content = <<checkAllFilesChange()) { + $status = proc_get_status($resource); + $pid = $status['pid']; + shell_exec("taskkill /F /T /PID $pid"); + proc_close($resource); + $resource = popen_processes($process_files); + } +}