Queue.php 21 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429430431432433434435436437438439440441442443444445446447448449450451452453454455456457458459460461462463464465466467468469470471472473474475476477478479480481482483484485486487488489490491492493494495496497498499500501502503504505506507508
  1. <?php
  2. namespace app\api\controller;
  3. use alipay\aop\AopClient;
  4. use alipay\aop\request\AlipayTradeAppPayRequest;
  5. use app\admin\model\BuyVipLogModel;
  6. use app\api\model\QueueModel;
  7. use app\api\model\UsersModel;
  8. use app\common\controller\Api;
  9. use app\common\lib\WxPay;
  10. use think\Db;
  11. use function GuzzleHttp\Psr7\str;
  12. /**
  13. * 排队接口
  14. */
  15. class Queue extends Api
  16. {
  17. protected $noNeedLogin = '*';
  18. /**
  19. * 排队页面
  20. *
  21. * @ApiTitle (排队页面)
  22. * @ApiSummary (排队页面顶部提示信息)
  23. * @ApiMethod (POST)
  24. * @ApiRoute (/api/queue/queuePage)
  25. * @ApiParams (name="user_id", type="int", required=true, description="用户id")
  26. */
  27. public function queuePage()
  28. {
  29. $userId = $this->request->post('user_id');
  30. if (!$userId) {
  31. $this->result('参数错误', [], 200);
  32. }
  33. $user = UsersModel::where('user_id', $userId)->find();
  34. if (!$user) {
  35. $this->result('用户不存在', [], 200);
  36. }
  37. // 获得的次数
  38. $getNum = $user->get_queue_num;
  39. // 基础次数
  40. $baseNum = Db::name('queue_set')->where('level', $user->user_level)->value('num');
  41. // 可用的基础次数
  42. $canUseBaseNum = $baseNum - $user->queue_num_today;
  43. // 总可用次数
  44. $total = $getNum + $canUseBaseNum;
  45. $this->result('ok', [
  46. 'tip' => '尊敬的VIP会员,你还可排队'.$total.'次,排队时间(09:00-22:00),其中推荐用户获得'.$getNum.'次,VIP会员奖励'.$canUseBaseNum.'次'
  47. ], 200);
  48. }
  49. /**
  50. * 排队
  51. *
  52. * @ApiTitle (排队)
  53. * @ApiSummary (排队)
  54. * @ApiMethod (POST)
  55. * @ApiRoute (/api/queue/queue)
  56. * @ApiParams (name="user_id", type="int", required=true, description="用户id")
  57. * @ApiParams (name="pay_type", type="int", required=true, description="支付方式:1-余额 2-微信 3-支付宝")
  58. */
  59. public function queue()
  60. {
  61. $userId = $this->request->post('user_id');
  62. $payType = $this->request->post('pay_type'); // 1-余额 2-微信 3-支付宝
  63. $amount = 0.01;
  64. $income = 2;
  65. if (!$userId || !$payType) {
  66. $this->result('参数错误', [], 100);
  67. }
  68. $user = UsersModel::get($userId);
  69. if (!$user) {
  70. $this->result('用户不存在', [], 100);
  71. }
  72. if ($user->user_level == 1) {
  73. $this->result('您还不是VIP用户,不能进行排队哦', [], 100);
  74. }
  75. // 查询所有的可排队次数是否已经用完
  76. // 情况一:获得次数有、基础次数有,可排队
  77. // 情况二:获得次数有、基础次数无,不存在该种情况
  78. // 情况三:获得次数无、基础次数有,可排队
  79. // 情况四:都无,不可排队
  80. $canQueueNum = Db::name('queue_set')->where('level', $user->user_level)->value('num');
  81. if ($user->queue_num_today == $canQueueNum) {
  82. $this->result('您的可用排队次数不足', [], 100);
  83. }
  84. $nowTime = time();
  85. $startTime = strtotime(date('Y-m-d').' 09:00:00');
  86. $endTime = strtotime(date('Y-m-d').' 22:00:00');
  87. if ($startTime < $nowTime && $nowTime < $endTime) {
  88. $nowHour = date("Y-m-d H:00:00", time());
  89. $nextHour = date("Y-m-d H:00:00", strtotime("+1 hour"));
  90. // 是否在本时间段内排过队
  91. $hasOnLine = QueueModel::where(['uid' => $userId, 'pay_status' => 1])
  92. ->whereTime('time', 'between', [$nowHour, $nextHour])
  93. ->find();
  94. if (!empty($hasOnLine)) {
  95. $this->result('您当前已排过队', [], 100);
  96. }
  97. // 进入队列时的时间
  98. $joinQueueTime = date('Y-m-d H:i:s', time());//halt($joinQueueTime);
  99. // 当前时间段排队人数
  100. $lineNum = QueueModel::whereTime('time', 'between', [$nowHour, $joinQueueTime])->where('pay_status',1)->count();
  101. $trade = createOutTradeNo();
  102. $lineInfo = array(
  103. 'uid' => $userId,
  104. 'trade' => $trade,
  105. 'amount' => $amount,
  106. 'line_num' => $lineNum,
  107. 'income' => $income,
  108. 'time' => date('Y-m-d H:i:s', time()),
  109. 'pay_type' => $payType
  110. );
  111. Db::startTrans();
  112. $insert = QueueModel::insert($lineInfo);
  113. if ($insert) {
  114. Db::commit();
  115. // 余额支付
  116. if ($payType == 1) {
  117. $this->result('订单创建成功', ['out_trade_no' => $trade, 'fee' => $amount], 200);
  118. }
  119. // 微信支付
  120. if ($payType == 2) {
  121. $payObj = new WxPay();
  122. $notifyUrl = config('site.httpurl').'/api/queue/wx_notify_url';
  123. $getPrePayInfo = $payObj->getPrePayOrder('优享街排队', $trade, ($amount * 100), $notifyUrl);
  124. if ($getPrePayInfo) {
  125. $getPayInfo = $payObj->getOrder($getPrePayInfo['prepay_id']);
  126. if ($getPayInfo) {
  127. $this->result('订单创建成功', $getPayInfo, 200);
  128. } else {
  129. $this->result('支付错误,请稍后再试', [], 100);
  130. }
  131. } else {
  132. $this->result('支付错误,请稍后再试', [], 100);
  133. }
  134. }
  135. // 支付宝支付
  136. if ($payType == 3) {
  137. $notify_url2 = config('site.httpurl').'/api/queue/alipay_notify_url';
  138. $aop = new AopClient;
  139. $aop->gatewayUrl = "https://openapi.alipay.com/gateway.do";
  140. $aop->appId = config('alipay.app_id');
  141. $aop->rsaPrivateKey = config('alipay.private_key');
  142. $aop->format = "json";
  143. $aop->charset = "utf-8";
  144. $aop->signType = "RSA2";
  145. $aop->alipayrsaPublicKey = config('alipay.public_key');
  146. //实例化具体API对应的request类,类名称和接口名称对应,当前调用接口名称:alipay.trade.app.pay
  147. $request = new AlipayTradeAppPayRequest();
  148. // 订单标题
  149. $subject = '优享街排队';
  150. // 订单详情
  151. $body = '优享街排队';
  152. // SDK已经封装掉了公共参数,这里只需要传入业务参数
  153. $bizcontent = json_encode([
  154. 'body' => $body,
  155. 'subject' => $subject,
  156. 'out_trade_no' => $trade,
  157. 'timeout_express' => '90m',
  158. 'total_amount' => $amount,
  159. 'product_code' => 'QUICK_MSECURITY_PAY'
  160. ]);
  161. $request->setNotifyUrl($notify_url2);
  162. $request->setBizContent($bizcontent);
  163. // 这里和普通的接口调用不同,使用的是sdkExecute
  164. $response = $aop->sdkExecute($request);
  165. // 注意:这里不需要使用htmlspecialchars进行转义,直接返回即可
  166. // return $response;
  167. $this->result('订单发起成功', ['payInfoString' => $response], 200);
  168. }
  169. } else {
  170. Db::rollback();
  171. $this->result('服务器繁忙', [], 100);
  172. }
  173. } else {
  174. $this->result('今日排队活动已经结束了哦', [], 100);
  175. }
  176. }
  177. /**
  178. * 排队余额支付
  179. *
  180. * @ApiTitle (排队余额支付)
  181. * @ApiSummary (排队余额支付)
  182. * @ApiMethod (POST)
  183. * @ApiRoute (/api/queue/balancePayQueue)
  184. * @ApiParams (name="user_id", type="int", required=true, description="用户id")
  185. * @ApiParams (name="fee", type="decimal", required=true, description="排队接口返回的支付金额fee字段的值")
  186. * @ApiParams (name="out_trade_no", type="string", required=true, description="排队接口返回的支付订单编号out_trade_no字段的值")
  187. * @ApiParams (name="pay_password", type="string", required=true, description="用户支付密码")
  188. */
  189. public function balancePayQueue()
  190. {
  191. $userId = $this->request->post('user_id');
  192. $fee = $this->request->post('fee');
  193. $trade = $this->request->post('out_trade_no');
  194. $payPwd = $this->request->post('pay_password');
  195. if (!$userId || !$fee || !$trade || !$payPwd) {
  196. $this->result('参数错误', [], 100);
  197. }
  198. $user = UsersModel::get($userId);
  199. // 用户是否存在
  200. if (!$user) {
  201. $this->result('用户不存在', [], 100);
  202. }
  203. // 订单是否存在
  204. $order = QueueModel::where('trade', $trade)->find();
  205. if (!$order) {
  206. $this->result('订单不存在', [], 100);
  207. }
  208. if ($fee <= 0) {
  209. $this->result('请输入正确的支付金额', [], 100);
  210. }
  211. // 余额是否满足本次扣费金额
  212. if ($user->user_money < $fee) {
  213. $this->result('账户余额不足', [], 100);
  214. }
  215. // 是否设置支付密码
  216. if (empty($user->user_paypwd)) {
  217. $this->result('请先设置支付密码', [], 100);
  218. }
  219. // 支付密码是否正确
  220. if ($user->user_paypwd != $payPwd) {
  221. $this->result('支付密码错误', [], 100);
  222. }
  223. Db::startTrans();
  224. // 扣除账户余额
  225. $dec = UsersModel::where('user_id', $userId)->setDec('user_money', $fee);
  226. // 增加排队收入
  227. $amount = 2;
  228. $inc = UsersModel::where('user_id', $userId)->setInc('user_money', $amount);
  229. // 更新支付状态和支付时间
  230. $updatePayStatus = QueueModel::where('trade', $trade)
  231. ->update(['pay_status' => 1, 'pay_time' => date('Y-m-d H:i:s', time())]);
  232. // 获得的次数是否够用
  233. if ($user->get_queue_num < 1) {
  234. // 不够用,使用基础次数
  235. $decBaseNum = UsersModel::where('user_id', $userId)->setInc('queue_num_today', 1);
  236. if ($dec && $inc && $updatePayStatus && $decBaseNum) {
  237. Db::commit();
  238. $this->result('排队成功', [], 200);
  239. } else {
  240. Db::rollback();
  241. $this->result('支付失败,稍后再试', [], 100);
  242. }
  243. } else {
  244. // 获得次数够用,直接减少一次次数
  245. $decGetNum = UsersModel::where('user_id', $userId)->setDec('get_queue_num', 1);
  246. if ($dec && $inc && $updatePayStatus && $decGetNum) {
  247. Db::commit();
  248. $this->result('排队成功', [], 200);
  249. } else {
  250. Db::rollback();
  251. $this->result('支付失败,稍后再试', [], 100);
  252. }
  253. }
  254. }
  255. /**
  256. * 排队微信支付异步回调处理
  257. * @ApiInternal
  258. */
  259. public function wx_notify_url()
  260. {
  261. //获取返回的xml格式数据
  262. $payXml = file_get_contents("php://input");
  263. //将xml格式转化为json格式
  264. $jsonXml = json_encode(simplexml_load_string($payXml, 'SimpleXMLElement', LIBXML_NOCDATA));
  265. //将json格式转成数组格式
  266. $result = json_decode($jsonXml, true);
  267. if ($result) {
  268. //如果成功返回
  269. if ($result['return_code'] == 'SUCCESS') {
  270. if ($result['result_code'] == 'SUCCESS') {
  271. // sign 值校验
  272. // 校验时不包含返回的 sign 字段,需踢除 sign 字段
  273. foreach($result as $k => $v) {
  274. if ($k == 'sign') {
  275. $sign = $result[$k];
  276. unset($result[$k]);
  277. };
  278. }
  279. //按字典排序
  280. ksort($result);
  281. //转为 url 键值对
  282. $signTemp = http_build_query($result);
  283. //md5处理,$key 为微信商户平台的 api 安全密钥
  284. $key = 'b3ae6bbf3cc4fa017eb169ae219e2c27';
  285. $signTemp = md5($signTemp.'&key='.$key);
  286. //转大写得最终 sign 值
  287. $resultSign = strtoupper($signTemp);
  288. //如果sign值正确
  289. if ($sign === $resultSign) {
  290. $order = QueueModel::where('trade', $result['out_trade_no'])->find();
  291. $user = UsersModel::where('user_id', $order->uid)->find();
  292. if ($order) {
  293. // 增加排队收入
  294. $amount = 2;
  295. $inc = UsersModel::where('user_id', $user->user_id)->setInc('user_money', $amount);
  296. // 更新支付状态和支付时间
  297. $updatePayStatus = QueueModel::where('trade', $result['out_trade_no'])
  298. ->update(['pay_status' => 1, 'pay_time' => date('Y-m-d H:i:s', time())]);
  299. // 获得的次数是否够用
  300. if ($user->get_queue_num < 1) {
  301. // 不够用,使用基础次数
  302. $decBaseNum = UsersModel::where('user_id', $user->user_id)->setInc('queue_num_today', 1);
  303. if ($inc && $updatePayStatus && $decBaseNum) {
  304. $successArray = array(
  305. 'return_code' => 'SUCCESS',
  306. 'return_msg' => 'OK'
  307. );
  308. return $this->arrayToXml($successArray);
  309. }
  310. } else {
  311. // 获得次数够用,直接减少一次次数
  312. $decGetNum = UsersModel::where('user_id', $user->user_id)->setDec('get_queue_num', 1);
  313. if ($inc && $updatePayStatus && $decGetNum) {
  314. $successArray = array(
  315. 'return_code' => 'SUCCESS',
  316. 'return_msg' => 'OK'
  317. );
  318. return $this->arrayToXml($successArray);
  319. }
  320. }
  321. }
  322. }
  323. } else {
  324. return false;
  325. }
  326. } else {
  327. return false;
  328. }
  329. }
  330. }
  331. /**
  332. * 数组转xml
  333. * @ApiInternal
  334. */
  335. public function arrayToXml($arr)
  336. {
  337. $xml = "<xml>";
  338. foreach ($arr as $key=>$val)
  339. {
  340. if (is_numeric($val))
  341. {
  342. $xml.="<".$key.">".$val."</".$key.">";
  343. }
  344. else
  345. $xml.="<".$key."><![CDATA[".$val."]]></".$key.">";
  346. }
  347. $xml.="</xml>";
  348. return $xml;
  349. }
  350. /**
  351. * 排队支付支付宝回调
  352. * @ApiInternal
  353. */
  354. public function alipay_notify_url()
  355. {
  356. $params = $this->request->post();
  357. if (!empty($params) && $params['trade_status'] == 'TRADE_SUCCESS') {
  358. // 验证签名
  359. $aop = new AopClient();
  360. $aop->alipayrsaPublicKey = config('alipay.public_key');
  361. // 此处反转义参数中的字符,否则验签不通过
  362. $params['fund_bill_list'] = htmlspecialchars_decode($params['fund_bill_list']);
  363. $checkSign = $aop->rsaCheckV1($params, null, 'RSA2');
  364. if ($checkSign) {
  365. // 是不是向此商户号付款
  366. if ($params['app_id'] == config('alipay.app_id')) {
  367. // 查询订单是否存在
  368. $order = QueueModel::where('trade', $params['out_trade_no'])->find();
  369. $user = UsersModel::where('user_id', $order->uid)->find();
  370. if ($order) {
  371. // 增加排队收入
  372. $amount = 2;
  373. $inc = UsersModel::where('user_id', $user->user_id)->setInc('user_money', $amount);
  374. // 更新支付状态和支付时间
  375. $updatePayStatus = QueueModel::where('trade', $params['out_trade_no'])
  376. ->update(['pay_status' => 1, 'pay_time' => date('Y-m-d H:i:s', time())]);
  377. // 获得的次数是否够用
  378. if ($user->get_queue_num < 1) {
  379. // 不够用,使用基础次数
  380. $decBaseNum = UsersModel::where('user_id', $user->user_id)->setInc('queue_num_today', 1);
  381. if ($inc && $updatePayStatus && $decBaseNum) {
  382. echo 'success';
  383. }
  384. } else {
  385. // 获得次数够用,直接减少一次次数
  386. $decGetNum = UsersModel::where('user_id', $user->user_id)->setDec('get_queue_num', 1);
  387. if ($inc && $updatePayStatus && $decGetNum) {
  388. echo 'success';
  389. }
  390. }
  391. }
  392. }
  393. }
  394. }
  395. }
  396. /**
  397. * 已成功排队的记录
  398. *
  399. * @ApiTitle (获取已成功排队列表)
  400. * @ApiSummary (此接口返回的是成功支付的排队订单,均是已完成订单,已排好顺序)
  401. * @ApiMethod (POST)
  402. * @ApiRoute (/api/queue/getQueueList)
  403. * @ApiParams (name="user_id", type="int", required=true, description="用户id")
  404. */
  405. public function getQueueList()
  406. {
  407. $userId = $this->request->post('user_id');
  408. if (!$userId) {
  409. $this->result('参数错误', [], 100);
  410. }
  411. if (!UsersModel::checkUserExist($userId)) {
  412. $this->result('用户不存在', [], 100);
  413. }
  414. $myList = QueueModel::where(['uid' => $userId, 'pay_status' => 1])
  415. ->field('id,trade,time,pay_time')
  416. ->order('id desc')
  417. ->select();
  418. $this->result('ok', $myList, 200);
  419. }
  420. /**
  421. * 查看当前排队记录详情
  422. *
  423. * @ApiTitle (查看当前排队记录详情)
  424. * @ApiSummary (查看当前排队记录详情)
  425. * @ApiMethod (POST)
  426. * @ApiRoute (/api/queue/lookQueueDetail)
  427. * @ApiParams (name="id", type="int", required=true, description="【获取已成功排队列表】接口中返回的id字段的值")
  428. */
  429. public function lookQueueDetail()
  430. {
  431. $id = $this->request->post('id');
  432. if (!$id) {
  433. $this->result('参数错误', [], 100);
  434. }
  435. $queueOrder = QueueModel::field('trade,line_num')->find($id);
  436. //$nowTime = date('Y-m-d H:i:s', time());
  437. //$nextTime = date("Y-m-d H:00:00", strtotime("+1 hour"));
  438. //$mistake = strtotime($nextTime) - strtotime($nowTime);
  439. //$hour=floor((strtotime($nextTime)-strtotime($nowTime))%86400/3600);
  440. //$minute=floor((strtotime($nextTime)-strtotime($nowTime))%86400/60);
  441. //$second=floor((strtotime($nextTime)-strtotime($nowTime))%86400%60);
  442. //halt($second);
  443. if (!empty($queueOrder)) {
  444. $this->result('ok', $queueOrder, 200);
  445. } else {
  446. $this->result('记录不存在', [], 100);
  447. }
  448. }
  449. /**
  450. * 排队记录
  451. *
  452. * @ApiTitle (获取排队记录)
  453. * @ApiSummary (获取排队记录)
  454. * @ApiMethod (POST)
  455. * @ApiRoute (/api/queue/getQueueRecord)
  456. * @ApiParams (name="user_id", type="int", required=true, description="用户id")
  457. * @ApiParams (name="type", type="int", required=true, description="获取类型 1-获取消费记录 2-获取收益记录")
  458. */
  459. public function getQueueRecord()
  460. {
  461. $userId = $this->request->post('user_id');
  462. $type = $this->request->post('type'); // 1-消费记录 2-收益记录
  463. if (!$userId || !$type) {
  464. $this->result('参数错误', [], 100);
  465. }
  466. if (!UsersModel::checkUserExist($userId)) {
  467. $this->result('用户不存在', [], 100);
  468. }
  469. if ($type == 1) {
  470. $record = QueueModel::where(['uid' => $userId, 'pay_status' => 1])
  471. ->field('amount,pay_type,time')
  472. ->order('id desc')
  473. ->select();
  474. $this->result('ok', $record, 200);
  475. }
  476. if ($type == 2) {
  477. $record = QueueModel::where(['uid' => $userId, 'pay_status' => 1])
  478. ->field('income,time')
  479. ->order('id desc')
  480. ->select();
  481. $this->result('ok', $record, 200);
  482. }
  483. }
  484. }