$this->getGateUrl() . '/gateway/ipay', 'debug' => 1,//调试 'logDir' => to_linux_path(storage_path() . '/logs/'), 'mch_id' => $this->pay_gateway['app_id'], //客户交易者账号 'token' => $this->pay_gateway['app_key'], 'interface' => $this->pay_gateway['mch_id'], 'return_url' => action('Pay\PayController@returnNotify', ['type' => $type]), 'notify_url' => action('Pay\PayController@notify', ['type' => $type]), ]; $this->pay_config = $config; } //统一下单支付 public function pay($pay_data) { //商户号 $accountId = $this->pay_config['mch_id']; //商户秘钥 $key = $this->pay_config['token']; $payData = [ 'merid' => $accountId,//商户 'businessOrderId' => strval($pay_data['out_trade_no']),//订单号 'orderName' => mt_rand(100000, 999999), //商品名称 'tradeMoney' => intval($pay_data['total_amount']) * 100, //订单金额(分) 'payType' => 2,//$this->pay_gateway['mch_id'],//对应的三方通道 'asynURL' => strval($this->pay_config['notify_url']),// //异步回调地址 'ClientID' => uniqid(), //用户请求id ]; $sign=$this->md5KeyLower($payData,$key,['ClientID']);//签名 $payData['sign'] = $sign; $this->debugLog('商户'.$accountId, []); $this->debugLog('下单接口请求内容'.json_encode($payData,true), []); $res = Curl::to($this->pay_config['url'])->withData($payData)->post(); $this->debugLog('下单接口返回' . is_array($res) ? json_encode($res, true) : json_encode(json_decode($res, 256), true)); $res=is_array($res)?$res:json_decode($res,256); if (is_array($res) && isset($res['code'])) { if ($res['code'] == 1000) { $url = $res['info']['codeurl']; $orderNo = $payData['businessOrderId']; $tradeNo = $res['info']['orderid']; $order = Order::where('order_sn', $orderNo)->first(); $order->pay_order_sn = $tradeNo; $order->save(); return redirect()->away($url);//跳转支付 } else { return $res; } } else { return $res; } } /** * 检查订单 * @param $orderid * @return bool|string */ private function checkOrder($orderid){ $url = $this->getGateUrl() . '/GateWay/OrderQuery'; $payData['merid'] = $this->pay_gateway['app_id']; $payData['businessOrderId'] = $orderid; $key = $this->pay_config['token']; $sign = $this->md5KeyLower($payData, $key); $payData['sign'] = $sign; $this->debugLog('查询接口请求' . json_encode($payData, true) ); $res = Curl::to($url)->withData($payData)->post(); $res = is_array($res) ? $res : json_decode($res, 256); $this->debugLog('查询接口返回' . is_array($res) ? json_encode($res, true) : json_encode(json_decode($res, 256), true)); if ($res['code'] == 1000) { if ($res['info']['orderState'] == 1 && $res['info']['businessOrderId']==$orderid) { return true; } } return false; } //统一验证签名 public function verify() { $para = $this->result_data; // $para['total_amount'] = $this->order['order_money'];//订单金额必须跟支付金额一致 $sign = $para['sign']; unset($para['sign']); $token = $this->pay_config['token']; $md5sign = $this->md5KeyLower($para, $token,['clientid','sign']); if ($md5sign == $sign) { return true; } else { return false; } } //返回给支付商的成功 public function success() { echo 'success';exit; } //统一回调处理 /** * 回调第一部,如果是同步,直接回传 * @param $return * @param $pay_type * @param $request * @return bool|\Illuminate\Http\RedirectResponse */ public function notify($return, $pay_type, $request) { $this->setLogName($pay_type); $this->debugLog('回调原始数据', $request->all()); $this->pay_type = $pay_type; $this->request = $request; //取得订单,是否有此订单,如果没有直接返回空 $order_sn = $request->input('orderid'); //取得订单 $this->getOrder($order_sn); $return = ($return) ? 1 : 0; if ($return) { //判断是否是商户id if ($this->order->merchant_id == 0) { return redirect()->route('web.user.index'); } // return $this->merchantReturnSend(); } /* $this->order = Order::where('order_sn', $order_sn)->first(); if(empty($this->order)) { return false; }*/ //取得通道id $this->gateway = $this->pay_gateway = config('gateway.config')[$this->gateway_id]; //dd($this->gateway); //取得支付配置 $this->payConfig($this->pay_type); $this->result_data = $request->all(); //检查三方的订单 $checkorder = $this->checkOrder($order_sn); if (!$checkorder) { $this->debugLog('三方订单未支付成功'); return false; } //dump($this->result_data); //进行验证签名 if (!$this->verify()) { $this->debugLog('签名验证失败'); return false; } return $this->payNotify(); } /** * 同步回调 * @return \Illuminate\Http\RedirectResponse */ public function returnPay() { //判断是否是商户id if ($this->order->merchant_id == 0) { return redirect()->route('web.user.index'); } // return $this->merchantReturnSend(); } //异步回调 /** * 获得通道信息,取得支付配置模型,验证签名, * @return bool|void */ private function payNotify() { //取得通道信息内容 //判断有没支付成功先,没有的话,直接就没有下一步了 $pay_status = $this->result_data['orderState']; //dd($this->result_data); if ($pay_status != 1) { $this->debugLog('订单支付失败', $this->request->all()); return false; } $data = $this->result_data; $this->pay_data = $this->result_data; //如果应答状态不是01/A6表示支付失败 $this->debugLog(json_encode($data,true)); $this->debugLog('-------------' . $this->order->order_sn . '--选择----' . $this->pay_type . '----' . $this->pay_method . '----开始回调处理-------------------------'); if (empty($data)) { return $this->debugLog('验证签名失败', $data); } try { $pay_cn_name = ''; //统一格式化 $this->unionHandle($data); //支付返回信息 $this->pay_data = $data; $this->debugLog('验证签名成功,数据', $data); //更新订单 $this->updateOrder(); $this->debugLog('updateOrder Ok'); } catch (\Exception $exception) { $this->debugLog('失败异常内容:' . $exception->getMessage()); // if (!in_array($data['respCode'], ['00', 'A6'])) { // $this->debugLog('订单支付失败', $this->request->all()); // return false;//支付失败 // } $this->order->notify_status = 1;//支付回调成功 $this->order->pay_status = 1;//支付状态 $this->order->pay_ok_at = date('Y-m-d H:i:s'); $this->order->save();//保存 Log::channel('pay_success')->info($pay_cn_name . '支付成功,但处理订单失败', ['data' => json_encode($data)]); } } /** * 统一格式化 */ public function unionHandle($data) { $result = [ 'order_sn' => $data['orderid'], 'pay_order_sn' => $data['payOrderId'], 'pay_money' => $data['money']/100, 'pay_type' => 'baobeiyepay_alipay', 'account' => '' ]; $this->setResult($result); Log::channel('pay_success')->info('支付成功', [ 'pay_type' => $result['pay_type'], 'pay_money' => $result['pay_money'], 'account' => $result['account'], 'order_sn' => $result['order_sn'], 'pay_order_sn' => $result['pay_order_sn']]); } /** * 这个调试需要配置在config/logging.php * @param $pay_type */ public function setLogName($pay_type) { $name = ''; switch ($pay_type) { case 'usdtpay': $name = 'usdtpay_'; case 'baobeiyepay': $name='baobeiyepay'; break; } $this->log_name = $name; } }