| 123456789101112131415161718192021222324252627282930313233343536373839404142434445464748495051525354555657585960616263646566676869707172737475767778798081828384858687888990919293949596979899100101102103104105106107108109110111112113114115116117118119120121122123124125126127128129130131132133134135136137138139140141142143144145146147148149150151152153154155156157158159160161162163164165166167168169170171172173174175176177178179180181182183184185186187188189190191192193194195196197198199200201202203204205206207208209210211212213214215216217218219220221222223224225226227228229230231232233234235236237238239240241242243244245246247248249250251252253254255256257258259260261262263264265266267268269270271272273274275276277278279280281282283284285286287288289290291292293294295296297298299300301302303304305306307308309310311312313314315316317318319320321322323324325326327328329330331332333334335336337338339340341342343344345346347348349350351352353354355356357358359360361362363364365366367368369370371372373374375376377378379380381382383384385386387388389390391392393394395396397398399400401402403404405406407408409410411412413414415416417418419420421422423424425426427428429 |
- <?php
- /*
- * FecShop file.
- *
- * @link http://www.fecshop.com/
- * @copyright Copyright (c) 2016 FecShop Software LLC
- * @license http://www.fecshop.com/license/
- */
- namespace fecshop\services\payment;
- //use fecshop\models\mysqldb\IpnMessage;
- use fecshop\services\Service;
- use Yii;
- /**
- * Payment Paypal services.
- * @author Terry Zhao <2358269014@qq.com>
- * @since 1.0
- */
- class Alipay extends Service
- {
- public $gatewayUrl;
- // 商家 appid
- public $appId;
- // 商家uid
- public $sellerId;
- // 应用私钥
- public $rsaPrivateKey;
- // 支付宝公钥
- public $alipayrsaPublicKey;
- public $format;
- public $charset;
- public $signType;
-
- public $devide;
- public $apiVersion = '1.0'; //'1.0';
-
- //protected $_returnUrl;
- //protected $_notifyUrl;
- protected $_AopClient;
- protected $_alipayRequest;
- protected $_productCode;
- protected $_order;
- //交易创建,等待买家付款
- const WAIT_BUYER_PAY = 'WAIT_BUYER_PAY';
- //未付款交易超时关闭,或支付完成后全额退款
- const TRADE_CLOSED = 'TRADE_CLOSED';
- //交易支付成功
- const TRADE_SUCCESS = 'TRADE_SUCCESS';
- //交易结束,不可退款
- const TRADE_FINISHED = 'TRADE_FINISHED';
-
- protected $_ipnMessageModelName = '\fecshop\models\mysqldb\IpnMessage';
- protected $_ipnMessageModel;
-
- // 允许更改的订单状态,不存在这里面的订单状态不允许修改
- protected $_allowChangOrderStatus;
- protected $_initAlipayLib = 0;
-
- /**
- * 支付宝:SDK工作目录
- * 存放日志,AOP缓存数据
- */
- public $alipay_aop_sdk_work_dir = '/tmp/';
- /**
- * 是否处于开发模式
- * 在你自己电脑上开发程序的时候千万不要设为false,以免缓存造成你的代码修改了不生效
- * 部署到生产环境正式运营后,如果性能压力大,可以把此常量设定为false,能提高运行速度(对应的代价就是你下次升级程序时要清一下缓存)
- */
- public $alipay_aop_sdk_dev_mode = true;
-
- public function init()
- {
- parent::init();
- list($this->_ipnMessageModelName, $this->_ipnMessageModel) = \Yii::mapGet($this->_ipnMessageModelName);
- $this->_allowChangOrderStatus = [
- Yii::$service->order->payment_status_pending,
- Yii::$service->order->payment_status_processing,
- ];
- // init by store config
- $this->appId = Yii::$app->store->get('payment_alipay', 'app_id');
- $this->sellerId = Yii::$app->store->get('payment_alipay', 'seller_id');
- $this->rsaPrivateKey = Yii::$app->store->get('payment_alipay', 'rsa_private_key');
- $this->alipayrsaPublicKey = Yii::$app->store->get('payment_alipay', 'rsa_public_key');
- if ($alipay_aop_sdk_work_dir = Yii::$app->store->get('payment_alipay', 'alipay_aop_sdk_work_dir')) {
- $this->alipay_aop_sdk_work_dir = $alipay_aop_sdk_work_dir;
- }
- $this->alipay_aop_sdk_dev_mode = Yii::$app->store->get('payment_alipay', 'alipay_aop_sdk_dev_mode') == 1 ? true : false ;
- // 沙盒还是正式环境
- $env = Yii::$app->store->get('payment_alipay', 'alipay_env');
- if ($env == Yii::$service->payment->env_sanbox) {
- $this->gatewayUrl = 'https://openapi.alipaydev.com/gateway.do';
- } else {
- $this->gatewayUrl = 'https://openapi.alipay.com/gateway.do';
- }
- }
-
-
- /**
- * 初始化 $this->_AopClient
- */
- protected function initParam()
- {
- /**
- * 引入 支付宝支付的SDK文件。
- */
- if (!$this->_initAlipayLib) {
- define("AOP_SDK_WORK_DIR", $this->alipay_aop_sdk_work_dir);
- define("AOP_SDK_DEV_MODE", $this->alipay_aop_sdk_dev_mode);
- $AopSdkFile = Yii::getAlias('@fecshop/lib/alipay/AopSdk.php');
- require($AopSdkFile);
- $this->_initAlipayLib = 1;
- }
- if (!$this->_AopClient) {
- $this->_AopClient = new \AopClient;
- $this->_AopClient->gatewayUrl = $this->gatewayUrl;
- $this->_AopClient->appId = $this->appId;
- $this->_AopClient->rsaPrivateKey = $this->rsaPrivateKey;
- $this->_AopClient->apiVersion = $this->apiVersion; //'1.0';
- $this->_AopClient->format = $this->format;
- $this->_AopClient->charset = $this->charset;
- $this->_AopClient->signType = $this->signType;
- $this->_AopClient->alipayrsaPublicKey= $this->alipayrsaPublicKey;
- }
- }
- /**
- * @param $out_trade_no | String ,[支付宝传递过来的]fecshop站内订单号
- * @param $total_amount | String ,[支付宝传递过来的]fecshop站内订单金额(CNY)
- * @param $seller_id | String ,[支付宝传递过来的]商家UID
- * @param $auth_app_id | String ,[支付宝传递过来的]商家appId
- * 验证订单数据是否正确,需要满足下面的条件:
- * 1、商户需要验证该通知数据中的out_trade_no是否为商户系统中创建的订单号
- * 2、判断total_amount是否确实为该订单的实际金额(即商户订单创建时的金额)
- * 3、校验通知中的seller_id(或者seller_email) 是否为out_trade_no这笔单据的对应的操作方(有的时候,一个商户可能有多个seller_id/seller_email)
- * 4、验证app_id是否为该商户本身。
- * 上述1、2、3、4有任何一个验证不通过,则表明本次通知是异常通知,
- * 务必忽略。在上述验证通过后商户必须根据支付宝不同类型的业务通知,
- * 正确的进行不同的业务处理,并且过滤重复的通知结果数据。
- * 在支付宝的业务通知中,只有交易通知状态为TRADE_SUCCESS或TRADE_FINISHED时,
- * 支付宝才会认定为买家付款成功。
- */
- protected function validateReviewOrder($out_trade_no, $total_amount, $seller_id, $auth_app_id)
- {
- 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');
- $order_total_amount = $this->_order['grand_total'];
- if ($order_total_amount != $total_amount) {
- 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 false;
- }
- if (!$this->sellerId) {
- Yii::$service->helper->errors->add('you must config sellerId in alipay payment config file');
-
- return false;
- }
- if ($seller_id != $this->sellerId) {
- Yii::$service->helper->errors->add('request sellerId({seller_id}) is not equle to config sellerId({this_seller_id})', ['seller_id'=>$seller_id , 'this_seller_id'=>$this->sellerId ]);
-
- return false;
- }
- if ($auth_app_id != $this->appId) {
- Yii::$service->helper->errors->add('request auth_app_id({auth_app_id}) is not equle to config appId({app_id})', ['auth_app_id'=>$auth_app_id, 'app_id'=>$this->appId ]);
-
- return false;
- }
-
- return true;
- }
- /**
- * 支付宝 支付成功后,返回网站,调用该函数进行支付宝订单支付状态查询
- * 如果支付成功,则修改订单状态为支付成功状态。
- */
- protected function actionReview()
- {
- $this->initParam();
- $trade_no = Yii::$app->request->get('trade_no');
- $out_trade_no = Yii::$app->request->get('out_trade_no');
- $total_amount = Yii::$app->request->get('total_amount');
- $seller_id = Yii::$app->request->get('seller_id');
- $auth_app_id = Yii::$app->request->get('auth_app_id');
- //验证订单的合法性
- if (!$this->validateReviewOrder($out_trade_no, $total_amount, $seller_id, $auth_app_id)) {
- return false;
- }
- $this->_AopClient->postCharset = $this->charset;
- $this->_alipayRequest = new \AlipayTradeQueryRequest();
- $bizContent = json_encode([
- 'out_trade_no' => $out_trade_no,
- 'trade_no' => $trade_no,
- ]);
- //echo $bizContent;
- $this->_alipayRequest->setBizContent($bizContent);
- $result = $this->_AopClient->execute($this->_alipayRequest);
- $responseNode = str_replace(".", "_", $this->_alipayRequest->getApiMethodName()) . "_response";
- $resultCode = $result->$responseNode->code;
- if (!empty($resultCode)&&$resultCode == 10000) {
- $this->paymentSuccess($out_trade_no, $trade_no);
- // 清空购物车
- Yii::$service->cart->clearCartProductAndCoupon();
-
- return true;
- } else {
- Yii::$service->helper->errors->add('Alipay payment fail,resultCode: {result_code}', ['result_code' => $resultCode]);
-
- return false;
- }
- }
- /**
- * 支付宝的消息接收IPN,执行的函数,接收的消息用来更改订单状态。
- * 您开启log后,可以在@app/runtime/fecshop_logs
- * 文件夹下执行:tail -f fecshop_debug.log , 来查看log输出。
- */
- public function receiveIpn()
- {
- Yii::info('alipay service receiveIpn():begin init param', 'fecshop_debug');
- $this->initParam();
- Yii::info('alipay service receiveIpn():begin rsaCheck', 'fecshop_debug');
- // 验签
- $checkV2Status = $this->_AopClient->rsaCheckV1($_POST, '', $this->signType);
- Yii::info('alipay service receiveIpn():rsacheck end', 'fecshop_debug');
- if ($checkV2Status) {
- Yii::info('alipay service receiveIpn():rsacheck success', 'fecshop_debug');
- $trade_no = Yii::$app->request->post('trade_no');
- $out_trade_no = Yii::$app->request->post('out_trade_no');
- $total_amount = Yii::$app->request->post('total_amount');
- $seller_id = Yii::$app->request->post('seller_id');
- $auth_app_id = Yii::$app->request->post('app_id');
- $trade_status = Yii::$app->request->post('trade_status');
- Yii::info('alipay service receiveIpn(): [ trade_no: ]'.$trade_no, 'fecshop_debug');
- Yii::info('alipay service receiveIpn(): [ out_trade_no: ]'.$out_trade_no, 'fecshop_debug');
- Yii::info('alipay service receiveIpn(): [ total_amount: ]'.$total_amount, 'fecshop_debug');
- Yii::info('alipay service receiveIpn(): [ seller_id: ]'.$seller_id, 'fecshop_debug');
- Yii::info('alipay service receiveIpn(): [ auth_app_id: ]'.$auth_app_id, 'fecshop_debug');
- Yii::info('alipay service receiveIpn(): [ trade_status: ]'.$trade_status, 'fecshop_debug');
-
- //验证订单的合法性
- if (!$this->validateReviewOrder($out_trade_no, $total_amount, $seller_id, $auth_app_id)) {
- Yii::info('alipay service receiveIpn(): validate order fail', 'fecshop_debug');
-
- return false;
- }
- Yii::info('alipay service receiveIpn():validate order success', 'fecshop_debug');
- if (self::TRADE_SUCCESS == $trade_status) {
- Yii::info('alipay service receiveIpn():alipay trade success ', 'fecshop_debug');
- if ($this->paymentSuccess($out_trade_no, $trade_no)) {
- Yii::info('alipay service receiveIpn():update order status success', 'fecshop_debug');
-
- return true;
- }
- }
- } else {
- return false;
- }
- }
- /**
- * @param $increment_id | String 订单号
- * @param $sendEmail | boolean 是否发送邮件
- * 订单支付成功后,需要更改订单支付状态等一系列的处理。
- */
- protected function paymentSuccess($increment_id, $trade_no, $sendEmail = true)
- {
- Yii::$service->store->currentLangCode = 'zh';
- 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;
- }
-
- /**
- * 根据订单,将内容提交给支付宝。跳转到支付宝支付页面。
- * 在下单页面点击place order按钮,跳转到支付宝的时候,执行该函数。
- */
- public function start($returnUrl = '', $type="POST")
- {
- // 初始化参数
- $this->initParam();
- // 根据wap 还是pc ,进行参数初始化
- if ($this->devide == 'wap') {
- $this->_alipayRequest = new \AlipayTradeWapPayRequest();
- $this->_productCode = 'QUICK_WAP_WAY';
- } elseif ($this->devide == 'pc') {
- $this->_productCode = 'FAST_INSTANT_TRADE_PAY';
- $this->_alipayRequest = new \AlipayTradePagePayRequest();
- } else {
- Yii::$service->helper->errors->add('you must config param [devide] in payment alipay service');
- return;
- }
-
- // 根据订单得到json格式的支付宝支付参数。
- $bizContent = $this->getStartBizContentAndSetPaymentMethod();
- if (!$bizContent) {
- Yii::$service->helper->errors->add('generate alipay bizContent error');
- }
- // 设置支付成功返回的url 和 支付消息接收url
- // 在调用这个函数之前一定要先设置 Yii::$service->payment->setPaymentMethod($payment_method);
- if (!$returnUrl) {
- $returnUrl = Yii::$service->payment->getStandardReturnUrl();
- }
- $notifyUrl = Yii::$service->payment->getStandardIpnUrl();
- /*
- echo $returnUrl;
- echo '#';
- echo $notifyUrl;
- echo '#';
- echo $bizContent;
- exit;
- */
- $this->_alipayRequest->setReturnUrl($returnUrl);
- $this->_alipayRequest->setNotifyUrl($notifyUrl);
- $this->_alipayRequest->setBizContent($bizContent);
- return $this->_AopClient->pageExecute($this->_alipayRequest, $type);
- }
- /**
- * 通过订单信息,得到支付宝支付传递的参数数据
- * 也就是一个json格式的数组。
- */
- 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);
- $increment_id = $currentOrderInfo['increment_id'];
- //$base_grand_total = $currentOrderInfo['base_grand_total'];
- //$total_amount = Yii::$service->page->currency->getCurrencyPrice($base_grand_total,'CNY');
- $total_amount = $currentOrderInfo['grand_total'];
- Yii::$service->payment->setPaymentMethod($currentOrderInfo['payment_method']);
- return json_encode([
- // param 参看:https://docs.open.alipay.com/common/105901
- 'out_trade_no' => $increment_id,
- 'product_code' => $this->_productCode,
- 'total_amount' => $total_amount,
- 'subject' => $subject,
- //'body' => '',
- ]);
- }
- }
- }
-
- // 支付宝的 标示
- public function getAlipayHandle()
- {
- return 'alipay_standard';
- }
- }
|