HuiPayBak.php 14 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398
  1. <?php
  2. namespace app\api\controller;
  3. use addons\epay\library\Service;
  4. use app\common\controller\Api;
  5. use app\utils\CurlUtil;
  6. use app\utils\Easywechat\MiniAppService;
  7. use app\utils\LogUtil;
  8. use app\utils\PayUtil;
  9. use think\Cache;
  10. use think\Db;
  11. use think\Exception;
  12. use think\Request;
  13. /**
  14. * 通行证
  15. */
  16. class HuiPay extends Api
  17. {
  18. // 日志模块名称
  19. const LOG_MODULE = 'HuiPay';
  20. protected $noNeedLogin = ['vip_recharge', 'gold_recharge', 'pay', 'pay_notify'];
  21. protected $noNeedRight = '*';
  22. // h5 跳转小程序
  23. const H5 = 'https://mp-pay-0gk77hzo6366ff37-1320524135.tcloudbaseapp.com';
  24. public function _initialize()
  25. {
  26. parent::_initialize();
  27. }
  28. public function __construct(Request $request = null)
  29. {
  30. parent::__construct($request);
  31. //日志统一写入
  32. register_shutdown_function([new LogUtil, 'close']);
  33. LogUtil::getInstance('Api/'); //设置日志存入通道
  34. }
  35. /**
  36. * 会员充值 (跳转小程序充值)
  37. * @return void
  38. */
  39. public function vip_recharge()
  40. {
  41. Db::startTrans();
  42. try {
  43. $rc_id = input('rc_id', 0);
  44. $pay_type = input('pay_type', 'wechat');
  45. $uid = $this->auth->id;
  46. if ($pay_type != 'wechat') {
  47. throw new Exception('支付类型有误');
  48. }
  49. if (!$rc_id) {
  50. throw new Exception('请选择会员套餐');
  51. }
  52. //赋值money
  53. $recharge_config = Db::name('payvip_config')->where('id', $rc_id)->find();
  54. $money = $recharge_config['money'];
  55. if ($money <= 0) {
  56. throw new Exception('支付金额必须大于0');
  57. }
  58. if ($money > 10000) {
  59. throw new Exception('支付金额太大');
  60. }
  61. //会员等级冲突
  62. //当前是会员,但是却要向下级续费,直接提示报错
  63. //修改等级,向上立刻改,向下不允许
  64. $wallet_info = model('wallet')->getWallet($this->auth->id);
  65. if ($wallet_info['vip_endtime'] > time() && $recharge_config['vip_level'] < $wallet_info['vip_level']) {
  66. throw new Exception('当前会员没有过期,不能续费');
  67. }
  68. //创建订单
  69. $data = [];
  70. $data['status'] = 0;
  71. $pay_no = createUniqueNo('V', $uid);
  72. $data['pay_no'] = $pay_no;
  73. $data['money'] = $money;
  74. $data['payment_class'] = $pay_type;
  75. $data['user_id'] = $uid;
  76. $data['ext_info'] = json_encode(['subject' => '充值vip支付']);
  77. $data['memo'] = '充值会员支付';
  78. $data['createtime'] = time();
  79. //$data['payment'] = 'miniapp';
  80. $data['payment'] = 'app';
  81. $pay_order = Db::name('pay_order')->insertGetId($data);
  82. //创建回调
  83. $even_data = [];
  84. $even_data['event'] = 'success';
  85. $even_data['class'] = 'app\common\model\Recharge';
  86. $even_data['method'] = 'vippaysucc';
  87. $even_data['args'] = json_encode(['user_id' => $uid, 'days' => $recharge_config['days'], 'vip_level' => $recharge_config['vip_level'], 'gold_num' => $recharge_config['gold_num'], 'money' => $money]);
  88. $even_data['pay_no'] = $pay_no;
  89. $pay_event = Db::name('pay_event')->insertGetId($even_data);
  90. if (!$pay_order || !$pay_event) {
  91. throw new Exception('下单失败');
  92. }
  93. // h5跳转小程序支付链接
  94. $url = self::H5 . "?order_no={$pay_no}&money=$money";
  95. Db::commit();
  96. $this->success('success', [
  97. 'url' => $url
  98. ]);
  99. } catch (Exception $e) {
  100. Db::rollback();
  101. $this->error($e->getMessage());
  102. }
  103. }
  104. /**
  105. * 金币充值
  106. * @return void
  107. */
  108. public function gold_recharge()
  109. {
  110. Db::startTrans();
  111. try {
  112. $rc_id = input_post('rc_id', 0);
  113. $pay_type = 'wechat';
  114. $freemoney = input_post('freemoney', 0, 'intval');
  115. $uid = $this->auth->id;
  116. if (!$rc_id && !$freemoney) {
  117. throw new Exception('请选择或填写充值金额');
  118. }
  119. if (!in_array($pay_type, ['wechat', 'alipay'])) {
  120. throw new Exception('错误的支付类型');
  121. }
  122. //赋值money
  123. if ($rc_id) {
  124. $recharge_config = Db::name('paygold_config')->where('id', $rc_id)->find();
  125. $money = $recharge_config ? $recharge_config['money'] : 0;
  126. $gold = $recharge_config ? $recharge_config['gold'] : 0;
  127. $first_gold = $recharge_config ? $recharge_config['first_gold'] : 0;
  128. $first_vipdays = $recharge_config ? $recharge_config['first_vipdays'] : 0;
  129. $vip_gold = $recharge_config ? $recharge_config['vip_gold'] : 0;
  130. }
  131. //自由输入覆盖
  132. if (!empty($freemoney)) {
  133. $rc_id = 0;
  134. $money = floatval($freemoney);
  135. $bili = config('site.money_to_gold') ?: 10;
  136. $gold = bcmul($money, $bili, 0);
  137. $first_gold = 0;
  138. $first_vipdays = 0;
  139. $vip_gold = 0;
  140. }
  141. if ($money <= 0) {
  142. throw new Exception('支付金额必须大于0');
  143. }
  144. if ($money > 10000) {
  145. throw new Exception('支付金额太大');
  146. }
  147. //查询是不是会员,若不是则不赠送金币
  148. $vip_endtime = Db::name('user_wallet')->where('user_id', $this->auth->id)->value('vip_endtime');
  149. if ($vip_endtime < time()) {
  150. $vip_gold = 0;
  151. }
  152. //创建订单
  153. $data = [];
  154. $data['status'] = 0;
  155. $pay_no = createUniqueNo('P', $uid);
  156. $data['pay_no'] = $pay_no;
  157. $data['money'] = $money;
  158. $data['payment_class'] = $pay_type;
  159. $data['user_id'] = $uid;
  160. $data['ext_info'] = json_encode(['subject' => '充值金币支付']);
  161. $data['memo'] = '充值金币支付';
  162. $data['createtime'] = time();
  163. $data['payment'] = 'app';
  164. $pay_order = Db::name('pay_order')->insertGetId($data);
  165. //创建回调
  166. $even_data = [];
  167. $even_data['event'] = 'success';
  168. $even_data['class'] = 'app\common\model\Recharge';
  169. $even_data['method'] = 'goldpaysucc';
  170. $even_data['args'] = json_encode(['user_id' => $uid, 'gold' => $gold, 'money' => $money, 'pg_id' => $rc_id, 'first_gold' => $first_gold, 'first_vipdays' => $first_vipdays, 'intro_uid' => $this->auth->intro_uid, 'vip_gold' => $vip_gold]);
  171. $even_data['pay_no'] = $pay_no;
  172. $pay_event = Db::name('pay_event')->insertGetId($even_data);
  173. if (!$pay_order || !$pay_event) {
  174. throw new Exception('下单失败');
  175. }
  176. // h5跳转小程序支付链接
  177. $url = self::H5 . "?order_no={$pay_no}&money=$money";
  178. Db::commit();
  179. $this->success('success', [
  180. 'url' => $url
  181. ]);
  182. } catch (Exception $e) {
  183. Db::rollback();
  184. $this->error($e->getMessage());
  185. }
  186. }
  187. /**
  188. * 汇付 支付
  189. */
  190. public function pay()
  191. {
  192. $params = \request()->post();
  193. $order_no = $params['order_no'] ?? '';
  194. if ($params['openid'] != 9696) {
  195. $wxInfo = Cache::get($params['openid'] ?? '');
  196. $openid = $wxInfo['openid'] ?? '';
  197. } else {
  198. $openid = 'ol8qS68vKSgWJ3Unrgfyi3rkakcQ';
  199. }
  200. if (empty($order_no) || empty($openid)) {
  201. $this->error('支付超时,请重新下单支付');
  202. }
  203. // 获取下单信息
  204. $order = Db::name('pay_order')->where('pay_no', $order_no)->find();
  205. if (!$order) {
  206. $this->error('订单不存在,请重新下单支付');
  207. }
  208. $money = $order['money'];
  209. $money = '0.01';// 测试支付
  210. // $order_no = "D0" . time();
  211. $pay = new PayUtil();
  212. $notify_url = CurlUtil::getHttp('/api/hui_pay/pay_notify', false);
  213. if (!$pay->jsPay($openid, $order_no, $money, $order['memo'] ?? '商品下单支付', $notify_url)) {
  214. $this->error($pay->getMessage());
  215. }
  216. $res = $pay->getData();
  217. if (empty($res['data']['pay_info']) || !$pay_info = json_decode($res['data']['pay_info'], true)) {
  218. $this->error('支付信息有误');
  219. // exit;
  220. // 这里 不加 exit 编辑器可能会提示 $pay_info 有可能不存在的变量 是因为 fastadmin 自带 error 写法不规范导致的
  221. }
  222. $this->success('success', [
  223. 'pay_info' => $pay_info,// 这里编辑器可能会提示 有可能不存在的变量 是因为 fastadmin 自带 error 写法不规范导致的
  224. 'order_no' => $order_no
  225. ]);
  226. }
  227. /**
  228. * 支付回调
  229. *
  230. * @param Request $request
  231. * @return string
  232. * @throws \think\db\exception\DataNotFoundException
  233. * @throws \think\db\exception\ModelNotFoundException
  234. * @throws \think\exception\DbException
  235. */
  236. public function pay_notify(Request $request)
  237. {
  238. $params = $request->param();
  239. // 消息主体信息
  240. $resp_data_json = htmlspecialchars_decode($params['resp_data'] ?? '');
  241. $resp_data = json_decode(stripslashes($resp_data_json), true);
  242. unset($params['resp_data']);
  243. LogUtil::info('支付回调参数', self::LOG_MODULE, __FUNCTION__, [
  244. 'params' => $params,
  245. 'resp_data2' => base64_encode($resp_data_json),
  246. 'resp_data' => $resp_data,
  247. ]);
  248. // 校验回调信息
  249. if (empty($params['resp_code']) || $params['resp_code'] != '00000000' || empty($resp_data['req_seq_id'])) {
  250. LogUtil::info('回调信息有误', self::LOG_MODULE, __FUNCTION__, "resp_code error");
  251. return self::response201('支付信息有误');
  252. }
  253. // 校验签名
  254. $huiPay = new PayUtil();
  255. if (!$huiPay->checkSign($params['sign'],$resp_data_json)){
  256. LogUtil::info('签名校验失败', self::LOG_MODULE, __FUNCTION__);
  257. return self::response201('签名校验失败');
  258. }
  259. if ($resp_data['resp_code'] != '00000000'){
  260. LogUtil::info('订单交易失败', self::LOG_MODULE, __FUNCTION__,$resp_data['resp_desc'] ?? '交易失败');
  261. return self::response201('支付信息有误');
  262. }
  263. // 开始处理业务逻辑
  264. $pay_no = $resp_data['req_seq_id'];
  265. LogUtil::info('订单号', self::LOG_MODULE, __FUNCTION__, $pay_no);
  266. //查询订单信息
  267. $order_info = Db::name('pay_order')->where('pay_no', $pay_no)->find();
  268. if (!$order_info) {
  269. LogUtil::info('订单信息不存在', self::LOG_MODULE, __FUNCTION__);
  270. return self::response201('订单信息不存在');
  271. }
  272. if ($order_info['status'] == 3){
  273. LogUtil::info('订单信息_已删除', self::LOG_MODULE, __FUNCTION__, [
  274. 'status' => $order_info['status']
  275. ]);
  276. return self::response201('充值入账失败');
  277. }
  278. if ($order_info['status'] == 1 || $order_info['status'] == 2) {
  279. LogUtil::info('充值入账更新余额失败_status已更新过', self::LOG_MODULE, __FUNCTION__, [
  280. 'status' => $order_info['status']
  281. ]);
  282. return self::response201('充值入账失败');
  283. }
  284. $extendType = '';
  285. //区分vip 还是 冲金币
  286. $is_g = stripos($pay_no, 'P');
  287. if ($is_g !== false) {
  288. $extendType = 'gold';
  289. } else {
  290. $is_v = stripos($pay_no, 'V');
  291. if ($is_v !== false) {
  292. $extendType = 'vip';
  293. }
  294. }
  295. //status已更新过
  296. $result = Db::name('pay_order')->where('pay_no', $pay_no)->setField(['status' => 2]);
  297. if (!$result) {
  298. LogUtil::info('充值入账更新余额失败_status更新状态2不成功', self::LOG_MODULE, __FUNCTION__, [
  299. 'status' => $order_info['status']
  300. ]);
  301. return self::response201('充值入账失败');
  302. }
  303. //你可以在此编写订单逻辑
  304. $payEvent = Db::name('pay_event')->where('pay_no', $pay_no)->find();
  305. $args = json_decode($payEvent['args'] ?? '', true);
  306. $rechargeM = new \app\common\model\Recharge();
  307. if ($extendType == 'gold') {
  308. $payRes = $rechargeM->goldpaysucc($pay_no, $args);
  309. $payTypeStr = '充值';
  310. } elseif ($extendType == 'vip') {
  311. $payRes = $rechargeM->vippaysucc($pay_no, $args);
  312. $payTypeStr = 'vip';
  313. } else {
  314. $payRes = false;
  315. $payTypeStr = '未知支付类型';
  316. }
  317. if (!$payRes) {
  318. LogUtil::info('更新失败', self::LOG_MODULE, __FUNCTION__, $payTypeStr);
  319. return self::response201('更新失败');
  320. }
  321. LogUtil::info('订单处理成功', self::LOG_MODULE, __FUNCTION__, $payTypeStr);
  322. return self::response200('success');
  323. }
  324. /**
  325. * 成功返回
  326. * @param string $message
  327. * @param $result
  328. * @return string
  329. */
  330. private static function response200(string $message = 'success', $result = null)
  331. {
  332. return json_encode([
  333. 'code' => 200,
  334. 'message' => $message,
  335. 'result' => $result,
  336. ], JSON_UNESCAPED_UNICODE);
  337. }
  338. /**
  339. * 失败返回201
  340. * @param string $message
  341. * @param $result
  342. * @return string json
  343. */
  344. private static function response201(string $message = 'error', $result = null)
  345. {
  346. return json_encode([
  347. 'code' => 201,
  348. 'message' => $message,
  349. 'result' => $result
  350. ], JSON_UNESCAPED_UNICODE);
  351. }
  352. }