* @since 1.0 */ class WxpayH5 extends Service { public $devide; public $configFile; public $subjectMaxLength = 30; public $tradeType; public $scanCodeBody = '微信H5支付'; public $deviceInfo = 'WEB'; public $expireTime = 600; protected $_order; // 允许更改的订单状态,不存在这里面的订单状态不允许修改 protected $_allowChangOrderStatus; public function init() { ini_set('date.timezone', 'Asia/Shanghai'); parent::init(); $wxpayConfigFile = Yii::getAlias($this->configFile); if (!is_file($wxpayConfigFile)) { throw new InvalidConfigException('wxpay config file:['.$wxpayConfigFile.'] is not exist'); } $appId = Yii::$app->store->get('payment_wxpay', 'wechat_service_app_id'); $appSecret = Yii::$app->store->get('payment_wxpay', 'wechat_service_app_secret'); $mchKey = Yii::$app->store->get('payment_wxpay', 'merchant_key'); $mchId = Yii::$app->store->get('payment_wxpay', 'merchant_mch_id'); define('WX_APP_ID', $appId); define('WX_APP_SECRET', $appSecret); define('WX_MCH_KEY', $mchKey); define('WX_MCH_ID', $mchId); //echo $appId;exit; require_once($wxpayConfigFile); $wxpayApiFile = Yii::getAlias('@fecshop/lib/wxpay/lib/WxPay.Api.php'); //$wxpayDataFile = Yii::getAlias('@fecshop/lib/wxpay/lib/WxPay.Data.php'); $wxpayNotifyFile = Yii::getAlias('@fecshop/lib/wxpay/lib/WxPay.Notify.php'); //$wxpayExceptionFile = Yii::getAlias('@fecshop/lib/wxpay/lib/WxPay.Exception.php'); $wxpayJsApiPayPayFile = Yii::getAlias('@fecshop/lib/wxpay/example/WxPay.JsApiPay.php'); //$wxpayNativePayFile = Yii::getAlias('@fecshop/lib/wxpay/example/WxPay.NativePay.php'); $wxpayLogFile = Yii::getAlias('@fecshop/lib/wxpay/example/log.php'); require_once($wxpayApiFile); //require_once($wxpayDataFile); require_once($wxpayNotifyFile); //require_once($wxpayExceptionFile); require_once($wxpayJsApiPayPayFile); require_once($wxpayLogFile); //交易类型 //JSAPI--公众号支付、NATIVE--原生扫码支付、APP--app支付,统一下单接口trade_type的传参可参考这里 //MICROPAY--刷卡支付,刷卡支付有单独的支付接口,不调用统一下单接口 $this->tradeType = 'MWEB'; $this->_allowChangOrderStatus = [ Yii::$service->order->payment_status_pending, Yii::$service->order->payment_status_processing, ]; } /** * 接收IPN消息的url,接收微信支付的异步消息,进而更改订单状态。 */ public function ipn() { $notifyFile = Yii::getAlias('@fecshop/services/payment/wxpay/notify.php'); require_once($notifyFile); \Yii::info('begin ipn', 'fecshop_debug'); $notify = new \PayNotifyCallBack(); $notify->Handle(false); } /** * @param $data | Array 数据格式如下: * array(18) { * ["appid"]=> string(18) "wx426b3015555a46be" * ["attach"]=>string(24) "微信支付测试产品" * ["bank_type"]=>string(3) "CFT" * ["cash_fee"]=>string(1) "1" * ["device_info"]=>string(3) "WEB" * ["fee_type"]=> string(3) "CNY" * ["is_subscribe"]=>string(1) "N" * ["mch_id"]=>string(10) "1900009851" * ["nonce_str"]=> string(32) "e91xn1hwgyw9ox5zecdag1l86vrhi94l" * ["openid"]=>string(28) "oHZx6uKw5nrwZmEfgIX8poeQIucw" * ["out_trade_no"]=>string(10) "1100000953" * ["result_code"]=>string(7) "SUCCESS" * ["return_code"]=>string(7) "SUCCESS" * ["sign"]=>string(32) "589AC2046E667584FF1967C3C091259A" * ["time_end"]=>string(14) "20171106160124" * ["total_fee"]=>string(1) "1" * ["trade_type"]=>string(6) "NATIVE" * ["transaction_id"]=>string(28) "4200000006201711062859872774" * } * 在微信sdk验证数据安全性后,会执行该函数,用来验证订单的金额的正确性 * 如果订单数据没有问题,则更改订单状态。 */ public function ipnUpdateOrder($data) { \Yii::info('ipn order process', 'fecshop_debug'); $incrementId = $data['out_trade_no']; $transaction_id = $data['transaction_id']; $total_fee = $data['total_fee']; $fee_type = $data['fee_type']; if ($incrementId && $transaction_id && $total_fee) { $this->_order = Yii::$service->order->getByIncrementId($incrementId); Yii::$service->payment->setPaymentMethod($this->_order['payment_method']); $base_grand_total = $this->_order['base_grand_total']; $order_total_amount = Yii::$service->page->currency->getCurrencyPrice($base_grand_total, 'CNY'); \Yii::info('check order totla amouont['.($order_total_amount * 100).' == '.$total_fee.']', 'fecshop_debug'); // 微信支付的人民币单位为分 if(bccomp($order_total_amount * 100, $total_fee) !== 0){ return false; } \Yii::info('updateOrderInfo', 'fecshop_debug'); // 更改订单状态 if ($this->updateOrderInfo($incrementId, $transaction_id, false)) { //支付成功调用服务执行订单状态改变,清空购物车和发送邮件操作 \Yii::info('updateOrderInfo Success', 'fecshop_debug'); return true; } } } public function getOpenidUrl($baseUrl) { $tools = new \JsApiPay(); return $tools->GetOpenidUrl($baseUrl); } function postXmlCurl($xml,$url,$second = 30){ $ch = curl_init(); //设置超时 curl_setopt($ch, CURLOPT_TIMEOUT, $second); curl_setopt($ch,CURLOPT_URL, $url); curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,FALSE); curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,FALSE); //设置 header curl_setopt($ch, CURLOPT_HEADER, FALSE); //要求结果为字符串且输出到屏幕上 curl_setopt($ch, CURLOPT_RETURNTRANSFER, TRUE); //post 提交方式 curl_setopt($ch, CURLOPT_POST, TRUE); curl_setopt($ch, CURLOPT_POSTFIELDS, $xml); //运行 curl $data = curl_exec($ch); //返回结果 if($data){ curl_close($ch); return $data; }else{ $error = curl_errno($ch); curl_close($ch); echo "curl 出错,错误码:$error"."
"; } } /** * @param $code | string 传递的微信code */ public function getScanCodeStart($code = "") { // 根据订单得到json格式的微信支付参数。 $trade_info = $this->getStartBizContentAndSetPaymentMethod(); if (!$trade_info) { Yii::$service->helper->errors->add('generate wxpay bizContent error'); return false; } $money= $trade_info['total_amount'] * 100; //微信支付的单位为分,所以要乘以100; //充值金额 微信支付单位为分 $userip = Yii::$service->helper->getCustomerIp();; //获得用户设备 IP $appid = \WxPayConfig::APPID ; //应用 APPID $mch_id = \WxPayConfig::MCHID; //微信支付商户号 $key = \WxPayConfig::KEY; //微信商户 API 密钥 $out_trade_no = $trade_info['increment_id'];//平台内部订单号 $nonce_str = Yii::$service->helper->createNoncestr();//随机字符串 $body = $this->scanCodeBody;//内容 $total_fee = $money; //金额 $spbill_create_ip = $userip; //IP $notify_url = Yii::$service->payment->getStandardIpnUrl(); //回调地址 $trade_type = $this->tradeType;//交易类型 具体看 API 里面有详细介绍 //$scene_info ='{"h5_info":{"type":"Wap","wap_url":"http://qq52o.me","wap_name":"支付"}}';//场景信息 必要参数 //echo $openId;exit; $wap_url = Yii::$service->url->homeUrl(); $scene_info ='{"h5_info":{"type":"Wap","wap_url":"'.$wap_url.'","wap_name":"'.$this->scanCodeBody.'"}}';//场景信息 必要参数 $signA ="appid=$appid&attach=$out_trade_no&body=$body&mch_id=$mch_id&nonce_str=$nonce_str¬ify_url=$notify_url&out_trade_no=$out_trade_no&scene_info=$scene_info&spbill_create_ip=$spbill_create_ip&total_fee=$total_fee&trade_type=$trade_type"; $strSignTmp = $signA."&key=$key"; //拼接字符串 注意顺序微信有个测试网址 顺序按照他的来 直接点下面的校正测试 包括下面 XML 是否正确 $sign = strtoupper(MD5($strSignTmp)); // MD5 后转换成大写 $post_data = " $appid $mch_id $body $out_trade_no $total_fee $spbill_create_ip $notify_url $trade_type $scene_info $out_trade_no $nonce_str $sign ";//拼接成 XML 格式 $url = "https://api.mch.weixin.qq.com/pay/unifiedorder";//微信传参地址 $dataxml = $this->postXmlCurl($post_data,$url); //后台 POST 微信传参地址 同时取得微信返回的参数 $objectxml = (array)simplexml_load_string($dataxml, 'SimpleXMLElement', LIBXML_NOCDATA); //将微信返回的 XML 转换成数组 return $objectxml; /* // 根据订单得到json格式的微信支付参数。 $trade_info = $this->getStartBizContentAndSetPaymentMethod(); if (!$trade_info) { Yii::$service->helper->errors->add('generate wxpay bizContent error'); return false; } $client_ip = Yii::$service->helper->getCustomerIp(); //①、获取用户openid $tools = new \JsApiPay(); if (!$code) { $openId = $tools->GetOpenid(); } else { $openId = $tools->GetOpenidByCode($code); } //echo $openId;exit; $wap_url = Yii::$service->url->homeUrl(); $wap_name = '微信H5支付'; $scene_info ='{"h5_info":{"type":"Wap","wap_url":"'.$wap_url.'","wap_name":"'.wap_name.'"}}';//场景信息 必要参数 //②、统一下单 $input = new \WxPayUnifiedOrder(); //$notify_url = Yii::$service->url->getUrl("payment/wxpayjsapi/ipn"); ////获取支付配置中的返回ipn url $notify_url = Yii::$service->payment->getStandardIpnUrl(); //$notify = new \NativePay(); //$input = new \WxPayUnifiedOrder(); $input->SetBody($this->scanCodeBody); //$input->SetAttach("商店的额外的自定义数据"); $input->SetAttach($trade_info['subject']); $input->SetDevice_info($this->deviceInfo); // 设置设备号 if ($trade_info['coupon_code']) { $input->SetGoods_tag($trade_info['coupon_code']); //设置商品标记,代金券或立减优惠功能的参数 } $input->SetOut_trade_no($trade_info['increment_id']); // Fecshop 订单号 $orderTotal = $trade_info['total_amount'] * 100; //微信支付的单位为分,所以要乘以100 $input->SetTotal_fee($orderTotal); $input->SetTime_start(date("YmdHis")); $input->SetTime_expire($this->getShangHaiExpireTime($this->expireTime)); $input->SetNotify_url($notify_url); //通知地址 改成自己接口通知的接口,要有公网域名,测试时直接行动此接口会产生日志 $input->SetTrade_type($this->tradeType); $input->SetSpbill_create_ip($client_ip); $input->SetSceneInfo($scene_info); $input->SetProduct_id($trade_info['product_ids']); //此为二维码中包含的商品ID //$input->SetOpenid($openId); //var_dump($input); $result = \WxPayApi::wapUnifiedOrder($input); return $result; */ } //打印输出数组信息 function printf_info($data) { foreach($data as $key=>$value){ echo "$key : $value
"; } } public function getShangHaiExpireTime($expire_time) { $timezone_out = date_default_timezone_get(); date_default_timezone_set('Asia/Shanghai'); $r_time = date("YmdHis", time() + $expire_time); date_default_timezone_set($timezone_out); return $r_time; } public function scanCodeCheckTradeIsSuccess($out_trade_no) { $result = Yii::$service->payment->wxpay->queryOrderByOut($out_trade_no); if (is_array($result) && !empty($result)) { $trade_state = $result['trade_state']; //最终的交易状态,必须为SUCCESS才是交易成功 $return_code = $result['result_code']; $trade_type = $result['trade_type']; //获取交易方式,这里使用的是扫码支付native $out_trade_no = $result['out_trade_no']; $total_amount = $result['total_fee']; $seller_id = $result['mch_id']; $auth_app_id = $result['appid']; $trade_no = $result['transaction_id']; $checkOrderStatus = Yii::$service->payment->wxpay->checkOrder($trade_state, $return_code, $trade_type, $out_trade_no, $total_amount, $seller_id, $auth_app_id); if ($checkOrderStatus) { return $this->updateOrderInfo($out_trade_no, $trade_no); } } } /** * 通过微信接口查询交易信息 * @param unknown $out_trade_no */ public function queryOrderByOut($out_trade_no) { $input = new \WxPayOrderQuery(); $input->SetOut_trade_no($out_trade_no); $result = \WxPayApi::orderQuery($input); return $result; } /** * 把返回的支付参数方式改成数组以适应微信的api * 生成二维码图片会用到这个函数 */ protected function getStartBizContentAndSetPaymentMethod() { $currentOrderInfo = Yii::$service->order->getCurrentOrderInfo(); if (isset($currentOrderInfo['products']) && is_array($currentOrderInfo['products'])) { $subject_arr = []; foreach ($currentOrderInfo['products'] as $product) { $subject_arr[] = $product['name']; } if (!empty($subject_arr)) { $subject = implode(',', $subject_arr); // 字符串太长会出问题,这里将产品的name链接起来,在截图一下 if (strlen($subject) > $this->subjectMaxLength) { $subject = mb_substr($subject, 0, $this->subjectMaxLength); } //echo $subject; $increment_id = $currentOrderInfo['increment_id']; $base_grand_total = $currentOrderInfo['base_grand_total']; $total_amount = Yii::$service->page->currency->getCurrencyPrice($base_grand_total, 'CNY'); Yii::$service->payment->setPaymentMethod($currentOrderInfo['payment_method']); $products = $currentOrderInfo['products']; $productIds = ''; if (is_array($products)) { foreach ($products as $product) { $productIds = $product['product_id']; break; } } return [ 'increment_id' => $increment_id, 'total_amount' => $total_amount, 'subject' => $subject, 'coupon_code' => $currentOrderInfo['coupon_code'], 'product_ids' => $productIds, ]; } } } /** * 检查订单是否合法 * 如果每项验证都通过则返回真 */ public function checkOrder($trade_state, $return_code, $trade_type, $out_trade_no, $total_amount, $seller_id, $auth_app_id) { if ($trade_state != 'SUCCESS') { Yii::$service->helper->errors->add('request trade_state is not equle to SUCCESS'); return false; } if ($return_code != 'SUCCESS') { Yii::$service->helper->errors->add('request return_code is not equle to SUCCESS'); return false; } if ($trade_type != 'NATIVE') { Yii::$service->helper->errors->add('request trade_type is not equle to NATIVE'); return false; } if (!$this->_order) { $this->_order = Yii::$service->order->getByIncrementId($out_trade_no); Yii::$service->payment->setPaymentMethod($this->_order['payment_method']); } if (!$this->_order) { Yii::$service->helper->errors->add('order increment id:{out_trade_no} is not exist.', ['out_trade_no' => $out_trade_no]); return false; } $base_grand_total = $this->_order['base_grand_total']; $order_total_amount = Yii::$service->page->currency->getCurrencyPrice($base_grand_total, 'CNY'); if ((string)($order_total_amount * 100) != $total_amount) { //由于微信中是以分为单位所以必须乘以100,二维码页面也已经作了处理,单位都是分,$order_total_amount * 100要转为字符串再比较 Yii::$service->helper->errors->add('order increment id:{out_trade_no} , total_amount({total_amount}) is not equal to order_total_amount({order_total_amount})', ['out_trade_no'=>$out_trade_no , 'total_amount'=>$total_amount , 'order_total_amount'=>$order_total_amount ]); //return ['o' => $order_total_amount * 100, 't' => $total_amount]; //测试时便于观察订单金额和微信实际支付的金额,生产环境要注释掉 return false; } return true; } /** * 微信 支付成功后,对订单的状态进行修改 * 如果支付成功,则修改订单状态为支付成功状态。 * @param $out_trade_no | string , fecshop的订单编号 increment_id * @param $trade_no | 微信支付交易号 * @param isClearCart | boolean 是否清空购物车 * */ protected function updateOrderInfo($out_trade_no, $trade_no, $isClearCart=true) { if (!empty($out_trade_no) && !empty($trade_no)) { if ($this->paymentSuccess($out_trade_no, $trade_no)) { // 清空购物车 if ($isClearCart) { Yii::$service->cart->clearCartProductAndCoupon(); } return true; } } else { Yii::$service->helper->errors->add('wxpay payment fail,resultCode: {result_code}', ['result_code' => $resultCode]); return false; } } /** * @param $increment_id | String 订单号 * @param $sendEmail | boolean 是否发送邮件 * 订单支付成功后,需要更改订单支付状态等一系列的处理。 */ protected function paymentSuccess($increment_id, $trade_no, $sendEmail = true) { if (!$this->_order) { $this->_order = Yii::$service->order->getByIncrementId($increment_id); Yii::$service->payment->setPaymentMethod($this->_order['payment_method']); } // 【优化后的代码 ##】 $orderstatus = Yii::$service->order->payment_status_confirmed; $updateArr['order_status'] = $orderstatus; $updateArr['txn_id'] = $trade_no; // 微信的交易号 $updateColumn = $this->_order->updateAll( $updateArr, [ 'and', ['order_id' => $this->_order['order_id']], ['in','order_status',$this->_allowChangOrderStatus] ] ); if (!empty($updateColumn)) { // 发送邮件,以及其他的一些操作(订单支付成功后的操作) Yii::$service->order->orderPaymentCompleteEvent($this->_order['increment_id']); } // 【优化后的代码 ##】 /* 注释掉的原来代码,上面进行了优化,保证更改只有一次,这样发邮件也就只有一次了 // 如果订单状态已经是processing,那么,不需要更改订单状态了。 if ($this->_order['order_status'] == Yii::$service->order->payment_status_confirmed){ return true; } $order = $this->_order; if (isset($order['increment_id']) && $order['increment_id']) { // 如果支付成功,则更改订单状态为支付成功 $order->order_status = Yii::$service->order->payment_status_confirmed; $order->txn_id = $trade_no; // 微信的交易号 // 更新订单信息 $order->save(); Yii::$service->order->orderPaymentCompleteEvent($order['increment_id']); // 得到当前的订单信息 // $orderInfo = Yii::$service->order->getOrderInfoByIncrementId($order['increment_id']); // 发送新订单邮件 // Yii::$service->email->order->sendCreateEmail($orderInfo); return true; } */ return true; } // 支付宝的 标示 public function getWxpayHandle() { return 'wxpay_standard'; } }