Supay.php 13 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359
  1. <?php
  2. namespace app\api\controller;
  3. use app\common\controller\Api;
  4. use think\Db;
  5. use addons\epay\library\Service;
  6. use app\common\model\Wallet;
  7. use think\Exception;
  8. /**
  9. * 苏宁支付h5拉起
  10. */
  11. class Supay extends Api
  12. {
  13. protected $noNeedLogin = ['testpay','notify'];
  14. protected $noNeedRight = ['*'];
  15. public function vip_recharge()
  16. {
  17. Db::startTrans();
  18. try {
  19. $rc_id = input('rc_id',0);
  20. $pay_type = input('pay_type','wechat');
  21. $uid = $this->auth->id;
  22. if(!$rc_id){
  23. throw new Exception('请选择会员套餐');
  24. }
  25. //赋值money
  26. $recharge_config = Db::name('payvip_config')->where('id',$rc_id)->find();
  27. $money = $recharge_config['money'];
  28. if($money <= 0) {
  29. throw new Exception('支付金额必须大于0');
  30. }
  31. if($money > 10000){
  32. throw new Exception('支付金额太大');
  33. }
  34. //会员等级冲突
  35. //当前是会员,但是却要向下级续费,直接提示报错
  36. //修改等级,向上立刻改,向下不允许
  37. $wallet_info = model('wallet')->getWallet($this->auth->id);
  38. if($wallet_info['vip_endtime'] > time() && $recharge_config['vip_level'] < $wallet_info['vip_level']){
  39. throw new Exception('当前会员没有过期,不能续费');
  40. }
  41. //创建订单
  42. $data = [];
  43. $data['status'] = 0;
  44. $pay_no = createUniqueNo('V',$uid);
  45. $data['pay_no'] = $pay_no;
  46. $data['money'] = $money;
  47. $data['payment_class'] = $pay_type;
  48. $data['user_id'] = $uid;
  49. $data['ext_info'] = json_encode(['subject'=>'充值vip支付']);
  50. $data['memo'] = '充值会员支付';
  51. $data['createtime'] = time();
  52. //$data['payment'] = 'miniapp';
  53. $data['payment'] = 'app';
  54. $orderid = Db::name('pay_order')->insertGetId($data);
  55. //创建回调
  56. $even_data = [];
  57. $even_data['event'] = 'success';
  58. $even_data['class'] = 'app\common\model\Recharge';
  59. $even_data['method'] = 'vippaysucc';
  60. $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]);
  61. $even_data['pay_no'] = $pay_no;
  62. Db::name('pay_event')->insertGetId($even_data);
  63. $res[] = '';
  64. if ($pay_type == 'wechat') {
  65. //$money = 0.01;
  66. $sandpay = new \app\common\library\SuningPay();
  67. $sandpayParams = [
  68. 'order_no' => $pay_no,
  69. 'goods_name' => '充值vip支付',
  70. 'money' => $money,
  71. 'type' => 'vip',
  72. ];
  73. $sandRes = $sandpay->wechath5($sandpayParams);
  74. if ($sandRes['status'] != 1) {
  75. throw new Exception('失败');
  76. }
  77. $res = json_decode($sandRes['data']['params'],true);
  78. $url = 'https://h5-min-pay-1gczed24bbbe3db8-1317709175.tcloudbaseapp.com/suning-pay.html';
  79. $ress['url'] = $url.'?orderId='.$res['tradeOrderId'];
  80. } else {
  81. //下单
  82. $params = [
  83. 'type' => $pay_type,
  84. 'orderid' => $pay_no,
  85. 'title' => $data['memo'],
  86. 'amount' => $data['money'],
  87. //'amount' => 0.01,
  88. 'method' => 'app',
  89. 'notifyurl' => 'http://app.zhiyinvip001.com/notify.php/paytype/'.$pay_type,
  90. 'returnurl' => '',
  91. ];
  92. $res = Service::submitOrder($params);
  93. }
  94. Db::commit();
  95. $this->success('success',$ress);
  96. } catch (Exception $e) {
  97. Db::rollback();
  98. $this->error($e->getMessage());
  99. }
  100. }
  101. //充值金币 创建订单
  102. public function gold_recharge()
  103. {
  104. exit;
  105. Db::startTrans();
  106. try {
  107. $rc_id = input_post('rc_id',0);
  108. $pay_type = 'wechat';
  109. $freemoney = input_post('freemoney', 0, 'intval');
  110. $uid = $this->auth->id;
  111. if(!$rc_id && !$freemoney){
  112. throw new Exception('请选择或填写充值金额');
  113. }
  114. if (!in_array($pay_type,['wechat','alipay'])) {
  115. throw new Exception('错误的支付类型');
  116. }
  117. //赋值money
  118. if($rc_id){
  119. $recharge_config = Db::name('paygold_config')->where('id',$rc_id)->find();
  120. $money = $recharge_config ? $recharge_config['money']: 0;
  121. $gold = $recharge_config ? $recharge_config['gold'] : 0;
  122. $first_gold = $recharge_config ? $recharge_config['first_gold'] : 0;
  123. $first_vipdays = $recharge_config ? $recharge_config['first_vipdays'] : 0;
  124. $vip_gold = $recharge_config ? $recharge_config['vip_gold'] : 0;
  125. }
  126. //自由输入覆盖
  127. if(!empty($freemoney)){
  128. $rc_id = 0;
  129. $money = floatval($freemoney);
  130. $bili = config('site.rmb_to_gold') ?: 10;
  131. $gold = bcmul($money,$bili,0);
  132. $first_gold = 0;
  133. $first_vipdays = 0;
  134. $vip_gold = 0;
  135. }
  136. if($money <= 0) {
  137. throw new Exception('支付金额必须大于0');
  138. }
  139. if($money > 10000){
  140. throw new Exception('支付金额太大');
  141. }
  142. //查询是不是会员,若不是则不赠送金币
  143. $vip_endtime = Db::name('user_wallet')->where('user_id',$this->auth->id)->value('vip_endtime');
  144. if ($vip_endtime < time()) {
  145. $vip_gold = 0;
  146. }
  147. //创建订单
  148. $data = [];
  149. $data['status'] = 0;
  150. $pay_no = createUniqueNo('P',$uid);
  151. $data['pay_no'] = $pay_no;
  152. $data['money'] = $money;
  153. $data['payment_class'] = $pay_type;
  154. $data['user_id'] = $uid;
  155. $data['ext_info'] = json_encode(['subject'=>'充值金币支付']);
  156. $data['memo'] = '充值金币支付';
  157. $data['createtime'] = time();
  158. $data['payment'] = 'app';
  159. $orderid = Db::name('pay_order')->insertGetId($data);
  160. //创建回调
  161. $even_data = [];
  162. $even_data['event'] = 'success';
  163. $even_data['class'] = 'app\common\model\Recharge';
  164. $even_data['method'] = 'goldpaysucc';
  165. $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]);
  166. $even_data['pay_no'] = $pay_no;
  167. Db::name('pay_event')->insertGetId($even_data);
  168. //$money = 0.01;
  169. $suningpay = new \app\common\library\SuningPay();
  170. $suningpayParams = [
  171. 'order_no' => $pay_no,
  172. 'goods_name' => '充值金币支付',
  173. 'money' => $money,
  174. 'type' => 'gold',
  175. ];
  176. $suningRes = $suningpay->wechath5($suningpayParams);
  177. if ($suningRes['status'] != 1) {
  178. throw new Exception('失败');
  179. }
  180. $res = json_decode($suningRes['data']['params'],true);
  181. $url = 'https://h5-min-pay-1gczed24bbbe3db8-1317709175.tcloudbaseapp.com/suning-pay.html';
  182. $ress['url'] = $url.'?orderId='.$res['tradeOrderId'];
  183. Db::commit();
  184. $this->success('成功', $ress);
  185. } catch (Exception $e) {
  186. Db::rollback();
  187. $this->error($e->getMessage());
  188. }
  189. }
  190. /**
  191. * 支付成功
  192. */
  193. public function notify()
  194. {
  195. //$input_post_data = json_encode($_POST);
  196. //filePut('[wallet][paySucc]充值参数 recharge money post'.$input_post_data);
  197. //$postData = isset($_POST['data']) ? $_POST['data'] : [];
  198. //$sign = isset($_POST['sign']) ? $_POST['sign'] : '';
  199. $postData=file_get_contents('php://input');
  200. $postData = urldecode($postData);
  201. $arr = explode('&', $postData);
  202. $nowArr = array();
  203. foreach($arr as $v) {
  204. $kv = explode('=', $v);
  205. $nowArr[$kv[0]] = $kv[1];
  206. }
  207. if($nowArr['responseCode'] == 'SJ0000' && $nowArr['transactStatus'] == '00')
  208. {
  209. $data = $this->verification($postData);
  210. if($data == 1)
  211. { //验签名通过
  212. //查询订单信息
  213. $PayResult = Db::name('pay_order');
  214. $where['pay_no'] = $nowArr['orderId'];
  215. $order_info = $PayResult->where($where)->find();
  216. if(!$order_info)
  217. {
  218. filePut('[wallet][paySucc]充值入账更新余额失败 recharge money fail 订单数据库检索不到'.$nowArr['orderId'],'suningno.txt');
  219. exit;
  220. }
  221. if ($order_info['status'] == 1 || $order_info['status'] == 2)
  222. {
  223. filePut('[wallet][paySucc]充值入账更新余额失败 recharge money fail status已更新过'.$order_info['status'].'___'.$nowArr['orderId'],'suningno.txt');
  224. echo 'true';exit;
  225. }
  226. $orderCode = $nowArr['orderId'];
  227. $extendType = '';
  228. //区分vip 还是 冲金币
  229. $is_g = stripos($orderCode,'P');
  230. if($is_g !== false)
  231. {
  232. $extendType = 'gold';
  233. }else{
  234. $is_v = stripos($orderCode,'V');
  235. if($is_v !== false)
  236. {
  237. $extendType = 'vip';
  238. }
  239. }
  240. //status已更新过
  241. $_data['status'] = 2;
  242. $result = $PayResult->where($where)->setField($_data);
  243. if ($result || 1)
  244. {
  245. //你可以在此编写订单逻辑
  246. $payEventWhere['pay_no'] = $orderCode;
  247. $payEvent = Db::name('pay_event')->where($payEventWhere)->find();
  248. $args = isset($payEvent['args']) ? $payEvent['args'] : '';
  249. $args = json_decode($args,true);
  250. $rechargeM = new \app\common\model\Recharge();
  251. if ($extendType == 'gold') {
  252. $payRes = $rechargeM->goldpaysucc($orderCode,$args);
  253. $payTypeStr = '充值';
  254. } elseif($extendType == 'vip') {
  255. $payRes = $rechargeM->vippaysucc($orderCode,$args);
  256. $payTypeStr = 'vip';
  257. } else {
  258. $payRes = false;
  259. $payTypeStr = '未知支付类型';
  260. }
  261. if(!$payRes)
  262. {
  263. filePut('[wallet][paySucc]'.$payTypeStr.'更新失败请查看问题'.$orderCode,'suningno.txt');
  264. exit;
  265. }
  266. filePut('[wallet][paySucc]回调成功'.$orderCode,'suningno.txt');
  267. echo "true";exit;
  268. }
  269. filePut('[wallet][paySucc]充值入账更新余额失败 更新状态2不成功'.$orderCode."\n".' data:'.$postData."\n",'suningno.txt');
  270. exit;
  271. }
  272. }
  273. else
  274. {
  275. filePut(' recharge gold fail 用户交易操作失败'.$postData,'suningno.txt');
  276. echo '支付异常';exit;
  277. }
  278. filePut('[wallet][paySucc]充值入账更新余额失败 验证签名不通过'.$orderCode."\n".' data:'.$postData."\n",'suningno.txt');
  279. echo 'Fail';exit;
  280. }
  281. protected function verification($str)
  282. {
  283. if($str != '') {
  284. $arr = explode('&', $str);
  285. $nowArr = array();
  286. foreach($arr as $v) {
  287. $kv = explode('=', $v);
  288. $nowArr[$kv[0]] = $kv[1];
  289. }
  290. ksort($nowArr);
  291. $newstr = '';
  292. foreach($nowArr as $key => $value) {
  293. if($value == '' || $key == 'signature' || $key== 'signAlgorithm') {
  294. continue;
  295. }
  296. if($key == 'version')
  297. {
  298. $newstr .= $key.'='.$value;
  299. } else
  300. {
  301. $newstr .= $key.'='.$value.'&';
  302. }
  303. }
  304. // print_r($nowArr);
  305. //var_dump($newstr);
  306. $sign = base64_decode($nowArr['signature']);
  307. // exit;
  308. $newstr = strtoupper(md5($newstr));
  309. // var_dump($newstr);
  310. $pubfile = file_get_contents(APP_PATH.'common/library/suning/publicnew.pem');
  311. $ret = openssl_verify($newstr,$sign,$pubfile);
  312. return $ret;
  313. }
  314. }
  315. }