Pay.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375
  1. <?php
  2. // +----------------------------------------------------------------------
  3. // | ThinkAdmin
  4. // +----------------------------------------------------------------------
  5. // | 版权所有 2014~2019 广州楚才信息科技有限公司 [ http://www.cuci.cc ]
  6. // +----------------------------------------------------------------------
  7. // | 官方网站: http://demo.thinkadmin.top
  8. // +----------------------------------------------------------------------
  9. // | 开源协议 ( https://mit-license.org )
  10. // +----------------------------------------------------------------------
  11. // | gitee 代码仓库:https://gitee.com/zoujingli/ThinkAdmin
  12. // | github 代码仓库:https://github.com/zoujingli/ThinkAdmin
  13. // +----------------------------------------------------------------------
  14. namespace app\api\controller;
  15. use ClassesWithParents\D;
  16. use EasyWeChat\Factory;
  17. use think\Controller;
  18. use think\Db;
  19. use think\Exception;
  20. use AlibabaCloud\Client\AlibabaCloud;
  21. use AlibabaCloud\Client\Exception\ClientException;
  22. use AlibabaCloud\Client\Exception\ServerException;
  23. use function AlibabaCloud\Client\value;
  24. /**
  25. * 支付管理类
  26. * Class Refund
  27. * @package app\api\controller\Refund
  28. */
  29. class Pay extends Controller
  30. {
  31. /**
  32. * 微信支付--商品支付成功回调订单
  33. */
  34. public function WxOrderNotify(){
  35. $payXml = file_get_contents("php://input");
  36. //将xml格式转化为json格式
  37. $jsonXml = json_encode(simplexml_load_string($payXml, 'SimpleXMLElement', LIBXML_NOCDATA));
  38. //将json格式转成数组格式 $result['out_trade_no']
  39. $result = json_decode($jsonXml, true);
  40. if ($result['return_code'] == 'SUCCESS' && $result['result_code'] == 'SUCCESS') {
  41. $return = $this->dealData($result);
  42. if ($return){
  43. $arr = array(
  44. 'return_code' => 'SUCCESS',
  45. 'return_msg' => 'OK',
  46. );
  47. return $this->arrayToXml($arr);
  48. }else{
  49. file_put_contents("order_pay_error.txt", file_get_contents("php://input") . "\n" . json_encode($result) . "\n" . "\n", FILE_APPEND);
  50. }
  51. }
  52. }
  53. /**
  54. * 支付宝支付--支付成功回调订单
  55. */
  56. public function alipayOrderNotify(){
  57. $result = input('post.');
  58. if ($result['trade_status'] == 'TRADE_SUCCESS' || $result['trade_status'] == 'TRADE_FINISHED') {
  59. $return = $this->dealData($result);
  60. if ($return){
  61. echo 'success';
  62. }else{
  63. file_put_contents("order_alipay_error.txt", file_get_contents("php://input") . "\n" . json_encode($result) . "\n" . "\n", FILE_APPEND);
  64. }
  65. }
  66. }
  67. /**
  68. * 处理数据库信息
  69. * @param $result
  70. * @return bool
  71. */
  72. function dealData($result){
  73. Db::startTrans();
  74. try {
  75. $order = Db::name('store_order')->where('order_no',$result['out_trade_no'])->find();
  76. // if($order['status'] != 0){
  77. // return true;
  78. // }
  79. Db::name('store_order')
  80. ->where('order_no',$result['out_trade_no'])
  81. ->update(['status'=>1,'pay_at'=>date('Y-m-d H:i:s'),'return_success_info'=>json_encode($result,true)]);
  82. $info = Db::name('store_order_info')->where('order_id',$order['id'])->count();
  83. if ($info>=$order['num']){
  84. Db::commit();
  85. return true;
  86. }
  87. $array = [];
  88. for ($i=0;$i<$order['num'];$i++){
  89. //获取排名
  90. $rank = getRanking($order['c_id'])+1;
  91. $tag = getTag($order['c_id'],$rank,$order['inventory']);
  92. saveRanking($order['c_id']);
  93. $company = '象寻数字科技(上海)有限公司';
  94. // $hash = getCompanyHash($order['c_id']);
  95. // $company_hash = $hash['hash'];
  96. // $ddcid = Db::name('hash')->where('hash',$hash['hash'])->value('ddcid');
  97. // $company_hash_time = $hash['create_at'] ? $hash['create_at'] : date('Y-m-d H:i:s');
  98. //Db::name('hash')->where('hash',$hash['hash'])->update(['status'=>1]);
  99. $company_hash = Db::name('hash2')->where('goods_id',$order['c_id'])->where('success',1)->field('hash,ddcid')->find();
  100. $company_hash_time = date('Y-m-d H:i:s');
  101. $collectors_hash = '';
  102. $date = [
  103. 'order_id'=>$order['id'],
  104. 'order_no'=>get_order_sn(),
  105. 'tag'=>$tag,
  106. 'mid'=>$order['mid'],
  107. 'c_id'=>$order['c_id'],
  108. 'pro_info'=>$order['pro_info'],
  109. 'company'=>$company,
  110. 'company_hash'=>$company_hash['hash'],
  111. 'company_hash_time'=>$company_hash_time,
  112. 'ddcid'=>$company_hash['ddcid'],
  113. 'collectors_hash'=>$collectors_hash,
  114. 'collectors_hash_time'=>'',
  115. 'type'=>2
  116. ];
  117. $array[] = $date;
  118. }
  119. Db::name('store_order_info')->insertAll($array);
  120. //送积分
  121. $by_collection_integral = getConfigValue('by_collection_integral');
  122. if ($by_collection_integral){
  123. $by_collection_integral = bcmul($by_collection_integral,$order['num']);
  124. memberMoneyChange($by_collection_integral,1,$order['mid'],'购买藏品',1,$order['id']);
  125. }
  126. Db::commit();
  127. return true;
  128. } catch (\Exception $e){
  129. Db::rollback();
  130. return false;
  131. }
  132. }
  133. /**
  134. * 数组转xml
  135. * @ApiInternal
  136. */
  137. public function arrayToXml($arr)
  138. {
  139. $xml = "<xml>";
  140. foreach ($arr as $key => $val) {
  141. if (is_numeric($val)) {
  142. $xml .= "<" . $key . ">" . $val . "</" . $key . ">";
  143. } else
  144. $xml .= "<" . $key . "><![CDATA[" . $val . "]]></" . $key . ">";
  145. }
  146. $xml .= "</xml>";
  147. return $xml;
  148. }
  149. /**
  150. * 微信充值支付--支付成功回调订单
  151. */
  152. public function WxRechargeNotify(){
  153. $payXml = file_get_contents("php://input");
  154. //将xml格式转化为json格式
  155. $jsonXml = json_encode(simplexml_load_string($payXml, 'SimpleXMLElement', LIBXML_NOCDATA));
  156. //将json格式转成数组格式 $result['out_trade_no']
  157. $result = json_decode($jsonXml, true);
  158. file_put_contents("wx_recharge_pay_error.txt", file_get_contents("php://input") . "\n" . json_encode($result) . "\n" . "\n", FILE_APPEND);
  159. if ($result['return_code'] == 'SUCCESS' && $result['result_code'] == 'SUCCESS') {
  160. $total_fee = $result['total_fee'];
  161. $total_fee1 = $total_fee / 100;
  162. $transaction_id = $result['transaction_id'];
  163. $order = Db::name('SystemRecharge')->where('number',$result['out_trade_no'])->find();
  164. if($order['status'] == 0) {
  165. if ($order['need_pay'] == $total_fee1) {
  166. Db::startTrans();
  167. try {
  168. $order_upd['status'] = 1;
  169. $order_upd['pay_time'] = time();
  170. $order_upd['pay_way'] = '微信';
  171. $order_upd['pay_money'] = $total_fee1;
  172. $order_upd['pay_number'] = $transaction_id;
  173. Db::name('SystemRecharge')
  174. ->where('number',$result['out_trade_no'])
  175. ->update($order_upd);
  176. $this -> addMoney($order['mem_id'],$order['money']);
  177. Db::commit();
  178. $arr = array(
  179. 'return_code' => 'SUCCESS',
  180. 'return_msg' => 'OK',
  181. );
  182. return $this->arrayToXml($arr);
  183. } catch (\Exception $e) {
  184. Db::rollback();
  185. $msg = $e->getMessage();
  186. file_put_contents("wx_recharge_pay_error.txt", $msg, FILE_APPEND);
  187. }
  188. }else{
  189. file_put_contents("wx_recharge_pay_error.txt", '金额验证失败', FILE_APPEND);
  190. }
  191. }else{
  192. file_put_contents("wx_recharge_pay_error.txt", '订单状态错误', FILE_APPEND);
  193. }
  194. }
  195. }
  196. /**
  197. * 支付宝充值支付--支付成功回调订单
  198. */
  199. public function alipayRechargeNotify(){
  200. $result = input('post.');
  201. $_POST = $result;
  202. file_put_contents("ali_recharge_pay_error.txt", file_get_contents("php://input") . "\n" . json_encode($result) . "\n" . "\n", FILE_APPEND);
  203. if ($result['trade_status'] == 'TRADE_SUCCESS' || $result['trade_status'] == 'TRADE_FINISHED') {
  204. if ($_POST['trade_status'] == 'TRADE_SUCCESS') {
  205. $out_trade_no = $_POST['out_trade_no'];//订单号
  206. $buyer_pay_amount = $_POST['receipt_amount'];//实际支付金额
  207. $order = Db::name('SystemRecharge')->where('number',$out_trade_no)->find();
  208. if($order['status'] == 0) {
  209. if ($order['need_pay'] == $buyer_pay_amount) {
  210. Db::startTrans();
  211. try {
  212. $order_upd['status'] = 1;
  213. $order_upd['pay_time'] = time();
  214. $order_upd['pay_way'] = '支付宝';
  215. $order_upd['pay_money'] = $buyer_pay_amount;
  216. $order_upd['pay_number'] = $_POST['trade_no'];
  217. Db::name('SystemRecharge')
  218. ->where('number',$result['out_trade_no'])
  219. ->update($order_upd);
  220. $this -> addMoney($order['mem_id'],$order['money']);
  221. Db::commit();
  222. echo 'success';
  223. } catch (\Exception $e) {
  224. Db::rollback();
  225. $msg = $e->getMessage();
  226. file_put_contents("ali_recharge_pay_error.txt", $msg, FILE_APPEND);
  227. }
  228. }else{
  229. file_put_contents("ali_recharge_pay_error.txt", '金额验证失败', FILE_APPEND);
  230. }
  231. }else{
  232. file_put_contents("ali_recharge_pay_error.txt", '订单状态错误', FILE_APPEND);
  233. }
  234. } else {
  235. file_put_contents("ali_recharge_pay_error.txt", '验证失败', FILE_APPEND);
  236. }
  237. }
  238. }
  239. function addMoney($mem_id,$money){
  240. if($money <= 0) return ['code'=>true,'message'=>'成功'];
  241. $old_money = Db::name('StoreMember') -> where('id',$mem_id)->field('id,money')->find();
  242. Db::name('StoreMember') -> where('id',$mem_id)->setInc('money',$money);
  243. $sre_data['mem_id']=$mem_id;
  244. $sre_data['change']=$money;
  245. $sre_data['pm']=1;
  246. $sre_data['old_money']=$old_money['money'];
  247. $sre_data['reason']='在线充值';
  248. $sre_data['time']=time();
  249. Db::name('SystemMoneyRecord')->insert($sre_data);
  250. }
  251. public function validateApplePay()
  252. {
  253. $receipt_data = input('receipt_data'); //receipt_data参数
  254. if (!$receipt_data) $this->error('receipt_data参数错误');
  255. $order_no = input('order_no'); //订单号
  256. if (!$order_no) $this->error('订单号参数错误');
  257. $order_type = input('order_type'); //订单类型 1订单 2充值
  258. if (!$order_type) $this->error('订单类型参数错误');
  259. $pay_money = input('pay_money'); //支付金额
  260. if (!$pay_money) $this->error('支付金额参数错误');
  261. // 验证参数
  262. if (strlen($receipt_data) < 20) {
  263. $this->error("非法参数");
  264. }
  265. // // 请求验证
  266. $html = $this->acurl($receipt_data);
  267. $data = json_decode($html, true);
  268. // 如果是沙盒数据 则验证沙盒模式
  269. if ($data['status'] == '21007') {
  270. // 请求验证
  271. $html = $this->acurl($receipt_data, 1);
  272. $data = json_decode($html, true);
  273. $data['sandbox'] = '1';
  274. }
  275. file_put_contents("validate_apple_pay_error.txt", $html . "\n" . json_encode($data) . "\n" . "\n", FILE_APPEND);
  276. // 判断是否购买成功
  277. if (intval($data['status']) === 0) {
  278. switch ($order_type){
  279. case '1':
  280. //订单
  281. $result['out_trade_no'] = $order_no;
  282. $return = $this->dealData($result);
  283. if ($return){
  284. $this->success("操作成功");
  285. }else{
  286. $this->error("操作失败");
  287. }
  288. break;
  289. case '2':
  290. //充值
  291. $out_trade_no = $order_no;//订单号
  292. $order = Db::name('SystemRecharge')->where('number',$out_trade_no)->find();
  293. $buyer_pay_amount = $pay_money;//实际支付金额
  294. if($order['status'] == 0) {
  295. if ($order['need_pay'] == $buyer_pay_amount) {
  296. Db::startTrans();
  297. try {
  298. $order_upd['status'] = 1;
  299. $order_upd['pay_time'] = time();
  300. $order_upd['pay_way'] = '苹果内付';
  301. $order_upd['pay_money'] = $buyer_pay_amount;
  302. $order_upd['pay_number'] = $out_trade_no;
  303. Db::name('SystemRecharge')
  304. ->where('number',$out_trade_no)
  305. ->update($order_upd);
  306. $this -> addMoney($order['mem_id'],$order['money']);
  307. Db::commit();
  308. $this->success("购买成功");
  309. } catch (\Exception $e) {
  310. Db::rollback();
  311. $msg = $e->getMessage();
  312. $this->error($msg);
  313. }
  314. }else{
  315. $this->error('金额验证失败');
  316. }
  317. }else{
  318. $this->error('订单状态错误');
  319. }
  320. break;
  321. default:
  322. $this->error('订单类型错误');
  323. }
  324. } else {
  325. $this->error("购买失败".$data['status']);
  326. }
  327. }
  328. /**
  329. * 21000 App Store不能读取你提供的JSON对象
  330. * 21002 receipt-data域的数据有问题
  331. * 21003 receipt无法通过验证
  332. * 21004 提供的shared secret不匹配你账号中的shared secret
  333. * 21005 receipt服务器当前不可用
  334. * 21006 receipt合法,但是订阅已过期。服务器接收到这个状态码时,receipt数据仍然会解码并一起发送
  335. * 21007 receipt是Sandbox receipt,但却发送至生产系统的验证服务
  336. * 21008 receipt是生产receipt,但却发送至Sandbox环境的验证服务
  337. */
  338. function acurl($receipt_data, $sandbox = 0)
  339. {
  340. //小票信息
  341. $POSTFIELDS = array("receipt-data" => $receipt_data);
  342. $POSTFIELDS = json_encode($POSTFIELDS);
  343. //正式购买地址 沙盒购买地址
  344. $url_buy = "https://buy.itunes.apple.com/verifyReceipt";
  345. $url_sandbox = "https://sandbox.itunes.apple.com/verifyReceipt";
  346. $url = $sandbox ? $url_sandbox : $url_buy;
  347. //简单的curl
  348. $ch = curl_init($url);
  349. curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
  350. curl_setopt($ch, CURLOPT_POST, 1);
  351. curl_setopt($ch, CURLOPT_POSTFIELDS, $POSTFIELDS);
  352. $result = curl_exec($ch);
  353. curl_close($ch);
  354. return $result;
  355. }
  356. }