Maison >développement back-end >tutoriel php >Explication détaillée de l'interface d'arrière-plan php d'App WeChat Payment

Explication détaillée de l'interface d'arrière-plan php d'App WeChat Payment

小云云
小云云original
2018-03-08 10:34:472971parcourir

Cet article vous présentera l'interface de paiement et de rappel en arrière-plan PHP (7.0) de l'App WeChat Pay (2016.10.11). Le framework est Thinkphp5.0 : j'espère qu'il pourra vous aider.

  • Divers paramètres du compte

  • Informations de commande

  • Demande prepay_id

  • * Retour au traitement des données APP

  • Rappel WeChat

  • Modifier le statut de la commande


Divers paramètres du compte

Les différents paramètres du compte sont comme lorsque WeChat demande le paiement de l'application, un email sera envoyé à la boîte mail de votre compte, qui aura le Allocation de paiement WeChat correspondante. Le numéro du commerçant (MCHID), APPID et APPSECRET sont renvoyés lors de la demande d'autorisation de paiement pour l'application, et la CLÉ doit être définie par l'utilisateur dans le backend du commerçant sur WeChat. C'est très important ! 🎜>

Informations sur la commande

  • ** Le client transférera les données du produit dans le panier vers le backend, y compris les informations sur l'utilisateur, etc. Après avoir obtenu les données, vous devez d'abord vérifier les données, la vérification ici détermine généralement les règles de transmission avec le client au début du projet, telles que la méthode de transmission, le nom du paramètre et la méthode de vérification des paramètres. L'accent est ici mis sur la vérification, et l'authentification par signature est. généralement utilisé :

  • //Etape 1 de signature : Trier les paramètres par ordre lexicographique

  • Voici un exemple de code (formatage des paramètres en paramètres d'url ; ):

/**
     * 格式化参数格式化成url参数
     */
    public function ToUrlParams()
    {
        $buff = "";
        foreach ($this->values as $k => $v)
        {            if($k != "sign" && $v != "" && !is_array($v)){
                $buff .= $k . "=" . $v . "&";
            }
        }

        $buff = trim($buff, "&");        return $buff;
    }
  • //Signature étape 2 : Ajouter une CLÉ après la chaîne (cette CLÉ est déterminée par négociation avec le personnel front-end) ;


  • //Troisième étape de la signature : cryptage MD5 ;


  • //Signature Étape 4 : Convertir tous les caractères en majuscules


Il est préférable de les encapsuler dès le début du projet, et ils seront inclus plus tard. Appelez-le simplement, par exemple :

(Vous n'en aurez besoin que plus tard, $param = $this->request('parameter name '));, puis pré-stocker les informations de commande.

request prepay_id

Je vais accéder directement au code ici, (l'adresse de téléchargement du document d'interface est donnée ici :), bien que il existe de nombreuses interfaces de requête auto-écrites sur Internet, mais puisque WeChat a été packagé, utilisez-le simplement :

$input = new \app\wxpay\WxPayUnifiedOrder();//这里引用微信的统一下单接口

$input->SetBody($data['gname']['g_name']);//商品或支付单简要描述
$input->SetAttach($data['gname']['g_name']);//置附加数据 
$input->SetOut_trade_no($order_sn); // 商户订单号
$input->SetTotal_fee(intval($data['data']['order_price']*100)); 
$input->SetTime_start(date("YmdHis"));//订单生成时间
$input->SetTime_expire(date("YmdHis", time() + 600));//订单失效时间
$input->SetGoods_tag($data['gname']['g_name']); //商品标记  
$input->SetNotify_url("http://www.weixin.qq.com/wxpay/notify.php"); // 支付成功后的回调地址,
$input->SetTrade_type("APP");
$order = \app\wxpay\WxPayApi::unifiedOrder($input);return $order['prepay_id'];
Voici l'adresse officielle de description de l'interface de commande unifiée de WeChat :

https:// pay.weixin.qq.com/wiki/doc/api/app/app.php?chapter=9_1

Retourner les informations du client

$info = array();
//账号的信息一般都放在配置文件里面,用到的地方也很多
$info['appid'] = config('APP_APPID');
$info['partnerid'] = config('APP_MCHID');
$info['package'] = config('APP_PACKAGE');
$info['noncestr'] = $this->random_number();//生成随机数,下面有生成实例,统一下单接口需要
$info['timestamp'] = time();
$info['prepayid'] = $prepay_id;
$info['sign'] = self::_makeSign($info);//生成签名return $info;
$info est l'information dont le client a besoin


🎜>Générer une instance de nombre aléatoire
//生成随机数
    public function random_number($len=21,$format='ALL' ){
        $is_abc = $is_numer = 0;
        $password = $tmp ='';
        switch($format){
            case 'ALL':
            $chars='ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';            break;
            case 'CHAR':
            $chars='ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz';            break;
            case 'NUMBER':
            $chars='0123456789';            break;
            default :
            $chars='ABCDEFGHIJKLMNOPQRSTUVWXYZabcdefghijklmnopqrstuvwxyz0123456789';            break;
        } // www.jb51.net
        mt_srand((double)microtime()*1000000*getmypid());        while(strlen($password)<$len){
            $tmp =substr($chars,(mt_rand()%strlen($chars)),1);            if(($is_numer <> 1 && is_numeric($tmp) && $tmp >0 )|| $format == &#39;CHAR&#39;){
                $is_numer = 1;
            }            if(($is_abc <> 1 && preg_match(&#39;/[a-zA-Z]/&#39;,$tmp)) || $format == &#39;NUMBER&#39;){
                $is_abc = 1;
            }
            $password.= $tmp;
        }        if($is_numer <> 1 || $is_abc <> 1 || empty($password) ){
            $password = $this->random_number($len,$format);
        }        return $password;
        }

Rappel WeChat


Notification du résultat du paiement notify.php (l'adresse ici est l'adresse de rappel renseignée lors de la commande, WeChat l'a déjà emballée) , adresse de téléchargement du document
http://mch.weixin.qq.com/wiki/doc/api/ jsapi.php?chapter=11_1

En fait, le code principal de cette page n'est que de deux lignes

[php] afficher la copie simple
    $notify = new PayNotifyCallBack();  
    $notify->Handle(false);

La plupart de la logique est traitée dans le fichier de fonction Handle WxPay.Notify.php

[php] afficher la copie simple
    final public function Handle($needSign = true)  
        {  
            $msg = "OK";  
            //当返回false的时候,表示notify中调用NotifyCallBack回调失败获取签名校验失败,此时直接回复失败  
            $result = WxpayApi::notify(array($this, &#39;NotifyCallBack&#39;), $msg);  
            if($result == false){  
                $this->SetReturn_code("FAIL");  
                $this->SetReturn_msg($msg);  
                $this->ReplyNotify(false);  
                return;  
            } else {  
                //该分支在成功回调到NotifyCallBack方法,处理完成之后流程  
                $this->SetReturn_code("SUCCESS");  
                $this->SetReturn_msg("OK");  
            }  
            $this->ReplyNotify($needSign);  
        }


Code principal :

[php] afficher la copie simple
    $result = WxpayApi::notify(array($this, 'NotifyCallBack'), $msg);

Fonction de suivi notifier le fichier WxPay.Api.php

[php] afficher la copie simple
    public static function notify($callback, &$msg)  
        {  
            //获取通知的数据  
            $xml = $GLOBALS[&#39;HTTP_RAW_POST_DATA&#39;];  
            //如果返回成功则验证签名  
            try {  
                $result = WxPayResults::Init($xml);  
            } catch (WxPayException $e){  
                $msg = $e->errorMessage();  
                return false;  
            }  

            return call_user_func($callback, $result);  
        }


Récupérez les données gay via $GLOBALS['HTTP_RAW_POST_DATA'], puis utilisez la fonction Init pour vérifier la signature, etc. La vérification de la signature est réussie et le code est exécuté.
Il faut expliquer ici que php7 lui-même ne prend pas en charge $GLOBALS['HTTP_RAW_POST_DATA']. Vous devez télécharger un plug-in. Vous pouvez consulter Baidu pour plus de détails. Ce que je veux dire, c'est que vous pouvez utiliser file_get_contents('php:/ /input'), pour des raisons spécifiques, veuillez vous référer au blog ci-dessous, qui est très détaillé (https://my.oschina.net/jiec/blog /485359)

[php] view plain copy
return call_user_func($callback, $result);

C'est-à-dire qu'une fonction de rappel est appelée, la fonction NotifyCallBack() et le paramètre $result est passé dans la fonction NotifyCallBack, notre réécriture. La fonction NotifyProcess() sera appelée (cette fonction est réécrite dans notify.php)

NotifyProcess() détermine qu'il n'y a pas de problème et définira les informations XML renvoyées sur succès

[php] afficher la copie simple
$this->SetReturn_code("SUCCESS");  
$this->SetReturn_msg("OK");

et enfin appeler la fonction this−>ReplyNotify(this−> ReplyNotify(needSign); echo success result

La fonction ReplyNotify doit modifier un code :

[php] voir la copie simple
final private function ReplyNotify($needSign = true)  
    {  
        //如果需要签名  
        if($needSign == true &&   
            $this->GetReturn_code($return_code) == "SUCCESS")  
        {  
            $this->SetSign();  
        }  
        WxpayApi::replyNotify($this->ToXml());  
    }

[php] voir la copie simple
$this->GetReturn_code($return_code) == "SUCCESS")

est remplacé par

[php] voir la copie simple copier
$this->GetReturn_code() == "SUCCESS")


Puis modifier le statut de la commande en fonction des informations renvoyées, principalement, Où modifier, j'ai créé une nouvelle méthode
//修改订单状态
    public function updateState($data){        if($data){
            $order_sn = $data[&#39;out_trade_no&#39;];\

            $data = array();
            $data[&#39;order_id&#39;] = $order_id;
            //修改订单状态(用curlpost方法请求至thinkphp目录下的Controller里面控制器里面的方法,修改状态)
            $url = &#39;www.test.com&#39;;
            header(&#39;content-type:text/html;charset=utf8&#39;);
            $curl = curl_init();
            curl_setopt($curl, CURLOPT_URL, $url);
            curl_setopt($curl, CURLOPT_RETURNTRANSFER, true);
            curl_setopt($curl, CURLOPT_POST, true);
            curl_setopt($curl, CURLOPT_POSTFIELDS, $data);
            $result = curl_exec($curl);
            curl_close($curl);            if($result == &#39;success&#39;){                return true;
            }else{                return false;
            }

        }
    }

dans notify.php puis ajouté
$notify = new PayNotifyCallBack();
$notify->Handle(false);
sous

dans notify.php
//接受参数,修改状态
$xml = file_get_contents("php://input");
$data = json_decode(json_encode(simplexml_load_string($xml, &#39;SimpleXMLElement&#39;, LIBXML_NOCDATA)), true);
$notify->updateState($data);

相关推荐:

PHP后台接口的一些问题

5 分钟实现微信支付接入教程

php初学者如何学习实现微信支付和支付宝支付的相关教程

Ce qui précède est le contenu détaillé de. pour plus d'informations, suivez d'autres articles connexes sur le site Web de PHP en chinois!

Déclaration:
Le contenu de cet article est volontairement contribué par les internautes et les droits d'auteur appartiennent à l'auteur original. Ce site n'assume aucune responsabilité légale correspondante. Si vous trouvez un contenu suspecté de plagiat ou de contrefaçon, veuillez contacter admin@php.cn