Member.php 10 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267
  1. <?php
  2. namespace app\api\controller;
  3. use app\admin\model\BuyVipLogModel;
  4. use app\admin\model\Cardtip;
  5. use app\api\model\QueueModel;
  6. use app\api\model\UsersModel;
  7. use app\common\controller\Api;
  8. use app\admin\model\Equity;
  9. use app\common\lib\WxPay;
  10. use think\Db;
  11. use function fast\array_except;
  12. /**
  13. * VIP接口
  14. */
  15. class Member extends Api
  16. {
  17. protected $noNeedLogin = '*';
  18. /**
  19. * 开通VIP页面
  20. *
  21. * @ApiTitle (开通VIP页面)
  22. * @ApiSummary (开通VIP页面)
  23. * @ApiMethod (POST)
  24. * @ApiRoute (/api/member/index)
  25. * @ApiParams (name="user_id", type="int", required=true, description="用户id")
  26. * @ApiReturnParams (name="code", type="int", required=true, sample="200")
  27. * @ApiReturnParams (name="msg", type="string", required=true, sample="ok")
  28. * @ApiReturnParams (name="data", type="array", description="数据返回")
  29. * @ApiReturn ()
  30. */
  31. public function index()
  32. {
  33. $userId = $this->request->post('user_id');
  34. if (!$userId) {
  35. $this->result('参数错误', [], 200);
  36. }
  37. //权益列表
  38. $list = Equity::select();
  39. foreach ($list as &$v) {
  40. $v['desc'] = explode(' ', $v['desc']);
  41. }
  42. //头部卡片
  43. $info = Cardtip::find(1);
  44. $info['avatar'] = UsersModel::where('user_id', $userId)->value('user_avatar');
  45. $this->result('ok', ['card' => $info, 'equityList' => $list], 200);
  46. }
  47. /**
  48. * 开通VIP
  49. *
  50. * @ApiTitle (开通VIP)
  51. * @ApiSummary (开通VIP)
  52. * @ApiMethod (POST)
  53. * @ApiRoute (/api/member/buyVip)
  54. * @ApiParams (name="user_id", type="int", required=true, description="用户id")
  55. * @ApiReturnParams (name="code", type="int", required=true, sample="200")
  56. * @ApiReturnParams (name="msg", type="string", required=true, sample="ok")
  57. * @ApiReturnParams (name="data", type="array", description="数据返回")
  58. * @ApiReturn ()
  59. */
  60. public function buyVip()
  61. {
  62. $userId = $this->request->post('user_id');
  63. $amount = 68;
  64. $notify_url = config('site.httpurl').'/api/member/notify_url';
  65. if (!$userId) {
  66. $this->result('参数错误', [], 200);
  67. }
  68. $out_trade_no = createOutTradeNo();
  69. $orderInfo = array(
  70. 'uid' => $userId,
  71. 'out_trade_no' => $out_trade_no,
  72. 'amount' => $amount,
  73. 'time' => date('Y-m-d H:i:s', time())
  74. );
  75. Db::startTrans();
  76. try {
  77. $insert = BuyVipLogModel::insert($orderInfo);
  78. if ($insert) {
  79. Db::commit();
  80. $payObj = new WxPay();
  81. $getPrePayInfo = $payObj->getPrePayOrder('VIP会员购买', $out_trade_no, ($amount * 100), $notify_url);
  82. if ($getPrePayInfo) {
  83. $payInfo = $payObj->getOrder($getPrePayInfo['prepay_id']);
  84. if ($payInfo) {
  85. $this->result('支付订单发起成功', $payInfo, 200);
  86. } else {
  87. $this->result('支付订单发起失败,请稍后再试', [], 100);
  88. }
  89. } else {
  90. $this->result('支付订单发起失败,请稍后再试', [], 100);
  91. }
  92. }
  93. } catch (\Exception $exception) {
  94. Db::rollback();
  95. $this->result('支付订单发起失败,请稍后再试('.$exception->getMessage().')', [], 100);
  96. }
  97. }
  98. /**
  99. * 排队
  100. *
  101. * @ApiTitle (排队)
  102. * @ApiSummary (排队)
  103. * @ApiMethod (POST)
  104. * @ApiRoute (/api/member/queue)
  105. * @ApiParams (name="user_id", type="int", required=true, description="用户id")
  106. * @ApiParams (name="pay_type", type="int", required=true, description="支付方式:1-余额 2-微信 3-支付宝")
  107. * @ApiReturnParams (name="code", type="int", required=true, sample="200")
  108. * @ApiReturnParams (name="msg", type="string", required=true, sample="ok")
  109. * @ApiReturnParams (name="data", type="array", description="数据返回")
  110. * @ApiReturn ()
  111. */
  112. public function queue()
  113. {
  114. $userId = $this->request->post('user_id');
  115. $payType = $this->request->post('pay_type'); // 1-余额 2-微信 3-支付宝
  116. $amount = 1.2;
  117. $income = 2;
  118. if (!$userId || !$payType) {
  119. $this->result('参数错误', [], 100);
  120. }
  121. $user = UsersModel::get($userId);
  122. if ($user->user_level == 1) {
  123. $this->result('您还不是VIP用户,不能进行排队哦', [], 100);
  124. }
  125. // 剩余总排队次数是否足够
  126. if ($user->user_line_num <= 0) {
  127. $this->result('您的排队次数不足,不能进行排队哦', [], 100);
  128. }
  129. // 每天可排队次数是否足够
  130. if ($user->can_line_num_perday <= 0) {
  131. $this->result('您每天可排队次数不足哦', [], 100);
  132. }
  133. // 今天排队次数是否足够
  134. if ($user->can_line_num_perday == $user->line_num_today) {
  135. $this->result('您今天的可排队次数已用完', [], 100);
  136. }
  137. $nowTime = time();
  138. $startTime = strtotime(date('Y-m-d').' 09:00:00');
  139. $endTime = strtotime(date('Y-m-d').' 22:00:00');
  140. if ($startTime < $nowTime && $nowTime < $endTime) {
  141. $nowHour = date("Y-m-d H:00:00", time());
  142. $nextHour = date("Y-m-d H:i:s", strtotime("+1 hour"));
  143. //是否在本时间段内排过队
  144. $hasOnLine = QueueModel::where(['uid' => $userId, 'pay_status' => 1])
  145. ->whereTime('time', 'between', [$nowHour, $nextHour])
  146. ->find();
  147. if (!empty($hasOnLine)) {
  148. $this->result('您当前已排过队', [], 100);
  149. }
  150. //当前时间段排队人数
  151. $lineNum = QueueModel::whereTime('time', 'between', [$nowHour, $nextHour])->where('pay_status',1)->count();
  152. $trade = createOutTradeNo();
  153. $lineInfo = array(
  154. 'uid' => $userId,
  155. 'trade' => $trade,
  156. 'amount' => $amount,
  157. 'line_num' => $lineNum,
  158. 'income' => $income,
  159. 'time' => date('Y-m-d H:i:s', time()),
  160. 'pay_type' => $payType
  161. );
  162. Db::startTrans();
  163. $insert = QueueModel::insert($lineInfo);
  164. if ($insert) {
  165. Db::commit();
  166. if ($payType == 1) {
  167. $this->result('订单创建成功', ['out_trade_no' => $trade, 'fee' => $amount], 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/member/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. * @ApiReturnParams (name="code", type="int", required=true, sample="200")
  189. * @ApiReturnParams (name="msg", type="string", required=true, sample="ok")
  190. * @ApiReturnParams (name="data", type="array", description="数据返回")
  191. * @ApiReturn ()
  192. */
  193. public function balancePayQueue()
  194. {
  195. $userId = $this->request->post('user_id');
  196. $fee = $this->request->post('fee');
  197. $trade = $this->request->post('out_trade_no');
  198. $payPwd = $this->request->post('pay_password');
  199. if (!$userId || !$fee || !$trade || !$payPwd) {
  200. $this->result('参数错误', [], 100);
  201. }
  202. $user = UsersModel::get($userId);
  203. // 用户是否存在
  204. if (!$user) {
  205. $this->result('用户不存在', [], 100);
  206. }
  207. // 订单是否存在
  208. $order = QueueModel::where('trade', $trade)->find();
  209. if (!$order) {
  210. $this->result('订单不存在', [], 100);
  211. }
  212. if ($fee <= 0) {
  213. $this->result('请输入正确的支付金额', [], 100);
  214. }
  215. // 余额是否满足本次扣费金额
  216. if ($user->user_money < $fee) {
  217. $this->result('账户余额不足', [], 100);
  218. }
  219. // 是否设置支付密码
  220. if (empty($user->user_paypwd)) {
  221. $this->result('请先设置支付密码', [], 100);
  222. }
  223. // 支付密码是否正确
  224. if ($user->user_paypwd != $payPwd) {
  225. $this->result('支付密码错误', [], 100);
  226. }
  227. Db::startTrans();
  228. // 扣除账户余额
  229. $dec = UsersModel::where('user_id', $userId)->setDec('user_money', $fee);
  230. // 增加排队收入
  231. $amount = 2;
  232. $inc = UsersModel::where('user_id', $userId)->setInc('user_money', $amount);
  233. // 更新支付状态和支付时间
  234. $updatePayStatus = QueueModel::where('trade', $trade)
  235. ->update(['pay_status' => 1, 'pay_time' => date('Y-m-d H:i:s', time())]);
  236. // 消耗一次今日可排队次数
  237. $addPerDayQueueNum = UsersModel::where('user_id', $userId)->setInc('line_num_today', 1);
  238. // 消耗一次总排队次数
  239. $decTotalQueueNum = UsersModel::where('user_id', $userId)->setDec('user_line_num', 1);
  240. if ($dec && $inc && $updatePayStatus && $addPerDayQueueNum && $decTotalQueueNum) {
  241. Db::commit();
  242. $this->result('排队成功', [], 200);
  243. } else {
  244. Db::rollback();
  245. $this->result('支付失败,稍后再试', [], 100);
  246. }
  247. }
  248. }