Order.php 12 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314
  1. <?php
  2. namespace app\controller\api;
  3. use app\extra\basic\Base;
  4. use app\extra\dyLife\Crypt;
  5. use app\extra\dyMini\Pay;
  6. use app\middleware\AuthMiddleware;
  7. use app\model\saas\SaasComplaint;
  8. use app\model\saas\SaasOrder;
  9. use app\model\saas\SaasOrderPrice;
  10. use app\service\saas\OrderService;
  11. use DI\Attribute\Inject;
  12. use LinFly\Annotation\Attributes\Route\Controller;
  13. use LinFly\Annotation\Attributes\Route\GetMapping;
  14. use LinFly\Annotation\Attributes\Route\Middleware;
  15. use LinFly\Annotation\Attributes\Route\PostMapping;
  16. use support\Request;
  17. use support\Response;
  18. #[Controller("/dy/order"),Middleware(AuthMiddleware::class)]
  19. class Order extends Base
  20. {
  21. #[Inject]
  22. protected SaasOrder $model;
  23. #[Inject]
  24. protected OrderService $service;
  25. #[GetMapping('list')]
  26. public function getDataList(Request $request): Response
  27. {
  28. try {
  29. $param = $request->all();
  30. if (!empty($param['size'])) {
  31. $param['pageSize'] = $param['size'];
  32. }
  33. $param['openid'] = $request->user['openid'];
  34. $data = $this->service->setModel()->getList($param,['product' => function($query){
  35. $query->field("product_id,product_name");
  36. }]);
  37. return successTrans("success.data",pageFormat($data));
  38. } catch (\Throwable $th) {
  39. return error($th->getMessage());
  40. }
  41. }
  42. #[PostMapping("del")]
  43. public function delOrderData(Request $request): Response
  44. {
  45. try {
  46. $param = $this->_valid([
  47. "order.require" => trans("empty.require"),
  48. "type.require" => trans("empty.require"),
  49. ],$request->method());
  50. if (!is_array($param)) return error($param);
  51. $order = $this->model->where("order_sn",$param['order'])->with(['product','poi'])->findOrEmpty();
  52. if ($order->isEmpty()) return errorTrans("empty.data");
  53. if ($order['openid'] <> $request->user['openid']) return errorTrans("empty.data");
  54. switch ($param['type']) {
  55. case "del": // 删除订单
  56. if ($order['status'] <> 5) return error("该订单不支持删除");
  57. $state = $order->delete();
  58. break;
  59. case "cancel": // 取消订单
  60. if ($order['status'] <> 0) return error("该订单不支持取消");
  61. $order->status = 5;
  62. $state = $order->save();
  63. break;
  64. default:
  65. $state = false;
  66. break;
  67. }
  68. if (!$state) return errorTrans("error.data");
  69. return successTrans("success.data");
  70. } catch (\Throwable $throwable) {
  71. return error($throwable->getMessage());
  72. }
  73. }
  74. #[PostMapping("refund")]
  75. public function orderRefund(Request $request): Response
  76. {
  77. try {
  78. $param = $this->_valid([
  79. "order.require" => trans("empty.require"),
  80. "refund.require" => trans("empty.require"),
  81. ],$request->method());
  82. if (!is_array($param)) return error($param);
  83. $order = $this->model->where("order_sn",$param['order'])->with(['product','poi'])->findOrEmpty();
  84. if ($order->isEmpty()) return errorTrans("empty.data");
  85. if ($order['openid'] <> $request->user['openid']) return errorTrans("empty.data");
  86. if ($order['status'] <> 1) return error("当前订单不支持退款");
  87. $reason = explode(",",$param['refund']);
  88. $payParam = [
  89. "order_sn" => $order['order_sn'],
  90. "total" => $order['pay_money'],
  91. "reason" => $reason[0]??'不想要了',
  92. "out_refund_no" => $order['pay_sn'],
  93. "notify_url" => "https://tran.jsshuita.cn/notify/refund"
  94. ];
  95. $byteAuthorization = (new Pay)->config([
  96. "appid" => sConf("wechat.mini_appid"),
  97. "secret" => sConf("wechat.mini_secret"),
  98. "salt" => sConf("wechat.mch_salt"),
  99. ])->createRefund($payParam);
  100. if (!$byteAuthorization) return error("发起退款失败");
  101. $order->status = 4;
  102. $order->refund_apply = getDateFull();
  103. $state = $order->save();
  104. if (!$state) return errorTrans("error.data");
  105. return successTrans("提交申请成功");
  106. } catch (\Throwable $throwable) {
  107. return error($throwable->getMessage());
  108. }
  109. }
  110. #[GetMapping('detail')]
  111. public function getDataDetail(Request $request): Response
  112. {
  113. try {
  114. $param = $this->_valid([
  115. "order.require" => trans("empty.require")
  116. ],$request->method());
  117. if (!is_array($param)) return error($param);
  118. $order = $this->model->where("out_order_no",$param['order'])->with(['product','poi'])->findOrEmpty();
  119. if ($order->isEmpty()) return errorTrans("empty.data");
  120. $order['end_time'] = timeDiff(strtotime("+30 minutes",strtotime($order['create_at'])),time());
  121. return $this->encode("ok",$order->toArray());
  122. } catch (\Throwable $th) {
  123. return error($th->getMessage());
  124. }
  125. }
  126. #[GetMapping("confirm")]
  127. public function confirmOrder(Request $request): Response
  128. {
  129. try {
  130. $param = $this->_valid([
  131. "order.require" => trans("empty.require")
  132. ],$request->method());
  133. if (!is_array($param)) return error($param);
  134. $order = $this->model->where("order_sn",$param['order'])->with(['product','poi'])->findOrEmpty();
  135. if ($order->isEmpty()) return errorTrans("empty.data");
  136. if ($order['status'] <> 0) return errorTrans("empty.data");
  137. return $this->encode("ok",$order->toArray());
  138. } catch (\Throwable $throwable) {
  139. return error($throwable->getMessage());
  140. }
  141. }
  142. /**
  143. * 差价支付
  144. * @param Request $request
  145. * @return Response
  146. */
  147. #[PostMapping("pricePay")]
  148. public function orderPricePay(Request $request): Response
  149. {
  150. try {
  151. $param = $this->_valid([
  152. "order.require" => trans("empty.require")
  153. ],$request->method());
  154. if (!is_array($param)) return error($param);
  155. $order = (new SaasOrderPrice)->where("order_sn",$param['order'])->findOrEmpty();
  156. $payParam = [
  157. "order_sn" => $order['order_sn'],
  158. "total" => $order['price'],
  159. "name" => $order['name'],
  160. "notify_url" => "https://tran.jsshuita.cn/notify/price"
  161. ];
  162. $byteAuthorization = (new Pay)->config([
  163. "appid" => sConf("wechat.mini_appid"),
  164. "secret" => sConf("wechat.mini_secret"),
  165. "salt" => sConf("wechat.mch_salt"),
  166. ])->createOrder($payParam);
  167. return success("ok",['pay' => $byteAuthorization]);
  168. } catch (\Throwable $throwable) {
  169. return error($throwable->getMessage());
  170. }
  171. }
  172. /**
  173. * 发起支付
  174. * @param Request $request
  175. * @return Response
  176. */
  177. #[PostMapping("toPay")]
  178. public function orderPay(Request $request): Response
  179. {
  180. try {
  181. $param = $this->_valid([
  182. "order.require" => trans("empty.require"),
  183. "mobile.default" => ""
  184. ],$request->method());
  185. if (!is_array($param)) return error($param);
  186. $order = $this->model->where("order_sn",$param['order'])->with(['product','poi'])->findOrEmpty();
  187. $payParam = [
  188. "order_sn" => $order['order_sn'],
  189. "total" => $order['price'],
  190. "name" => $order['product']['product_name'],
  191. "notify_url" => "https://tran.jsshuita.cn/notify/douyin"
  192. ];
  193. $byteAuthorization = (new Pay)->config([
  194. "appid" => sConf("wechat.mini_appid"),
  195. "secret" => sConf("wechat.mini_secret"),
  196. "salt" => sConf("wechat.mch_salt"),
  197. ])->createOrder($payParam);
  198. if (!empty($param['mobile'])) {
  199. $order->mobile = $param['mobile'];
  200. $order->save();
  201. }
  202. return success("ok",['pay' => $byteAuthorization]);
  203. } catch (\Throwable $throwable) {
  204. return error($throwable->getMessage());
  205. }
  206. }
  207. /**
  208. * 交易组件获取手机号码
  209. * @param Request $request
  210. * @return Response
  211. */
  212. #[PostMapping("mobile")]
  213. public function getOrderMobile(Request $request): Response
  214. {
  215. try {
  216. $param = $request->post();
  217. $mobile = [];
  218. if (!empty($param['code'])) {
  219. $mobileStr = (new Crypt)->config($this->getDyConfig())->token()->getMobile($param['code']);
  220. if (!empty($mobileStr)) {
  221. $mobile = $this->decrypt2code(sConf('wechat.min_private_key'), $mobileStr);
  222. }
  223. }
  224. $mobileStr = "";
  225. if (!empty($mobile['purePhoneNumber'])) {
  226. $mobileStr = $mobile['purePhoneNumber'];
  227. }
  228. return success("ok",['mobile' => $mobileStr]);
  229. } catch (\Throwable $throwable) {
  230. return error($throwable->getMessage());
  231. }
  232. }
  233. /**
  234. * 订单投诉
  235. * @param Request $request
  236. * @return Response
  237. */
  238. #[PostMapping("complaint")]
  239. public function setOrderComplaint(Request $request): Response
  240. {
  241. try {
  242. $param = $this->_valid([
  243. "content.require" => trans("empty.require"),
  244. "mobile.default" => "",
  245. "order.require" => trans("empty.require"),
  246. "text.require" => trans("empty.require"),
  247. "poi_id.require" => trans("empty.require"),
  248. ],$request->method());
  249. if (!is_array($param)) return error($param);
  250. $order = (new SaasOrder)->where("out_order_no",$param['order'])->findOrEmpty();
  251. if ($order->isEmpty()) return error("该笔交易不存在");
  252. if ($order['openid'] <> $request->user['openid']) return error("非法操作");
  253. $state = (new SaasComplaint)->insertGetId([
  254. "openid" => $request->user['openid'],
  255. "poi_id" => $param['poi_id'],
  256. "content" => $param['content'],
  257. "question" => $param['text'],
  258. "order_sn" => $order['order'],
  259. "mobile" => $param['mobile']??$order['mobile'],
  260. ]);
  261. if (!$state) return error("提交失败");
  262. return success("提交成功,请耐心等待");
  263. } catch (\Throwable $throwable) {
  264. return error($throwable->getMessage());
  265. }
  266. }
  267. protected function decrypt2code($private_key, $ciphertext_str) {
  268. // 解码 base64 密文
  269. $ciphertext = base64_decode($ciphertext_str);
  270. // 使用私钥解密
  271. openssl_private_decrypt($ciphertext, $plaintext, $private_key, OPENSSL_PKCS1_PADDING);
  272. if ($plaintext === false) {
  273. return [];
  274. }
  275. return json_decode($plaintext,true);
  276. }
  277. protected function decrypt($encrypted_data, $session_key, $iv) {
  278. $data = base64_decode($encrypted_data);
  279. $key = base64_decode($session_key);
  280. $iv_decoded = base64_decode($iv);
  281. // 使用 AES-256-CBC 模式解密
  282. $decrypted = openssl_decrypt($data, 'AES-256-CBC', $key, OPENSSL_RAW_DATA, $iv_decoded);
  283. return $decrypted;
  284. }
  285. }