From 9dbc5cc4f26d11d13b4a0a2fe212dd769aae4677 Mon Sep 17 00:00:00 2001 From: =?UTF-8?q?=E9=82=B9=E6=99=AF=E7=AB=8B?= Date: Wed, 20 Jan 2021 18:17:14 +0800 Subject: [PATCH] =?UTF-8?q?=E5=90=8C=E6=AD=A5=E4=BB=A3=E7=A0=81?= MIME-Version: 1.0 Content-Type: text/plain; charset=UTF-8 Content-Transfer-Encoding: 8bit --- app/data/command/OrderClear.php | 87 ++++++++++ app/data/command/UserBalance.php | 43 +++++ app/data/controller/ShopGoods.php | 23 ++- app/data/controller/ShopOrder.php | 33 +++- app/data/controller/ShopOrderSend.php | 33 +++- app/data/controller/ShopPayment.php | 130 ++++++++++++++ app/data/controller/UserBalance.php | 124 +++++++++++++ app/data/controller/UserNotify.php | 94 ++++++++++ app/data/controller/api/Auth.php | 15 +- app/data/controller/api/Data.php | 13 ++ app/data/controller/api/Notify.php | 6 +- app/data/controller/api/auth/Address.php | 8 +- app/data/controller/api/auth/Balance.php | 75 ++++++++ app/data/controller/api/auth/Center.php | 48 +++--- app/data/controller/api/auth/Config.php | 6 +- app/data/controller/api/auth/Order.php | 163 +++++++++++------- app/data/service/NewsService.php | 2 +- app/data/service/OrderService.php | 18 +- app/data/service/PaymentService.php | 135 ++++++++++----- app/data/service/UserService.php | 130 ++++++++------ .../service/payment/BalancePyamentService.php | 67 +++++++ .../service/payment/EmptyPaymentService.php | 61 +++++++ app/data/sys.php | 12 ++ app/data/view/shop_goods/form.html | 4 +- app/data/view/shop_goods/index.html | 24 ++- app/data/view/shop_goods/stock.html | 88 ++++++---- app/data/view/shop_order/index.html | 20 ++- app/data/view/shop_order/index_search.html | 25 ++- app/data/view/shop_order/truck.html | 21 ++- app/data/view/shop_order_send/config.html | 43 +++++ app/data/view/shop_order_send/index.html | 25 ++- .../view/shop_order_send/index_search.html | 40 +++-- app/data/view/shop_payment/form.html | 89 ++++++++++ app/data/view/shop_payment/form_alipay.html | 18 ++ app/data/view/shop_payment/form_joinpay.html | 23 +++ app/data/view/shop_payment/form_wechat.html | 32 ++++ app/data/view/shop_payment/index.html | 70 ++++++++ app/data/view/shop_payment/index_search.html | 54 ++++++ app/data/view/shop_truck_company/index.html | 10 +- .../view/shop_truck_template/form_region.html | 2 + app/data/view/shop_truck_template/index.html | 8 +- app/data/view/user/index.html | 10 ++ app/data/view/user_balance/form.html | 70 ++++++++ app/data/view/user_balance/index.html | 63 +++++++ app/data/view/user_balance/index_search.html | 30 ++++ app/data/view/user_notify/form.html | 40 +++++ app/data/view/user_notify/index.html | 69 ++++++++ app/data/view/user_notify/index_search.html | 39 +++++ config/database.php | 2 +- 49 files changed, 1927 insertions(+), 318 deletions(-) create mode 100644 app/data/command/OrderClear.php create mode 100644 app/data/command/UserBalance.php create mode 100644 app/data/controller/ShopPayment.php create mode 100644 app/data/controller/UserBalance.php create mode 100644 app/data/controller/UserNotify.php create mode 100644 app/data/controller/api/auth/Balance.php create mode 100644 app/data/service/payment/BalancePyamentService.php create mode 100644 app/data/service/payment/EmptyPaymentService.php create mode 100644 app/data/view/shop_order_send/config.html create mode 100644 app/data/view/shop_payment/form.html create mode 100644 app/data/view/shop_payment/form_alipay.html create mode 100644 app/data/view/shop_payment/form_joinpay.html create mode 100644 app/data/view/shop_payment/form_wechat.html create mode 100644 app/data/view/shop_payment/index.html create mode 100644 app/data/view/shop_payment/index_search.html create mode 100644 app/data/view/user_balance/form.html create mode 100644 app/data/view/user_balance/index.html create mode 100644 app/data/view/user_balance/index_search.html create mode 100644 app/data/view/user_notify/form.html create mode 100644 app/data/view/user_notify/index.html create mode 100644 app/data/view/user_notify/index_search.html diff --git a/app/data/command/OrderClear.php b/app/data/command/OrderClear.php new file mode 100644 index 000000000..10e85873b --- /dev/null +++ b/app/data/command/OrderClear.php @@ -0,0 +1,87 @@ +setName('xdata:OrderClear'); + $this->setDescription('定时清理商城订单数据'); + } + + /** + * 业务指令执行 + * @param Input $input + * @param Output $output + * @return void + * @throws Exception + */ + protected function execute(Input $input, Output $output) + { + $this->_autoCancelOrder(); + $this->_autoRemoveOrder(); + } + + /** + * 自动取消30分钟未支付的订单 + * @throws Exception + */ + private function _autoCancelOrder() + { + try { + $map = []; + $map[] = ['status', '=', '1']; + $map[] = ['payment_status', '=', '0']; + $map[] = ['create_at', '<', date('Y-m-d H:i:s', strtotime('-30 minutes'))]; + [$count, $used] = [$this->app->db->name('ShopOrder')->where($map)->count(), 0]; + $this->app->db->name('ShopOrder')->where($map)->select()->map(function ($item) use ($count, &$used) { + $this->queue->message($count, ++$used, "开始取消未支付的订单 {$item['order_no']}"); + $this->app->db->name('ShopOrder')->where(['order_no' => $item['order_no']])->update([ + 'status' => '0', + 'cancel_status' => '1', + 'cancel_datetime' => date('Y-m-d H:i:s'), + 'cancel_remark' => '30分钟未完成支付已自动取消', + ]); + OrderService::instance()->syncStock($item['order_no']); + $this->queue->message($count, $used, "完成取消未支付的订单 {$item['order_no']}", 1); + }); + } catch (\Exception $exception) { + $this->queue->error($exception->getMessage()); + } + } + + /** + * 自动清理已经取消的订单 + * @throws Exception + */ + private function _autoRemoveOrder() + { + try { + $map = []; + $map[] = ['status', '=', 0]; + $map[] = ['payment_status', '=', 0]; + $map[] = ['create_at', '<', date('Y-m-d H:i:s', strtotime('-3 days'))]; + [$count, $used] = [$this->app->db->name('ShopOrder')->where($map)->count(), 0]; + $this->app->db->name('ShopOrder')->where($map)->select()->map(function ($item) use ($count, &$used) { + $this->queue->message($count, ++$used, "开始清理已取消的订单 {$item['order_no']}"); + $this->app->db->name('ShopOrder')->where(['order_no' => $item['order_no']])->delete(); + $this->app->db->name('ShopOrderItem')->where(['order_no' => $item['order_no']])->delete(); + $this->queue->message($count, $used, "完成清理已取消的订单 {$item['order_no']}", 1); + }); + } catch (\Exception $exception) { + $this->queue->error($exception->getMessage()); + } + } +} \ No newline at end of file diff --git a/app/data/command/UserBalance.php b/app/data/command/UserBalance.php new file mode 100644 index 000000000..a9fe112c5 --- /dev/null +++ b/app/data/command/UserBalance.php @@ -0,0 +1,43 @@ +setName('xdata:UserBalance'); + $this->setDescription('批量重新计算用户余额'); + } + + /** + * @param Input $input + * @param Output $output + * @return void + * @throws Exception + */ + protected function execute(Input $input, Output $output) + { + try { + [$total, $count] = [$this->app->db->name('DataUser')->count(), 0]; + foreach ($this->app->db->name('DataUser')->field('id')->cursor() as $user) { + $this->queue->message($total, ++$count, "正在计算用户 [{$user['id']}] 的余额"); + UserService::instance()->balance($user['id']); + $this->queue->message($total, $count, "完成计算用户 [{$user['id']}] 的余额"); + } + } catch (\Exception $exception) { + $this->queue->error($exception->getMessage()); + } + } +} \ No newline at end of file diff --git a/app/data/controller/ShopGoods.php b/app/data/controller/ShopGoods.php index c8184793b..896fc4af9 100644 --- a/app/data/controller/ShopGoods.php +++ b/app/data/controller/ShopGoods.php @@ -163,22 +163,21 @@ class ShopGoods extends Controller if (empty($data['slider'])) $this->error('轮播图不能为空!'); // 商品规格保存 $data['mark'] = arr2str($data['mark'] ?? []); - [$count, $items] = [0, json_decode($data['data_items'], true)]; - foreach ($items as $item) { - $count += intval($item[0]['status']); - if (empty($data['price_market'])) $data['price_market'] = $item[0]['market']; - } + [$count, $items] = [0, array_column(json_decode($data['data_items'], true), 0)]; + foreach ($items as $item) $count += intval($item['status']); if (empty($count)) $this->error('无效的的商品价格信息!'); + if (empty($data['price_market'])) $data['price_market'] = min(array_column($items, 'market')); + if (empty($data['price_selling'])) $data['price_selling'] = min(array_column($items, 'selling')); $this->app->db->name('ShopGoodsItem')->where(['goods_code' => $data['code']])->update(['status' => 0]); foreach ($items as $item) data_save('ShopGoodsItem', [ 'goods_code' => $data['code'], - 'goods_sku' => $item[0]['sku'], - 'goods_spec' => $item[0]['key'], - 'price_market' => $item[0]['market'], - 'price_selling' => $item[0]['selling'], - 'number_virtual' => $item[0]['virtual'], - 'number_express' => $item[0]['express'], - 'status' => $item[0]['status'] ? 1 : 0, + 'goods_sku' => $item['sku'], + 'goods_spec' => $item['key'], + 'price_market' => $item['market'], + 'price_selling' => $item['selling'], + 'number_virtual' => $item['virtual'], + 'number_express' => $item['express'], + 'status' => $item['status'] ? 1 : 0, ], 'goods_spec', ['goods_code' => $data['code']]); } } diff --git a/app/data/controller/ShopOrder.php b/app/data/controller/ShopOrder.php index eec8f73f2..eb80f8ff4 100644 --- a/app/data/controller/ShopOrder.php +++ b/app/data/controller/ShopOrder.php @@ -3,7 +3,9 @@ namespace app\data\controller; use app\data\service\OrderService; +use app\data\service\PaymentService; use app\data\service\TruckService; +use app\data\service\UserService; use think\admin\Controller; use think\exception\HttpResponseException; @@ -20,6 +22,18 @@ class ShopOrder extends Controller */ private $table = 'ShopOrder'; + /** + * 支付方式 + * @var array + */ + protected $payments = []; + + protected function initialize() + { + parent::initialize(); + $this->payments = PaymentService::types(); + } + /** * 订单数据管理 * @auth true @@ -46,7 +60,7 @@ class ShopOrder extends Controller $db = $this->_query('ShopOrderSend')->like('address_name#truck_address_name,address_phone#truck_address_phone,address_province|address_city|address_area|address_content#truck_address_content')->db(); if ($db->getOptions('where')) $query->whereRaw("order_no in {$db->field('order_no')->buildSql()}"); // 用户搜索查询 - $db = $this->_query('DataUser')->like('phone#member_phone,nickname#member_nickname')->db(); + $db = $this->_query('DataUser')->like('phone#user_phone,nickname#user_nickname')->db(); if ($db->getOptions('where')) $query->whereRaw("uid in {$db->field('id')->buildSql()}"); // 推荐人搜索查询 $db = $this->_query('DataUser')->like('phone#from_phone,nickname#from_nickname')->db(); @@ -54,7 +68,7 @@ class ShopOrder extends Controller // 列表选项卡 if (is_numeric($this->type = trim(input('type', 'ta'), 't'))) $query->where(['status' => $this->type]); // 分页排序处理 - $query->order('id desc'); + $query->where(['deleted' => 0])->order('id desc'); if (input('output') === 'json') { $this->success('获取数据成功', $query->page(true, false)); } else { @@ -71,7 +85,12 @@ class ShopOrder extends Controller */ protected function _index_page_filter(array &$data) { + UserService::instance()->buildByUid($data); + UserService::instance()->buildByUid($data, 'from', 'fromer'); OrderService::instance()->buildItemData($data); + foreach ($data as &$vo) { + $vo['payment_name'] = PaymentService::name($vo['payment_type']); + } } /** @@ -140,6 +159,15 @@ class ShopOrder extends Controller } } + /** + * 清理订单数据 + * @auth true + */ + public function clean() + { + $this->_queue('定时清理无效订单数据', "xdata:OrderClear", 0, [], 0, 60); + } + /** * 取消未支付的订单 * @auth true @@ -164,6 +192,7 @@ class ShopOrder extends Controller 'cancel_datetime' => date('Y-m-d H:i:s'), ]); if ($result !== false) { + OrderService::instance()->syncStock($order['order_no']); $this->success('取消未支付的订单成功!'); } else { $this->error('取消支付的订单失败!'); diff --git a/app/data/controller/ShopOrderSend.php b/app/data/controller/ShopOrderSend.php index 45283ff16..6659f280f 100644 --- a/app/data/controller/ShopOrderSend.php +++ b/app/data/controller/ShopOrderSend.php @@ -2,6 +2,8 @@ namespace app\data\controller; +use app\data\service\OrderService; +use app\data\service\UserService; use think\admin\Controller; /** @@ -28,6 +30,8 @@ class ShopOrderSend extends Controller public function index() { $this->title = '订单发货管理'; + // 发货地址数据 + $this->address = sysdata('ordersend'); // 状态数据统计 $this->total = ['t0' => 0, 't1' => 0, 't2' => 0, 'ta' => 0]; $this->app->db->name($this->table)->fieldRaw('status,count(1) total')->group('status')->select()->map(function ($vo) { @@ -36,10 +40,10 @@ class ShopOrderSend extends Controller }); // 订单列表查询 $query = $this->_query($this->table)->order('id desc'); - $query->dateBetween('address_datetime,send_datetime')->equal('status')->like('send_number#truck_number'); + $query->dateBetween('address_datetime,send_datetime')->equal('status')->like('send_number#truck_number,order_no'); $query->like('address_phone,address_name,address_province|address_city|address_area|address_content#address_content'); // 用户搜索查询 - $db = $this->_query('DataUser')->like('phone#member_phone,nickname#member_nickname')->db(); + $db = $this->_query('DataUser')->like('phone#user_phone,nickname#user_nickname')->db(); if ($db->getOptions('where')) $query->whereRaw("uid in {$db->field('id')->buildSql()}"); // 订单搜索查询 $db = $this->_query('ShopOrder')->whereIn('status', [3, 4, 5])->db(); @@ -60,12 +64,31 @@ class ShopOrderSend extends Controller /** * 订单列表处理 * @param array $data + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException */ protected function _index_page_filter(array &$data) { - $uids = array_unique(array_column($data, 'uid')); - $users = $this->app->db->name('DataUser')->whereIn('id', $uids)->column('*', 'id'); - foreach ($data as &$vo) $vo['member'] = $users[$vo['uid']] ?? []; + OrderService::instance()->buildItemData($data, false); + } + + /** + * 修改发货地址 + * @auth true + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + */ + public function config() + { + if ($this->request->isGet()) { + $this->vo = sysdata('ordersend'); + $this->fetch(); + } else { + sysdata('ordersend', $this->request->post()); + $this->success('发货地址保存成功'); + } } } \ No newline at end of file diff --git a/app/data/controller/ShopPayment.php b/app/data/controller/ShopPayment.php new file mode 100644 index 000000000..2c30be9fe --- /dev/null +++ b/app/data/controller/ShopPayment.php @@ -0,0 +1,130 @@ +title = '支付参数管理'; + $query = $this->_query($this->table); + $query->where(['deleted' => 0])->order('sort desc,id desc'); + $query->equal('type,status')->like('name')->dateBetween('create_at')->page(); + } + + /** + * 添加支付参数 + * @auth true + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + */ + public function add() + { + $this->title = '添加支付参数'; + $this->_form($this->table, 'form'); + } + + /** + * 编辑支付参数 + * @auth true + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + */ + public function edit() + { + $this->title = '编辑支付参数'; + $this->_form($this->table, 'form'); + } + + /** + * 数据表单处理 + * @param array $data + */ + protected function _form_filter(array &$data) + { + if (empty($data['code'])) { + $data['code'] = CodeExtend::uniqidNumber(14, 'M'); + } + if ($this->request->isGet()) { + $this->payments = []; + foreach ($this->types as $k => $vo) { + $allow = []; + foreach ($vo['bind'] as $api) if (isset(UserService::TYPES[$api])) { + $allow[$api] = UserService::TYPES[$api]['name']; + } + $this->payments[$k] = array_merge($vo, ['allow' => join('、', $allow)]); + } + $data['content'] = json_decode($data['content'] ?? '[]', true) ?: []; + } else { + if (empty($data['type'])) $this->error('请选择支付参数并配置支付参数!'); + $data['content'] = json_encode($this->request->post() ?: [], JSON_UNESCAPED_UNICODE); + } + } + + /** + * 表单结果处理 + * @param boolean $state + */ + protected function _form_result(bool $state) + { + if ($state) { + $this->success('支付参数保存成功!', 'javascript:history.back()'); + } + } + + /** + * 修改支付参数状态 + * @auth true + * @throws \think\db\exception\DbException + */ + public function state() + { + $this->_save($this->table, $this->_vali([ + 'status.in:0,1' => '状态值范围异常!', + 'status.require' => '状态值不能为空!', + ])); + } + + /** + * 删除支付参数 + * @auth true + * @throws \think\db\exception\DbException + */ + public function remove() + { + $this->_delete($this->table); + } + +} \ No newline at end of file diff --git a/app/data/controller/UserBalance.php b/app/data/controller/UserBalance.php new file mode 100644 index 000000000..1325398c4 --- /dev/null +++ b/app/data/controller/UserBalance.php @@ -0,0 +1,124 @@ +title = '余额充值管理'; + $query = $this->_query($this->table); + // 用户搜索查询 + $db = $this->_query('DataUser')->like('phone#user_phone,nickname#user_nickname')->db(); + if ($db->getOptions('where')) $query->whereRaw("uid in {$db->field('id')->buildSql()}"); + // 数据查询分页 + $query->where(['deleted' => 0])->like('code,name')->dateBetween('create_at')->order('id desc')->page(); + } + + /** + * 数据列表处理 + * @param array $data + */ + protected function _index_page_filter(array &$data) + { + UserService::instance()->buildByUid($data); + $uids = array_unique(array_column($data, 'create_by')); + $users = $this->app->db->name('SystemUser')->whereIn('id', $uids)->column('username', 'id'); + foreach ($data as &$vo) $vo['create_byname'] = $users[$vo['create_by']] ?? $vo['create_by']; + } + + /** + * 添加余额充值 + * @auth true + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + */ + public function add() + { + $data = $this->_vali(['uid.require' => '用户UID不能为空!']); + $this->user = $this->app->db->name('DataUser')->where(['id' => $data['uid']])->find(); + if (empty($this->user)) $this->error('待充值的用户不存在!'); + $this->_form($this->table, 'form'); + } + + /** + * 表单数据处理 + * @param array $data + */ + protected function _form_filter(array &$data) + { + if (empty($data['code'])) { + $data['code'] = CodeExtend::uniqidDate('16', 'B'); + } + if ($this->request->isPost()) { + if ($data['amount'] <= 0) { + $this->error('充值金额不能少于零'); + } + $data['create_by'] = AdminService::instance()->getUserId(); + } + } + + /** + * 表单结果处理 + * @param bool $state + * @param array $data + * @throws \think\db\exception\DbException + */ + protected function _form_result(bool $state, array $data) + { + if ($state && isset($data['uid'])) { + UserService::instance()->balance($data['uid']); + } + } + + /** + * 删除充值记录 + * @auth true + * @throws \think\db\exception\DbException + */ + public function remove() + { + $this->_delete($this->table); + } + + /** + * 删除结果处理 + * @param bool $state + * @throws \think\db\exception\DbException + */ + protected function _delete_result(bool $state) + { + if ($state) { + $ids = str2arr(input('id', '')); + $query = $this->app->db->name($this->table); + foreach ($query->whereIn('id', $ids)->cursor() as $vo) { + UserService::instance()->balance($vo['uid']); + } + } + } +} \ No newline at end of file diff --git a/app/data/controller/UserNotify.php b/app/data/controller/UserNotify.php new file mode 100644 index 000000000..9c65656d8 --- /dev/null +++ b/app/data/controller/UserNotify.php @@ -0,0 +1,94 @@ +title = '系统通知管理'; + $query = $this->_query($this->table); + $query->like('name')->equal('status')->dateBetween('create_at'); + $query->where(['deleted' => 0])->order('sort desc,id desc')->page(); + } + + /** + * 添加系统通知 + * @auth true + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + */ + public function add() + { + $this->_form($this->table, 'form'); + } + + /** + * 编辑系统通知 + * @auth true + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + */ + public function edit() + { + $this->_form($this->table, 'form'); + } + + /** + * 表单结果处理 + * @param boolean $state + */ + protected function _form_result(bool $state) + { + if ($state) { + $this->success('内容保存成功!', 'javascript:history.back()'); + } + } + + /** + * 修改系统通知状态 + * @auth true + * @throws \think\db\exception\DbException + */ + public function state() + { + $this->_save($this->table, $this->_vali([ + 'status.in:0,1' => '状态值范围异常!', + 'status.require' => '状态值不能为空!', + ])); + } + + /** + * 删除系统通知 + * @auth true + * @throws \think\db\exception\DbException + */ + public function remove() + { + $this->_delete($this->table); + } + +} \ No newline at end of file diff --git a/app/data/controller/api/Auth.php b/app/data/controller/api/Auth.php index 7208d5cdf..524ee1255 100644 --- a/app/data/controller/api/Auth.php +++ b/app/data/controller/api/Auth.php @@ -41,10 +41,13 @@ abstract class Auth extends Controller */ protected function initialize() { - $this->type = input('api', UserService::APITYPE_WXAPP); + // 接口数据类型 + $this->type = input('api') ?: $this->request->header('api-type'); + $this->type = $this->type ?: UserService::APITYPE_WXAPP; if (empty(UserService::TYPES[$this->type])) { $this->error("接口通道[{$this->type}]未定义规则!"); } + // 获取用户数据 $this->user = $this->getUser(); $this->uuid = $this->user['id']; } @@ -56,14 +59,14 @@ abstract class Auth extends Controller protected function getUser(): array { try { - $service = UserService::instance(); + $user = UserService::instance(); if (empty($this->uuid)) { - $token = input('token') ?: $this->request->header('token'); - if (empty($token)) $this->error('登录认证令牌不能为空!'); - [$state, $info, $this->uuid] = $service->checkUserToken($this->type, $token); + $token = input('token') ?: $this->request->header('api-token'); + if (empty($token)) $this->error('登录认证TOKEN不能为空!'); + [$state, $info, $this->uuid] = $user->check($this->type, $token); if (empty($state)) $this->error($info, '{-null-}', 401); } - return $service->get($this->type, $this->uuid); + return $user->get($this->type, $this->uuid); } catch (HttpResponseException $exception) { throw $exception; } catch (\Exception $exception) { diff --git a/app/data/controller/api/Data.php b/app/data/controller/api/Data.php index 97afa171d..c826140d8 100644 --- a/app/data/controller/api/Data.php +++ b/app/data/controller/api/Data.php @@ -22,4 +22,17 @@ class Data extends Controller $data = sysdata(input('keys', 'slider')); $this->success('获取轮播图片数据', $data); } + + /** + * 获取系统通知数据 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + */ + public function getNotify() + { + $query = $this->_query('DataUserNotify')->where(['status' => 1, 'deleted' => 0]); + $result = $query->equal('id')->order('sort desc,id desc')->page(true, false, false, 20); + $this->success('获取系统通知数据', $result); + } } \ No newline at end of file diff --git a/app/data/controller/api/Notify.php b/app/data/controller/api/Notify.php index 6c7f32a6c..35d17362e 100644 --- a/app/data/controller/api/Notify.php +++ b/app/data/controller/api/Notify.php @@ -17,7 +17,7 @@ class Notify extends Controller /** * 微信支付通知 * @param string $scene 支付场景 - * @param string $param 支付通道 + * @param string $param 支付参数 * @return string * @throws \WeChat\Exceptions\InvalidResponseException * @throws \think\Exception @@ -37,7 +37,7 @@ class Notify extends Controller /** * 支付宝支付通知 * @param string $scene 支付场景 - * @param string $param 支付通道 + * @param string $param 支付参数 * @return string * @throws \WeChat\Exceptions\InvalidResponseException * @throws \think\Exception @@ -57,7 +57,7 @@ class Notify extends Controller /** * 汇聚支付通知 * @param string $scene 支付场景 - * @param string $param 支付通道 + * @param string $param 支付参数 * @return string * @throws \WeChat\Exceptions\InvalidResponseException * @throws \think\Exception diff --git a/app/data/controller/api/auth/Address.php b/app/data/controller/api/auth/Address.php index 0b320115e..21cbeb130 100644 --- a/app/data/controller/api/auth/Address.php +++ b/app/data/controller/api/auth/Address.php @@ -45,12 +45,12 @@ class Address extends Auth if ($count > 0) $this->error('抱歉,该地址已经存在!'); $data['code'] = CodeExtend::uniqidDate(12, 'A'); if ($this->app->db->name($this->table)->insert($data) === false) { - $this->error('添加收货地址失败!'); + $this->error('添加地址失败!'); } } else { $map = ['uid' => $this->uuid, 'code' => $data['code']]; $address = $this->app->db->name($this->table)->where($map)->find(); - if (empty($address)) $this->error('修改收货地址不存在!'); + if (empty($address)) $this->error('修改地址不存在!'); $this->app->db->name($this->table)->where($map)->update($data); } // 去除其它默认选项 @@ -58,7 +58,7 @@ class Address extends Auth $map = [['uid', '=', $this->uuid], ['code', '<>', $data['code']]]; $this->app->db->name($this->table)->where($map)->update(['type' => 0]); } - $this->success('添加收货地址成功!', $this->_getAddress($data['code'])); + $this->success('地址保存成功!', $this->_getAddress($data['code'])); } /** @@ -72,7 +72,7 @@ class Address extends Auth $query = $this->_query($this->table)->withoutField('deleted'); $query->equal('code')->where(['uid' => $this->uuid, 'deleted' => 0]); $result = $query->order('type desc,id desc')->page(false, false, false, 15); - $this->success('获取收货地址数据!', $result); + $this->success('获取地址数据!', $result); } /** diff --git a/app/data/controller/api/auth/Balance.php b/app/data/controller/api/auth/Balance.php new file mode 100644 index 000000000..c1b5fd8ee --- /dev/null +++ b/app/data/controller/api/auth/Balance.php @@ -0,0 +1,75 @@ +_query($this->table); + $query->where(['uid|from' => $this->uuid, 'deleted' => 0]); + $result = $query->order('id desc')->page(true, false, false, 15); + if (count($result['list']) > 0) { + UserService::instance()->buildByUid($result['list'], 'uid', 'selfer'); + UserService::instance()->buildByUid($result['list'], 'from', 'fromer'); + } + $this->success('获取数据成功', $result); + } + + /** + * 创建余额转账申请 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + */ + public function add() + { + $data = $this->_vali([ + 'from.value' => $this->uuid, + 'code.value' => CodeExtend::uniqidDate(18, 'T'), + 'uid.require' => '目标用户不能为空!', + 'name.default' => '用户余额转账', + 'amount.require' => '转账金额不能为空!', + ]); + if ($data['uid'] == $this->uuid) { + $this->error('不能给自己转账!'); + } + // 检测目标用户状态 + $map = ['id' => $data['uid'], 'deleted' => 0]; + $user = $this->app->db->name('DataUser')->where($map)->find(); + if (empty($user)) $this->error('目标用户不存在!'); + // 检测用户是否有足够的余额 + [$total, $used] = UserService::instance()->balance($this->uuid); + if ($data['amount'] > $total - $used) $this->error('可转账余额不足!'); + // 写入余额转账记录 + if ($this->app->db->name($this->table)->insert($data) !== false) { + UserService::instance()->balance($data['uid']); + UserService::instance()->balance($data['from']); + $this->success('余额转账成功!'); + } else { + $this->error('余额转账失败!'); + } + } +} \ No newline at end of file diff --git a/app/data/controller/api/auth/Center.php b/app/data/controller/api/auth/Center.php index 0052301cf..a15e013c7 100644 --- a/app/data/controller/api/auth/Center.php +++ b/app/data/controller/api/auth/Center.php @@ -54,14 +54,6 @@ class Center extends Auth $this->success('获取用户资料', $this->getUser()); } - /** - * 获取用户数据统计 - */ - public function total() - { - $this->success('获取用户统计!', UserService::instance()->total($this->uuid)); - } - /** * Base64 图片上传 */ @@ -106,6 +98,32 @@ class Center extends Auth } } + /** + * 获取我邀请的朋友 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + */ + public function getFrom() + { + $where = []; + $where[] = ['deleted', '=', '0']; + $where[] = ['path', 'like', "%-{$this->uuid}-%"]; + // 查询邀请的朋友 + $query = $this->_query($this->table); + $query->like('nickname|username#nickname')->equal('from,id#uid'); + $query->field('id,from,username,nickname,headimg,amount_total,create_at'); + $result = $query->where($where)->order('id desc')->page(true, false, false, 15); + // 统计当前用户所有下属数 + $userTotal = $this->app->db->name($this->table)->where($where)->count(); + // 统计当前用户本月下属数 + $where[] = ['create_at', 'like', date('Y-m-%')]; + $userMonth = $this->app->db->name($this->table)->where($where)->count(); + // 返回结果列表数据及统计 + $result['total'] = ['user_total' => $userTotal, 'user_month' => $userMonth]; + $this->success('获取我邀请的朋友', $result); + } + /** * 绑定用户邀请人 * @throws \think\Exception @@ -122,23 +140,13 @@ class Center extends Auth $from = $this->app->db->name($this->table)->where(['id' => $data['from']])->find(); if (empty($from)) $this->error('邀请人状态异常', UserService::instance()->get($this->type, $this->uuid)); if ($this->user['from'] > 0) $this->error('已绑定了邀请人', UserService::instance()->total($this->uuid)); + if (is_numeric(stripos($from['path'], "-{$this->uuid}-"))) $this->error('不能绑定下属'); $data['path'] = rtrim($from['path'] ?: '-', '-') . '-' . $from['id'] . '-'; + $data['layer'] = substr_count($data['path'], '-'); if ($this->app->db->name($this->table)->where(['id' => $this->uuid])->update($data) !== false) { $this->success('绑定邀请人成功', UserService::instance()->total($this->uuid)); } else { $this->error('绑定邀请人失败', UserService::instance()->total($this->uuid)); } } - - /** - * 获取我邀请的朋友 - * @throws \think\db\exception\DataNotFoundException - * @throws \think\db\exception\DbException - * @throws \think\db\exception\ModelNotFoundException - */ - public function getFrom() - { - $query = $this->_query($this->table)->field('id,from,username,nickname,headimg,create_at'); - $this->success('获取我邀请的朋友', $query->where(['from' => $this->uuid])->order('id desc')->page(true, false, false, 15)); - } } \ No newline at end of file diff --git a/app/data/controller/api/auth/Config.php b/app/data/controller/api/auth/Config.php index 9fcd022e9..d26e04a4b 100644 --- a/app/data/controller/api/auth/Config.php +++ b/app/data/controller/api/auth/Config.php @@ -13,7 +13,7 @@ use app\data\service\PaymentService; class Config extends Auth { /** - * 获取支付通道数据 + * 获取支付参数数据 * @throws \think\db\exception\DataNotFoundException * @throws \think\db\exception\DbException * @throws \think\db\exception\ModelNotFoundException @@ -27,8 +27,8 @@ class Config extends Auth } } $map = ['status' => 1, 'deleted' => 0]; - $query = $this->app->db->name('DataPayment')->where($map)->whereIn('type', $types); + $query = $this->app->db->name('ShopPayment')->where($map)->whereIn('type', $types); $collect = $query->order('sort desc,id desc')->field('code,name,type')->select(); - $this->success('获取支付通道数据', $collect->toArray()); + $this->success('获取支付参数数据', $collect->toArray()); } } \ No newline at end of file diff --git a/app/data/controller/api/auth/Order.php b/app/data/controller/api/auth/Order.php index 29324b2e5..9e9c2bc1b 100644 --- a/app/data/controller/api/auth/Order.php +++ b/app/data/controller/api/auth/Order.php @@ -37,10 +37,7 @@ class Order extends Auth */ public function get() { - $map = [['uid', '=', $this->uuid]]; - if (!$this->request->has('order_no', 'param', true)) { - $map[] = ['status', 'in', [0, 2, 3, 4, 5]]; - } + $map = ['uid' => $this->uuid, 'deleted' => 0]; $query = $this->_query('ShopOrder')->in('status')->equal('order_no'); $result = $query->where($map)->order('id desc')->page(true, false, false, 20); if (count($result['list']) > 0) OrderService::instance()->buildItemData($result['list']); @@ -57,32 +54,31 @@ class Order extends Auth { // 商品规则 $rules = $this->request->post('items', ''); - if (empty($rules)) $this->error('商品规则不能为空!'); + if (empty($rules)) $this->error('商品不能为空'); // 订单数据 [$codes, $items] = [[], []]; - $order = ['uid' => $this->uuid, 'from' => input('from_mid', '0'), 'status' => 1]; + $order = ['uid' => $this->uuid, 'status' => 1]; $order['order_no'] = CodeExtend::uniqidDate(18, 'N'); // 推荐人处理 - if ($order['from'] == $this->uuid) { - $order['from'] = 0; - } + $order['from'] = input('from_uid', $this->user['from']); + if ($order['from'] == $this->uuid) $order['from'] = 0; if ($order['from'] > 0) { $map = ['id' => $order['from'], 'status' => 1]; - $from = $this->app->db->name('ShopMember')->where($map)->find(); - if (empty($from)) $this->error('推荐人信息异常!'); + $from = $this->app->db->name('DataUser')->where($map)->find(); + if (empty($from)) $this->error('推荐人异常'); } foreach (explode('||', $rules) as $rule) { [$code, $spec, $count] = explode('@', $rule); // 商品信息检查 $map = ['code' => $code, 'status' => 1, 'deleted' => 0]; $goodsInfo = $this->app->db->name('ShopGoods')->where($map)->find(); - if (empty($goodsInfo)) $this->error('商品主体异常,请稍候再试!'); + if (empty($goodsInfo)) $this->error('商品数据异常'); $map = ['goods_code' => $code, 'goods_spec' => $spec, 'status' => 1]; $goodsItem = $this->app->db->name('ShopGoodsItem')->where($map)->find(); - if (empty($goodsItem)) $this->error('商品规格异常,请稍候再试!'); + if (empty($goodsItem)) $this->error('商品规格异常'); // 商品库存检查 if ($goodsItem['stock_sales'] + $count > $goodsItem['stock_total']) { - $this->error('商品库存不足,请购买其它商品!'); + $this->error('商品库存不足'); } // 订单详情处理 $items[] = [ @@ -106,29 +102,34 @@ class Order extends Auth ]; } try { - // 统计订单商品 - $order['amount_reduct'] = OrderService::instance()->getReduct(); - // 统计订单金额 + // 统计商品数量 $order['number_goods'] = array_sum(array_column($items, 'stock_sales')); + // 统计商品金额 $order['amount_goods'] = array_sum(array_column($items, 'total_selling')); - $order['amount_total'] = $order['amount_goods'] - $order['amount_reduct']; - // 支付金额不能为零 - if ($order['amount_total'] <= 0) $order['amount_total'] = 0.01; + // 订单随机免减 + $order['amount_reduct'] = OrderService::instance()->getReduct(); + if ($order['amount_reduct'] > $order['amount_goods']) { + $order['amount_reduct'] = $order['amount_goods']; + } + // 统计订单金额 + $order['amount_real'] = $order['amount_goods'] - $order['amount_reduct']; + $order['amount_total'] = $order['amount_goods']; + // 写入订单商品数据 $this->app->db->name('ShopOrder')->insert($order); $this->app->db->name('ShopOrderItem')->insertAll($items); // 同步商品库存销量 foreach ($codes as $code) GoodsService::instance()->syncStock($code); - // 返回订单数据接口 - $order['items'] = $items; // 触发订单创建事件 $this->app->event->trigger('ShopOrderCreate', $order['order_no']); + // 组装订单商品数据 + $order['items'] = $items; // 返回处理成功数据 - $this->success('预购订单创建成功,请补全收货地址', $order); + $this->success('商品下单成功', $order); } catch (HttpResponseException $exception) { throw $exception; } catch (\Exception $exception) { - $this->error("创建订单失败,{$exception->getMessage()}"); + $this->error("商品下单失败,{$exception->getMessage()}"); } } @@ -141,13 +142,13 @@ class Order extends Auth public function express() { $data = $this->_vali([ - 'code.require' => '收货地址不能为空!', - 'order_no.require' => '订单单号不能为空!', + 'code.require' => '地址不能为空', + 'order_no.require' => '单号不能为空', ]); // 用户收货地址 $map = ['uid' => $this->uuid, 'code' => $data['code']]; $addr = $this->app->db->name('DataUserAddress')->where($map)->find(); - if (empty($addr)) $this->error('用户收货地址异常!'); + if (empty($addr)) $this->error('收货地址异常'); // 订单状态检查 $map = ['uid' => $this->uuid, 'order_no' => $data['order_no']]; $tCount = $this->app->db->name('ShopOrderItem')->where($map)->sum('truck_count'); @@ -167,18 +168,18 @@ class Order extends Auth public function perfect() { $data = $this->_vali([ - 'code.require' => '地址编号不能为空!', - 'order_no.require' => '订单单号不能为空!', + 'code.require' => '地址不能为空', + 'order_no.require' => '单号不能为空', ]); // 用户收货地址 $map = ['uid' => $this->uuid, 'code' => $data['code'], 'deleted' => 0]; $addr = $this->app->db->name('DataUserAddress')->where($map)->find(); - if (empty($addr)) $this->error('用户收货地址异常!'); + if (empty($addr)) $this->error('收货地址异常'); // 订单状态检查 $map = ['uid' => $this->uuid, 'order_no' => $data['order_no']]; $order = $this->app->db->name('ShopOrder')->where($map)->whereIn('status', [1, 2])->find(); $tCount = $this->app->db->name('ShopOrderItem')->where($map)->sum('truck_count'); - if (empty($order)) $this->error('不能修改收货地址哦!'); + if (empty($order)) $this->error('不能修改地址'); // 根据地址计算运费 $map = ['status' => 1, 'deleted' => 0, 'order_no' => $data['order_no']]; $tCodes = $this->app->db->name('ShopOrderItem')->where($map)->column('truck_tcode'); @@ -192,25 +193,30 @@ class Order extends Auth $express['address_code'] = $data['code']; $express['address_name'] = $addr['name']; $express['address_phone'] = $addr['phone']; + $express['address_idcode'] = $addr['idcode']; $express['address_province'] = $addr['province']; $express['address_city'] = $addr['city']; $express['address_area'] = $addr['area']; $express['address_content'] = $addr['address']; $express['address_datetime'] = date('Y-m-d H:i:s'); data_save('ShopOrderSend', $express, 'order_no'); - // 更新订单状态,刷新订单金额 - $map = ['uid' => $this->uuid, 'order_no' => $data['order_no']]; + // 组装更新订单数据 $update = ['status' => 2, 'amount_express' => $express['template_amount']]; - $update['amount_total'] = $order['amount_goods'] + $amount - $order['amount_reduct'] - $order['amount_discount']; + // 重新计算订单金额 + $update['amount_real'] = $order['amount_goods'] + $amount - $order['amount_reduct'] - $order['amount_discount']; + $update['amount_total'] = $order['amount_goods'] + $amount; // 支付金额不能为零 - if ($update['amount_total'] <= 0) $update['amount_total'] = 0.01; + if ($update['amount_real'] <= 0) $update['amount_real'] = 0.00; + if ($update['amount_total'] <= 0) $update['amount_total'] = 0.00; + // 更新用户订单数据 + $map = ['uid' => $this->uuid, 'order_no' => $data['order_no']]; if ($this->app->db->name('ShopOrder')->where($map)->update($update) !== false) { // 触发订单确认事件 $this->app->event->trigger('ShopOrderPerfect', $order['order_no']); // 返回处理成功数据 - $this->success('订单确认成功!', ['order_no' => $order['order_no']]); + $this->success('订单确认成功', ['order_no' => $order['order_no']]); } else { - $this->error('订单确认失败,请稍候再试!'); + $this->error('订单确认失败'); } } @@ -223,27 +229,30 @@ class Order extends Auth public function payment() { $data = $this->_vali([ - 'order_no.require' => '订单单号不能为空!', - 'payment_code.require' => '支付通道不能为空!', + 'order_no.require' => '单号不能为空', + 'payment_code.require' => '参数不能为空', 'payment_back.default' => '', # 支付回跳地址 ]); $map = ['order_no' => $data['order_no']]; $order = $this->app->db->name('ShopOrder')->where($map)->find(); - if (empty($order)) $this->error('获取订单数据失败!'); - if ($order['status'] != 2) $this->error('订单不能发起支付哦!'); - if ($order['payment_status'] > 0) $this->error('订单已经完成支付!'); + if (empty($order)) $this->error('读取订单失败'); + if ($order['status'] != 2) $this->error('不能发起支付'); + if ($order['payment_status'] > 0) $this->error('已经完成支付'); try { $openid = ''; if (in_array($this->type, [UserService::APITYPE_WXAPP, UserService::APITYPE_WECHAT])) { $openid = $this->user[UserService::TYPES[$this->type]['auth']] ?? ''; - if (empty($openid)) $this->error("无法创建支付,未获取到OPENID"); + if (empty($openid)) $this->error("无法创建支付"); } - $params = PaymentService::instance($data['payment_code'])->create($openid, $order['order_no'], $order['amount_total'], '商城订单支付', '', $data['payment_back']); - $this->success('获取支付参数成功!', $params); + // 返回订单数据及支付发起参数 + $type = $order['amount_real'] <= 0 ? 'empty' : $data['payment_code']; + $param = PaymentService::instance($type)->create($openid, $order['order_no'], $order['amount_real'], '商城订单支付', '', $data['payment_back']); + $order = $this->app->db->name('ShopOrder')->where($map)->find() ?: new \stdClass(); + $this->success('获取支付参数', ['order' => $order, 'param' => $param]); } catch (HttpResponseException $exception) { - throw $exception; + throw $exception; } catch (\Exception $exception) { - $this->error("创建支付参数失败,{$exception->getMessage()}"); + $this->error($exception->getMessage()); } } @@ -257,10 +266,10 @@ class Order extends Auth { $map = $this->_vali([ 'uid.value' => $this->uuid, - 'order_no.require' => '订单号不能为空!', + 'order_no.require' => '单号不能为空', ]); $order = $this->app->db->name('ShopOrder')->where($map)->find(); - if (empty($order)) $this->error('订单查询失败,请稍候再试!'); + if (empty($order)) $this->error('读取订单失败'); if (in_array($order['status'], [1, 2])) { $result = $this->app->db->name('ShopOrder')->where($map)->update([ 'status' => 0, @@ -272,12 +281,46 @@ class Order extends Auth // 触发订单取消事件 $this->app->event->trigger('ShopOrderCancel', $order['order_no']); // 返回处理成功数据 - $this->success('订单取消成功!'); + $this->success('订单取消成功'); } else { - $this->error('订单取消失败,请稍候再试!'); + $this->error('订单取消失败'); } } else { - $this->error('该订单状态不能取消哦~'); + $this->error('订单不可取消'); + } + } + + /** + * 用户主动删除已取消的订单 + * @throws \think\db\exception\DataNotFoundException + * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException + */ + public function remove() + { + $map = $this->_vali([ + 'uid.value' => $this->uuid, + 'order_no.require' => '单号不能为空', + ]); + $order = $this->app->db->name('ShopOrder')->where($map)->find(); + if (empty($order)) $this->error('读取订单失败'); + if (in_array($order['status'], [0])) { + $result = $this->app->db->name('ShopOrder')->where($map)->update([ + 'status' => 0, + 'deleted' => 1, + 'deleted_remark' => '用户主动删除订单!', + 'deleted_datetime' => date('Y-m-d H:i:s'), + ]); + if ($result !== false) { + // 触发订单删除事件 + $this->app->event->trigger('ShopOrderRemove', $order['order_no']); + // 返回处理成功数据 + $this->success('订单删除成功'); + } else { + $this->error('订单删除失败'); + } + } else { + $this->error('订单不可删除'); } } @@ -291,21 +334,21 @@ class Order extends Auth { $map = $this->_vali([ 'uid.value' => $this->uuid, - 'order_no.require' => '订单号不能为空!', + 'order_no.require' => '单号不能为空', ]); $order = $this->app->db->name('ShopOrder')->where($map)->find(); - if (empty($order)) $this->error('订单查询失败,请稍候再试!'); + if (empty($order)) $this->error('读取订单失败'); if (in_array($order['status'], [4])) { if ($this->app->db->name('ShopOrder')->where($map)->update(['status' => 5]) !== false) { // 触发订单确认事件 $this->app->event->trigger('ShopOrderConfirm', $order['order_no']); // 返回处理成功数据 - $this->success('订单确认成功!'); + $this->success('订单确认成功'); } else { - $this->error('订单确认失败,请稍候再试!'); + $this->error('订单确认失败'); } } else { - $this->error('订单不能确认收货哦~'); + $this->error('订单确认失败'); } } @@ -323,7 +366,7 @@ class Order extends Auth $query->where($map)->group('status')->select()->each(function ($item) use (&$data) { $data["t{$item['status']}"] = $item['count']; }); - $this->success('获取状态统计成功!', $data); + $this->success('获取统计成功', $data); } /** @@ -333,8 +376,8 @@ class Order extends Auth { try { $data = $this->_vali([ - 'code.require' => '快递编号不能为空!', - 'number.require' => '配送单号不能为空!', + 'code.require' => '快递不能为空', + 'number.require' => '单号不能为空', ]); $result = TruckService::instance()->query($data['code'], $data['number']); empty($result['code']) ? $this->error($result['info']) : $this->success('快递追踪信息', $result); diff --git a/app/data/service/NewsService.php b/app/data/service/NewsService.php index 9795ff6db..7f868c8a0 100644 --- a/app/data/service/NewsService.php +++ b/app/data/service/NewsService.php @@ -46,7 +46,7 @@ class NewsService extends Service foreach ($list as &$vo) $vo['record'] = $items[$vo['code']] ?? []; /*! 绑定用户数据 */ $colls = 'id,phone,nickname,username,headimg,status'; - UserService::instance()->buildByUid($list, 'uid', 'member', $colls); + UserService::instance()->buildByUid($list, 'uid', 'user', $colls); } return $list; } diff --git a/app/data/service/OrderService.php b/app/data/service/OrderService.php index 617e2ae87..ce03f7de4 100644 --- a/app/data/service/OrderService.php +++ b/app/data/service/OrderService.php @@ -11,17 +11,6 @@ use think\admin\Service; */ class OrderService extends Service { - /** - * 同步订单支付状态 - * @param string $orderno - * @return bool - */ - public function syncAmount(string $orderno): bool - { - //@todo 处理订单支付完成的动作 - return true; - } - /** * 获取随机减免金额 * @return float @@ -50,12 +39,13 @@ class OrderService extends Service /** * 绑定订单详情数据 * @param array $data + * @param bool $fromer * @return array * @throws \think\db\exception\DataNotFoundException * @throws \think\db\exception\DbException * @throws \think\db\exception\ModelNotFoundException */ - public function buildItemData(array &$data = []): array + public function buildItemData(array &$data = [], $fromer = true): array { // 关联发货信息 $nobs = array_unique(array_column($data, 'order_no')); @@ -66,8 +56,8 @@ class OrderService extends Service $items = $query->withoutField('id,uid,status,deleted,create_at')->whereIn('order_no', $nobs)->select()->toArray(); // 关联用户数据 $fields = 'username,phone,nickname,headimg,status'; - UserService::instance()->buildByUid($data, 'uid', 'member', $fields); - UserService::instance()->buildByUid($data, 'from', 'fromer', $fields); + UserService::instance()->buildByUid($data, 'uid', 'user', $fields); + if ($fromer) UserService::instance()->buildByUid($data, 'from', 'fromer', $fields); foreach ($data as &$vo) { $vo['sales'] = 0; $vo['truck'] = $trucks[$vo['order_no']] ?? []; diff --git a/app/data/service/PaymentService.php b/app/data/service/PaymentService.php index 782cf8319..5653d7750 100644 --- a/app/data/service/PaymentService.php +++ b/app/data/service/PaymentService.php @@ -3,6 +3,8 @@ namespace app\data\service; use app\data\service\payment\AlipayPaymentService; +use app\data\service\payment\BalancePyamentService; +use app\data\service\payment\EmptyPaymentService; use app\data\service\payment\JoinPaymentService; use app\data\service\payment\WechatPaymentService; use think\App; @@ -17,7 +19,11 @@ use think\Exception; abstract class PaymentService { - // 汇聚支付通道 + // 用户余额支付 + const PAYMENT_EMPTY = 'empty'; + const PAYMENT_BALANCE = 'balance'; + + // 汇聚支付参数 const PAYMENT_JOINPAY_GZH = 'joinpay_gzh'; const PAYMENT_JOINPAY_XCX = 'joinpay_xcx'; @@ -28,64 +34,78 @@ abstract class PaymentService const PAYMENT_WECHAT_WAP = 'wechat_wap'; const PAYMENT_WECHAT_QRC = 'wechat_qrc'; - // 支付宝支付通道 + // 支付宝支付参数 const PAYMENT_ALIAPY_APP = 'alipay_app'; const PAYMENT_ALIPAY_WAP = 'alipay_wap'; const PAYMENT_ALIPAY_WEB = 'alipay_web'; - // 支付通道配置 + // 支付参数配置 const TYPES = [ // 微信支付配置(不需要的直接注释) - PaymentService::PAYMENT_WECHAT_WAP => [ + self::PAYMENT_EMPTY => [ + 'type' => 'EMPTY', + 'name' => '订单无需支付', + 'bind' => [], + ], + self::PAYMENT_BALANCE => [ + 'type' => 'BALANCE', + 'name' => '账号余额支付', + 'bind' => [ + UserService::APITYPE_WAP, UserService::APITYPE_WEB, + UserService::APITYPE_WXAPP, UserService::APITYPE_WECHAT, + UserService::APITYPE_IOSAPP, UserService::APITYPE_ANDROID, + ], + ], + self::PAYMENT_WECHAT_WAP => [ 'type' => 'MWEB', - 'name' => '微信商户 H5 支付', + 'name' => '微信WAP支付', 'bind' => [UserService::APITYPE_WAP], ], - PaymentService::PAYMENT_WECHAT_APP => [ + self::PAYMENT_WECHAT_APP => [ 'type' => 'APP', - 'name' => '微信商户 APP 支付', + 'name' => '微信APP支付', 'bind' => [UserService::APITYPE_IOSAPP, UserService::APITYPE_ANDROID], ], - PaymentService::PAYMENT_WECHAT_XCX => [ + self::PAYMENT_WECHAT_XCX => [ 'type' => 'JSAPI', - 'name' => '微信商户 小程序 支付', + 'name' => '微信小程序支付', 'bind' => [UserService::APITYPE_WXAPP], ], - PaymentService::PAYMENT_WECHAT_GZH => [ + self::PAYMENT_WECHAT_GZH => [ 'type' => 'JSAPI', - 'name' => '微信商户 公众号 支付', + 'name' => '微信公众号支付', 'bind' => [UserService::APITYPE_WECHAT], ], - PaymentService::PAYMENT_WECHAT_QRC => [ + self::PAYMENT_WECHAT_QRC => [ 'type' => 'NATIVE', - 'name' => '微信商户 二维码 支付', + 'name' => '微信二维码支付', 'bind' => [UserService::APITYPE_WEB], ], // 支付宝支持配置(不需要的直接注释) - PaymentService::PAYMENT_ALIPAY_WAP => [ + self::PAYMENT_ALIPAY_WAP => [ 'type' => '', - 'name' => '支付宝 WAP 支付', + 'name' => '支付宝WAP支付', 'bind' => [UserService::APITYPE_WAP], ], - PaymentService::PAYMENT_ALIPAY_WEB => [ + self::PAYMENT_ALIPAY_WEB => [ 'type' => '', - 'name' => '支付宝 WEB 支付', + 'name' => '支付宝WEB支付', 'bind' => [UserService::APITYPE_WEB], ], - PaymentService::PAYMENT_ALIAPY_APP => [ + self::PAYMENT_ALIAPY_APP => [ 'type' => '', - 'name' => '支付宝 APP 支付', + 'name' => '支付宝APP支付', 'bind' => [UserService::APITYPE_ANDROID, UserService::APITYPE_IOSAPP], ], // 汇聚支持配置(不需要的直接注释) - PaymentService::PAYMENT_JOINPAY_XCX => [ + self::PAYMENT_JOINPAY_XCX => [ 'type' => 'WEIXIN_XCX', - 'name' => '汇聚 小程序 支付', + 'name' => '汇聚小程序支付', 'bind' => [UserService::APITYPE_WXAPP], ], - PaymentService::PAYMENT_JOINPAY_GZH => [ + self::PAYMENT_JOINPAY_GZH => [ 'type' => 'WEIXIN_GZH', - 'name' => '汇聚 公众号 支付', + 'name' => '汇聚公众号支付', 'bind' => [UserService::APITYPE_WECHAT], ], ]; @@ -97,7 +117,7 @@ abstract class PaymentService protected $app; /** - * 支付通道编号 + * 支付参数编号 * @var string */ protected $code; @@ -109,7 +129,7 @@ abstract class PaymentService protected $type; /** - * 当前支付通道 + * 当前支付参数 * @var array */ protected $params; @@ -123,9 +143,9 @@ abstract class PaymentService /** * PaymentService constructor. * @param App $app 当前应用对象 - * @param string $code 支付通道编号 + * @param string $code 支付参数编号 * @param string $type 支付类型代码 - * @param array $params 支付通道配置 + * @param array $params 支付参数配置 */ public function __construct(App $app, string $code, string $type, array $params) { @@ -140,17 +160,23 @@ abstract class PaymentService /** * 根据配置实例支付服务 - * @param string $code 支付通道编号 + * @param string $code 支付参数编号 * @return JoinPaymentService|WechatPaymentService|AlipayPaymentService * @throws Exception */ public static function instance(string $code): PaymentService { + if ($code === 'empty') { + $vars = ['code' => 'empty', 'type' => 'empty', 'params' => []]; + return static::$driver[$code] = Container::getInstance()->make(EmptyPaymentService::class, $vars); + } [, $type, $params] = self::config($code); if (isset(static::$driver[$code])) return static::$driver[$code]; $vars = ['code' => $code, 'type' => $type, 'params' => $params]; - // 实例化具体支付通道类型 - if (stripos($type, 'alipay_') === 0) { + // 实例化具体支付参数类型 + if (stripos($type, 'balance') === 0) { + return static::$driver[$code] = Container::getInstance()->make(BalancePyamentService::class, $vars); + } elseif (stripos($type, 'alipay_') === 0) { return static::$driver[$code] = Container::getInstance()->make(AlipayPaymentService::class, $vars); } elseif (stripos($type, 'wechat_') === 0) { return static::$driver[$code] = Container::getInstance()->make(WechatPaymentService::class, $vars); @@ -161,6 +187,31 @@ abstract class PaymentService } } + /** + * 获取支付通道名称 + * @param string $type + * @return string + */ + public static function name(string $type): string + { + return self::TYPES[$type]['name'] ?? $type; + } + + /** + * 获取支付类型 + * @return array + */ + public static function types(): array + { + $types = []; + foreach (self::TYPES as $k => $v) if (isset($v['bind'])) { + if (array_intersect($v['bind'], array_keys(UserService::TYPES))) { + $types[$k] = $v; + } + } + return $types; + } + /** * 获取通道配置参数 * @param string $code @@ -173,17 +224,17 @@ abstract class PaymentService try { if (empty($payment)) { $map = ['code' => $code, 'status' => 1, 'deleted' => 0]; - $payment = app()->db->name('DataPayment')->where($map)->find(); + $payment = app()->db->name('ShopPayment')->where($map)->find(); } if (empty($payment)) { - throw new Exception("支付通道[#{$code}]禁用关闭"); + throw new Exception("支付参数[#{$code}]禁用关闭"); } $params = @json_decode($payment['content'], true); if (empty($params)) { - throw new Exception("支付通道[#{$code}]配置无效"); + throw new Exception("支付参数[#{$code}]配置无效"); } if (empty(static::TYPES[$payment['type']])) { - throw new Exception("支付通道[@{$payment['type']}]匹配失败"); + throw new Exception("支付参数[@{$payment['type']}]匹配失败"); } return [$payment['code'], $payment['type'], $params]; } catch (\Exception $exception) { @@ -196,12 +247,13 @@ abstract class PaymentService * @param string $orderNo 订单单号 * @param string $paymentTrade 交易单号 * @param string $paymentAmount 支付金额 + * @param string $paymentRemark 支付描述 * @return boolean * @throws \think\db\exception\DataNotFoundException * @throws \think\db\exception\DbException * @throws \think\db\exception\ModelNotFoundException */ - public function updateOrder(string $orderNo, string $paymentTrade, string $paymentAmount): bool + public function updateOrder(string $orderNo, string $paymentTrade, string $paymentAmount, $paymentRemark = '在线支付'): bool { // 检查订单支付状态 $map = ['order_no' => $orderNo, 'payment_status' => 0, 'status' => 2]; @@ -214,14 +266,12 @@ abstract class PaymentService 'payment_code' => $this->code, 'payment_trade' => $paymentTrade, 'payment_amount' => $paymentAmount, + 'payment_remark' => $paymentRemark, 'payment_status' => 1, - 'payment_remark' => '在线支付', 'payment_datetime' => date('Y-m-d H:i:s'), ]; if (empty($data['payment_type'])) unset($data['payment_type']); $this->app->db->name('ShopOrder')->where($map)->update($data); - // 调用用户升级机制 - OrderService::instance()->syncAmount($order['order_no']); // 触发订单更新事件 $this->app->event->trigger('ShopOrderPayment', $orderNo); return true; @@ -235,7 +285,7 @@ abstract class PaymentService */ protected function createPaymentAction(string $orderNo, string $paymentTitle, string $paymentAmount) { - $this->app->db->name('DataPaymentItem')->insert([ + $this->app->db->name('ShopPaymentItem')->insert([ 'payment_code' => $this->code, 'payment_type' => $this->type, 'order_amount' => $paymentAmount, 'order_name' => $paymentTitle, 'order_no' => $orderNo, ]); @@ -246,15 +296,16 @@ abstract class PaymentService * @param string $orderNo 商户订单单号 * @param string $paymentTrade 平台交易单号 * @param string $paymentAmount 实际到账金额 + * @param string $paymentRemark 平台支付备注 * @return boolean * @throws \think\db\exception\DataNotFoundException * @throws \think\db\exception\DbException * @throws \think\db\exception\ModelNotFoundException */ - protected function updatePaymentAction(string $orderNo, string $paymentTrade, string $paymentAmount): bool + protected function updatePaymentAction(string $orderNo, string $paymentTrade, string $paymentAmount, string $paymentRemark = '在线支付'): bool { // 更新支付记录 - data_save('DataPaymentItem', [ + data_save('ShopPaymentItem', [ 'order_no' => $orderNo, 'payment_code' => $this->code, 'payment_type' => $this->type, @@ -267,7 +318,7 @@ abstract class PaymentService 'payment_type' => $this->type, ]); // 更新记录状态 - return $this->updateOrder($orderNo, $paymentTrade, $paymentAmount); + return $this->updateOrder($orderNo, $paymentTrade, $paymentAmount, $paymentRemark); } /** diff --git a/app/data/service/UserService.php b/app/data/service/UserService.php index ffabca3d7..4d53c1e56 100644 --- a/app/data/service/UserService.php +++ b/app/data/service/UserService.php @@ -15,7 +15,7 @@ class UserService extends Service const APITYPE_WEB = 'web'; const APITYPE_WXAPP = 'wxapp'; const APITYPE_WECHAT = 'wechat'; - const APITYPE_IOSAPP = 'ios'; + const APITYPE_IOSAPP = 'iosapp'; const APITYPE_ANDROID = 'android'; const TYPES = [ @@ -64,8 +64,8 @@ class UserService extends Service $data = $this->app->db->name('DataUserToken')->where(['uid' => $uuid, 'type' => $type])->where(function ($query) { $query->where(['tokenv' => ''])->whereOr(['tokenv' => $this->_buildTokenVerify()]); })->findOrEmpty(); - unset($user['deleted'], $user['password']); $user['token'] = ['token' => $data['token'], 'expire' => $data['time']]; + unset($user['deleted'], $user['password']); return $user; } @@ -89,43 +89,56 @@ class UserService extends Service } else { $uuid = $this->app->db->name('DataUser')->strict(false)->insertGetId($data); } - if ($force) $this->buildUserToken(intval($uuid), $type); + if ($force) $this->token(intval($uuid), $type); return $this->get($type, $uuid); } /** - * 获取用户数据统计 - * @param int $uid 用户UID - * @return array + * 同步刷新用户余额 + * @param int $uuid 用户UID + * @param array $nots 排除的订单 + * @return array [total,count] + * @throws \think\db\exception\DbException */ - public function total(int $uid): array + public function balance(int $uuid, array $nots = []): array { - $query = $this->app->db->name('DataUser'); - return ['my_invite' => $query->where(['from' => $uid])->count()]; + $total = $this->app->db->name('DataUserBalance')->where(['uid' => $uuid, 'deleted' => 0])->sum('amount'); + $total += $this->app->db->name('DataUserBalanceTransfer')->where(['uid' => $uuid, 'deleted' => 0])->sum('amount'); + $count = $this->app->db->name('DataUserBalanceTransfer')->where(['from' => $uuid, 'deleted' => 0])->sum('amount'); + if (empty($nots)) { + $count += $this->app->db->name('ShopOrder')->whereRaw("uid={$uuid} and status>1")->sum('amount_balance'); + $this->app->db->name('DataUser')->where(['id' => $uuid])->update(['balance_total' => $total, 'balance_used' => $count]); + } else { + $count += $this->app->db->name('ShopOrder')->whereRaw("uid={$uuid} and status>1")->whereNotIn('order_no', $nots)->sum('amount_balance'); + } + return [$total, $count]; } /** - * 生成新的用户令牌 - * @param int $uid 授权用户 + * 检查 TOKEN 是否有效 * @param string $type 接口类型 - * @return array [创建状态, 状态描述, 令牌数据] + * @param string $token 认证令牌 + * @param array $data 认证数据 + * @return array [ 检查状态,状态描述,用户UID, 有效时间 ] + * @throws \think\db\exception\DataNotFoundException * @throws \think\db\exception\DbException + * @throws \think\db\exception\ModelNotFoundException */ - public function buildUserToken(int $uid, string $type): array + public function check(string $type, string $token, array $data = []): array { - // 清理历史认证及已过期的认证 - $map1 = [['time', '<', $time = time()]]; - $map2 = [['uid', '=', $uid], ['type', '=', $type]]; - $this->app->db->name('DataUserToken')->whereOr([$map1, $map2])->delete(); - // 创建新的用户认证数据 - do $map = ['type' => $type, 'token' => md5(uniqid('', true) . rand(100, 999))]; - while ($this->app->db->name('DataUserToken')->where($map)->count() > 0); - // 写入接口用户认证数据 - $data = array_merge($map, ['uid' => $uid, 'time' => $time + $this->expire, 'tokenv' => $this->_buildTokenVerify()]); - if ($this->app->db->name('DataUserToken')->insert($data) !== false) { - return [1, '刷新认证成功', $data]; + if (empty($data)) { + $map = ['type' => $type, 'token' => $token]; + $data = $this->app->db->name('DataUserToken')->where($map)->find(); + } + if (empty($data) || empty($data['uid'])) { + return [0, '请重新登录,登录认证无效', 0, 0]; + } elseif ($data['time'] < time()) { + return [0, '请重新登录,登录认证失效', 0, 0]; + } elseif ($token !== 'token' && $data['tokenv'] !== $this->_buildTokenVerify()) { + return [0, '请重新登录,客户端已更换', 0, 0]; } else { - return [0, '刷新认证失败', []]; + $this->expire($type, $token); + return [1, '登录验证成功', $data['uid'], $data['time']]; } } @@ -135,7 +148,7 @@ class UserService extends Service * @param string $token 授权令牌 * @throws \think\db\exception\DbException */ - public function expireUserToken(string $type, string $token) + public function expire(string $type, string $token) { $map = ['type' => $type, 'token' => $token]; $this->app->db->name('DataUserToken')->where($map)->update([ @@ -143,34 +156,6 @@ class UserService extends Service ]); } - /** - * 检查接口授权 TOKEN 是否有效 - * @param string $type 接口类型 - * @param string $token 认证令牌 - * @param array $data 认证数据 - * @return array [ 检查状态,状态描述,用户UID, 有效时间 ] - * @throws \think\db\exception\DataNotFoundException - * @throws \think\db\exception\DbException - * @throws \think\db\exception\ModelNotFoundException - */ - public function checkUserToken(string $type, string $token, array $data = []): array - { - if (empty($data)) { - $map = ['type' => $type, 'token' => $token]; - $data = $this->app->db->name('DataUserToken')->where($map)->find(); - } - if (empty($data) || empty($data['uid'])) { - return [0, '请重新登录,登录认证无效', 0, 0]; - } elseif ($data['time'] < time()) { - return [0, '请重新登录,登录认证已失效', 0, 0]; - } elseif ($data['tokenv'] !== $this->_buildTokenVerify() && $token !== 'token') { - return [0, '请重新登录,客户端已更换', 0, 0]; - } else { - $this->expireUserToken($type, $token); - return [1, '登录验证成功', $data['uid'], $data['time']]; - } - } - /** * 列表绑定用户数据 * @param array $list 原数据列表 @@ -188,6 +173,42 @@ class UserService extends Service return $list; } + /** + * 获取用户数据统计 + * @param int $uuid 用户UID + * @return array + */ + public function total(int $uuid): array + { + $query = $this->app->db->name('DataUser'); + return ['my_invite' => $query->where(['from' => $uuid])->count()]; + } + + /** + * 生成新的用户令牌 + * @param int $uuid 授权用户 + * @param string $type 接口类型 + * @return array [创建状态, 状态描述, 令牌数据] + * @throws \think\db\exception\DbException + */ + public function token(int $uuid, string $type): array + { + // 清理无效认证数据 + $map1 = [['time', '<', $time = time()]]; + $map2 = [['uid', '=', $uuid], ['type', '=', $type]]; + $this->app->db->name('DataUserToken')->whereOr([$map1, $map2])->delete(); + // 创建新的认证数据 + do $map = ['type' => $type, 'token' => md5(uniqid() . rand(100, 999))]; + while ($this->app->db->name('DataUserToken')->where($map)->count() > 0); + // 写入用户认证数据 + $data = array_merge($map, ['uid' => $uuid, 'time' => $time + $this->expire, 'tokenv' => $this->_buildTokenVerify()]); + if ($this->app->db->name('DataUserToken')->insert($data) !== false) { + return [1, '刷新认证成功', $data]; + } else { + return [0, '刷新认证失败', []]; + } + } + /** * 获取令牌的认证值 * @return string @@ -196,5 +217,4 @@ class UserService extends Service { return md5($this->app->request->server('HTTP_USER_AGENT', '-')); } - } \ No newline at end of file diff --git a/app/data/service/payment/BalancePyamentService.php b/app/data/service/payment/BalancePyamentService.php new file mode 100644 index 000000000..1a59df029 --- /dev/null +++ b/app/data/service/payment/BalancePyamentService.php @@ -0,0 +1,67 @@ +app->db->name('ShopOrder')->where(['order_no' => $orderNo])->find(); + if (empty($order)) throw new Exception("订单不存在"); + if ($order['status'] !== 2) throw new Exception("不可发起支付"); + // 创建支付行为 + $this->createPaymentAction($orderNo, $paymentTitle, $paymentAmount); + // 扣减用户余额 + [$total, $used] = UserService::instance()->balance($order['uid'], [$orderNo]); + if ($paymentAmount > $total - $used) throw new Exception("可抵扣余额不足"); + $this->app->db->name('ShopOrder')->where(['order_no' => $orderNo])->update(['amount_balance' => $paymentAmount]); + // 更新支付行为 + $this->updatePaymentAction($orderNo, CodeExtend::uniqidDate(20), $paymentAmount, '账户余额支付'); + // 刷新用户余额 + UserService::instance()->balance($order['uid']); + return ['info' => '余额支付完成']; + } +} \ No newline at end of file diff --git a/app/data/service/payment/EmptyPaymentService.php b/app/data/service/payment/EmptyPaymentService.php new file mode 100644 index 000000000..e9b0d616e --- /dev/null +++ b/app/data/service/payment/EmptyPaymentService.php @@ -0,0 +1,61 @@ +app->db->name('ShopOrder')->where(['order_no' => $orderNo])->find(); + if (empty($order)) throw new Exception("订单不存在"); + if ($order['status'] !== 2) throw new Exception("不可发起支付"); + // 创建支付行为 + $this->createPaymentAction($orderNo, $paymentTitle, $paymentAmount); + // 更新支付行为 + $this->updatePaymentAction($orderNo, CodeExtend::uniqidDate(20), $paymentAmount, '无需支付'); + return ['info' => '无需支付']; + } +} \ No newline at end of file diff --git a/app/data/sys.php b/app/data/sys.php index cc06fd384..de52125f1 100644 --- a/app/data/sys.php +++ b/app/data/sys.php @@ -1,4 +1,16 @@ request->isCli()) { + $console->addCommand(OrderClear::class); + $console->addCommand(UserBalance::class); + } +}); + if (!function_exists('show_goods_spec')) { /** * 商品规格过滤显示 diff --git a/app/data/view/shop_goods/form.html b/app/data/view/shop_goods/form.html index 3bf6e629f..ed79540e6 100644 --- a/app/data/view/shop_goods/form.html +++ b/app/data/view/shop_goods/form.html @@ -112,7 +112,7 @@ - +