Order.php 10 KB

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