Recharge.php 16 KB

123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387
  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\RechargeModel;
  7. use app\api\model\UsersModel;
  8. use app\common\controller\Api;
  9. use app\common\lib\Promote;
  10. use app\common\lib\WxPay;
  11. /**
  12. * 充值接口
  13. */
  14. class Recharge extends Api
  15. {
  16. protected $noNeedLogin = '*';
  17. /**
  18. * 充值页面
  19. *
  20. * @ApiTitle (充值页面)
  21. * @ApiSummary (充值页面)
  22. * @ApiMethod (POST)
  23. * @ApiRoute (/api/recharge/rechargePage)
  24. * @ApiParams (name="user_id", type="int", required=true, description="用户id")
  25. */
  26. public function rechargePage()
  27. {
  28. $userId = $this->request->post('user_id');
  29. if (!$userId) {
  30. $this->result('参数错误', [], 100);
  31. }
  32. if (!UsersModel::checkUserExist($userId)) {
  33. $this->result('用户不存在', [], 100);
  34. }
  35. $user = UsersModel::get($userId);
  36. $this->result('ok', ['quota' => $user->vip_discount_quota], 200);
  37. }
  38. /**
  39. * 话费充值
  40. *
  41. * @ApiTitle (话费充值)
  42. * @ApiSummary (话费充值)
  43. * @ApiMethod (POST)
  44. * @ApiRoute (/api/recharge/recharge)
  45. * @ApiParams (name="user_id", type="int", required=true, description="用户id")
  46. * @ApiParams (name="price", type="int", required=true, description="充值面额")
  47. * @ApiParams (name="tel", type="string", required=true, description="充值手机号")
  48. * @ApiParams (name="pay_type", type="int", required=true, description="支付方式 1-微信 2-支付宝")
  49. */
  50. public function recharge()
  51. {
  52. $userId = $this->request->post('user_id');
  53. $price = $this->request->post('price');
  54. $tel = $this->request->post('tel');
  55. $payType = $this->request->post('pay_type');
  56. if (!$userId || !$price || !$tel || !$payType) {
  57. $this->result('参数错误', [], 100);
  58. }
  59. $user = UsersModel::get($userId);
  60. if (!$user) {
  61. $this->result('用户不存在', [], 100);
  62. }
  63. //if ($user->user_level == 1) {
  64. // $this->result('您还不是VIP用户,不能进行充值', [], 100);
  65. //}
  66. // 检测手机号是否可以充值
  67. $url = 'http://op.juhe.cn/ofpay/mobile/telcheck?phoneno='.$tel.'&cardnum='.$price.'&key=f5ff49671fede25118a6b01131ff88a4';
  68. $res = $this->sendRequest($url);
  69. if ($res['error_code'] != 0) {
  70. $this->result('充值错误,请检查手机号码是否正确', [], 100);
  71. }
  72. // 根据会员等级判断是否享折扣
  73. if ($user->user_level == 1) {
  74. $final_fee = $price;
  75. $isDiscount = 0;
  76. } else {
  77. // 折扣额度是否足够
  78. if ($user->vip_discount_quota >= $price) {
  79. $final_fee = $price * 0.88;
  80. $isDiscount = 1;
  81. } else {
  82. $final_fee = $price;
  83. $isDiscount = 0;
  84. }
  85. }
  86. if ($tel == 13287120502 || $tel == 15615490741 || $tel == 15588511702) {
  87. $final_fee = 0.01;
  88. $isDiscount = 0;
  89. }
  90. // 生成充值订单
  91. $out_trade_no = createOutTradeNo();
  92. $orderInfo = array(
  93. 'uid' => $userId,
  94. 'tel' => $tel,
  95. 'price' => $price,
  96. 'create_time' => date('Y-m-d H:i:s', time()),
  97. 'pay_type' => $payType,
  98. 'out_trade_no' => $out_trade_no,
  99. 'final_fee' => $final_fee,
  100. 'is_discount' => $isDiscount
  101. );
  102. $add = RechargeModel::create($orderInfo);
  103. if ($add) {
  104. if ($payType == 1) {
  105. $notify_url = config('site.httpurl').'/api/recharge/recharge_wx_notify';
  106. $payObj = new WxPay();
  107. $getPrePayInfo = $payObj->getPrePayOrder('话费充值', $out_trade_no, ($final_fee * 100), $notify_url);
  108. $getPayInfo = $payObj->getOrder($getPrePayInfo['prepay_id']);
  109. $this->result('订单创建成功', ['payInfoArray' => $getPayInfo], 200);
  110. }
  111. if ($payType == 2) {
  112. $notify_url2 = config('site.httpurl').'/api/recharge/recharge_alipay_notify';
  113. $aop = new AopClient;
  114. $aop->gatewayUrl = "https://openapi.alipay.com/gateway.do";
  115. $aop->appId = config('alipay.app_id');
  116. $aop->rsaPrivateKey = config('alipay.private_key');
  117. $aop->format = "json";
  118. $aop->charset = "utf-8";
  119. $aop->signType = "RSA2";
  120. $aop->alipayrsaPublicKey = config('alipay.public_key');
  121. //实例化具体API对应的request类,类名称和接口名称对应,当前调用接口名称:alipay.trade.app.pay
  122. $request = new AlipayTradeAppPayRequest();
  123. // 订单标题
  124. $subject = '话费充值';
  125. // 订单详情
  126. $body = '话费充值';
  127. // SDK已经封装掉了公共参数,这里只需要传入业务参数
  128. $bizcontent = json_encode([
  129. 'body' => $body,
  130. 'subject' => $subject,
  131. 'out_trade_no' => $out_trade_no,
  132. 'timeout_express' => '90m',
  133. 'total_amount' => $final_fee,
  134. 'product_code' => 'QUICK_MSECURITY_PAY'
  135. ]);
  136. $request->setNotifyUrl($notify_url2);
  137. $request->setBizContent($bizcontent);
  138. // 这里和普通的接口调用不同,使用的是sdkExecute
  139. $response = $aop->sdkExecute($request);
  140. // 注意:这里不需要使用htmlspecialchars进行转义,直接返回即可
  141. // return $response;
  142. $this->result('订单创建成功', ['payInfoString' => $response], 200);
  143. }
  144. } else {
  145. $this->result('订单创建失败', [], 200);
  146. }
  147. }
  148. /**
  149. * 1
  150. * @ApiInternal
  151. */
  152. public function sendRequest($url)
  153. {
  154. $ch = curl_init();
  155. curl_setopt($ch, CURLOPT_URL, $url);
  156. curl_setopt($ch, CURLOPT_SSL_VERIFYPEER, FALSE);
  157. curl_setopt($ch, CURLOPT_SSL_VERIFYHOST, FALSE);
  158. curl_setopt($ch, CURLOPT_RETURNTRANSFER, 1);
  159. $output = curl_exec($ch);
  160. curl_close($ch);
  161. return json_decode($output, true);
  162. }
  163. /**
  164. * 话费充值微信支付异步回调
  165. * @ApiInternal
  166. */
  167. public function recharge_wx_notify()
  168. {
  169. //获取返回的xml格式数据
  170. $payXml = file_get_contents("php://input");
  171. //将xml格式转化为json格式
  172. $jsonXml = json_encode(simplexml_load_string($payXml, 'SimpleXMLElement', LIBXML_NOCDATA));
  173. //将json格式转成数组格式
  174. $result = json_decode($jsonXml, true);
  175. if ($result) {
  176. //如果成功返回
  177. if ($result['return_code'] == 'SUCCESS') {
  178. if ($result['result_code'] == 'SUCCESS') {
  179. // sign 值校验
  180. // 校验时不包含返回的 sign 字段,需踢除 sign 字段
  181. foreach($result as $k => $v) {
  182. if ($k == 'sign') {
  183. $sign = $result[$k];
  184. unset($result[$k]);
  185. };
  186. }
  187. // 按字典排序
  188. ksort($result);
  189. // 转为 url 键值对
  190. $signTemp = http_build_query($result);
  191. // md5处理,$key 为微信商户平台的 api 安全密钥
  192. $key = 'b3ae6bbf3cc4fa017eb169ae219e2c27';
  193. $signTemp = md5($signTemp.'&key='.$key);
  194. // 转大写得最终 sign 值
  195. $resultSign = strtoupper($signTemp);
  196. // 如果sign值正确
  197. if ($sign === $resultSign) {
  198. // 查询订单是否存在
  199. $order = RechargeModel::where('out_trade_no', $result['out_trade_no'])->find();
  200. if (!empty($order)) {
  201. // 修改订单支付状态和支付时间
  202. $updatePayInfo = RechargeModel::where('out_trade_no', $result['out_trade_no'])
  203. ->update(['pay_status' => 1, 'pay_time' => date('Y-m-d H:i:s', time())]);
  204. // 修改充值状态
  205. $updateRechargeStatus = RechargeModel::where('out_trade_no', $result['out_trade_no'])
  206. ->update(['recharge_status' => 2]);
  207. // 扣除优惠额度
  208. if ($order->is_discount == 1) {
  209. UsersModel::where('user_id', $order->uid)->setDec('vip_discount_quota', $order->price);
  210. }
  211. // 调用聚合话费充值接口
  212. $key = 'f5ff49671fede25118a6b01131ff88a4';
  213. $phoneno = $order->tel;
  214. $cardnum = $order->price;
  215. $orderid = $order->out_trade_no;
  216. $sign = md5('JHfe821bbb6cef8f34d3dc7e68efec19ef'.$key.$phoneno.$cardnum.$orderid);
  217. $url = 'http://op.juhe.cn/ofpay/mobile/onlineorder?key='.$key.'&phoneno='.$phoneno.'&cardnum='.$cardnum.'&orderid='.$orderid.'&sign='.$sign;
  218. $this->sendRequest($url);
  219. if ($updatePayInfo && $updateRechargeStatus) {
  220. $successArray = array(
  221. 'return_code' => 'SUCCESS',
  222. 'return_msg' => 'OK'
  223. );
  224. return $this->arrayToXml($successArray);
  225. }
  226. }
  227. }
  228. }
  229. }
  230. }
  231. }
  232. /**
  233. * 数组转xml
  234. * @ApiInternal
  235. */
  236. public function arrayToXml($arr)
  237. {
  238. $xml = "<xml>";
  239. foreach ($arr as $key=>$val)
  240. {
  241. if (is_numeric($val))
  242. {
  243. $xml.="<".$key.">".$val."</".$key.">";
  244. }
  245. else
  246. $xml.="<".$key."><![CDATA[".$val."]]></".$key.">";
  247. }
  248. $xml.="</xml>";
  249. return $xml;
  250. }
  251. /**
  252. * 话费充值支付宝支付异步回调
  253. * @ApiInternal
  254. */
  255. public function recharge_alipay_notify()
  256. {
  257. $params = $this->request->post();
  258. if (!empty($params) && $params['trade_status'] == 'TRADE_SUCCESS') {
  259. // 验证签名
  260. $aop = new AopClient();
  261. $aop->alipayrsaPublicKey = config('alipay.public_key');
  262. // 此处反转义参数中的字符,否则验签不通过
  263. $params['fund_bill_list'] = htmlspecialchars_decode($params['fund_bill_list']);
  264. $checkSign = $aop->rsaCheckV1($params, null, 'RSA2');
  265. if ($checkSign) {
  266. // 是不是向此商户号付款
  267. if ($params['app_id'] == config('alipay.app_id')) {
  268. // 查询订单是否存在
  269. $order = RechargeModel::where('out_trade_no', $params['out_trade_no'])->find();
  270. if (!empty($order)) {
  271. // 修改订单支付状态和支付时间
  272. $updatePayInfo = RechargeModel::where('out_trade_no', $params['out_trade_no'])
  273. ->update(['pay_status' => 1, 'pay_time' => date('Y-m-d H:i:s', time())]);
  274. // 修改充值状态
  275. $updateRechargeStatus = RechargeModel::where('out_trade_no', $params['out_trade_no'])
  276. ->update(['recharge_status' => 2]);
  277. // 扣除优惠额度
  278. if ($order->is_discount == 1) {
  279. UsersModel::where('user_id', $order->uid)->setDec('vip_discount_quota', $order->price);
  280. }
  281. // 调用聚合话费充值接口
  282. $key = 'f5ff49671fede25118a6b01131ff88a4';
  283. $phoneno = $order->tel;
  284. $cardnum = $order->price;
  285. $orderid = $order->out_trade_no;
  286. $sign = md5('JHfe821bbb6cef8f34d3dc7e68efec19ef'.$key.$phoneno.$cardnum.$orderid);
  287. $url = 'http://op.juhe.cn/ofpay/mobile/onlineorder?key='.$key.'&phoneno='.$phoneno.'&cardnum='.$cardnum.'&orderid='.$orderid.'&sign='.$sign;
  288. $this->sendRequest($url);
  289. if ($updatePayInfo && $updateRechargeStatus) {
  290. echo 'success';
  291. }
  292. }
  293. }
  294. }
  295. }
  296. }
  297. /**
  298. * 话费充值聚合异步回调 废弃
  299. * @ApiInternal
  300. */
  301. public function recharge_notify()
  302. {
  303. $params = $this->request->post();
  304. $order = RechargeModel::where('out_trade_no', $params['orderid'])->find();
  305. if ($order) {
  306. $sign = md5('f5ff49671fede25118a6b01131ff88a4'.$order->sporder_id.$order->out_trade_no);
  307. if ($sign == $params['sign']) {
  308. //TODO
  309. }
  310. }
  311. }
  312. /**
  313. * 充值记录
  314. *
  315. * @ApiTitle (充值记录)
  316. * @ApiSummary (充值记录)
  317. * @ApiMethod (POST)
  318. * @ApiRoute (/api/recharge/getRechargeRecord)
  319. * @ApiParams (name="user_id", type="int", required=true, description="用户id")
  320. */
  321. public function getRechargeRecord()
  322. {
  323. $userId = $this->request->post('user_id');
  324. if (!$userId) {
  325. $this->result('参数错误', [], 200);
  326. }
  327. $list = RechargeModel::where('uid', $userId)
  328. ->where('pay_status', 1)
  329. ->field('id,final_fee,create_time,recharge_status')
  330. ->select();
  331. $this->result('ok', $list, 200);
  332. }
  333. /**
  334. * 充值详情
  335. *
  336. * @ApiTitle (充值详情)
  337. * @ApiSummary (充值详情)
  338. * @ApiMethod (POST)
  339. * @ApiRoute (/api/recharge/rechargeDetail)
  340. * @ApiParams (name="id", type="int", required=true, description="记录id")
  341. */
  342. public function rechargeDetail()
  343. {
  344. $id = $this->request->post('id');
  345. if (!$id) {
  346. $this->result('参数错误', [], 200);
  347. }
  348. $order = RechargeModel::get($id);
  349. if (empty($order)) {
  350. $this->result('订单不存在', [], 200);
  351. }
  352. $orderid = $order->out_trade_no;
  353. $key = 'f5ff49671fede25118a6b01131ff88a4';
  354. // 查询聚合订单状态信息
  355. $url = 'http://op.juhe.cn/ofpay/mobile/ordersta?key='.$key.'&orderid='.$orderid;
  356. $result = $this->sendRequest($url);
  357. // 再根据聚合数据的订单状态更新数据库状态达到状态一致
  358. if ($result['error_code'] === 0) {
  359. if ($result['result']['game_state'] == 1) { // 成功
  360. RechargeModel::where('id', $id)->update(['recharge_status' => 2]);
  361. }
  362. if ($result['result']['game_state'] == 9) { // 失败
  363. RechargeModel::where('id', $id)->update(['recharge_status' => 0]);
  364. }
  365. if ($result['result']['game_state'] == 0) { // 充值中
  366. RechargeModel::where('id', $id)->update(['recharge_status' => 1]);
  367. }
  368. }
  369. // 再次查询
  370. $order = RechargeModel::get($id);
  371. // 过滤字段
  372. unset($order->id, $order->uid, $order->is_discount, $order->pay_status, $order->pay_time);
  373. $this->result('ok', $order, 200);
  374. }
  375. }