微信小程序实现支付功能

小程序支付,没有封装支付代码:直接上一段可用的流程代码吧:
微信小程序支付官网文档有详细的说明,这里我就不再赘述啦:
客户端js:

wx.request({
url:‘https://www.xxxx.com/api/payfee.php‘,//改成你自己的链接
header:{
‘Content-Type‘:‘application/x-www-form-urlencoded‘
},
method:‘POST‘,
success:function(res){
console.log(res.data);
console.log(‘调起支付‘);
wx.requestPayment({
‘timeStamp‘: res.data.timeStamp,
‘nonceStr‘: res.data.nonceStr,
‘package‘: res.data.package,
‘signType‘:‘MD5‘,
‘paySign‘: res.data.paySign,
‘success‘:function(res){
console.log(‘success‘);
wx.showToast({
title:‘支付成功‘,
icon:‘success‘,
duration:3000
});
},
‘fail‘:function(res){
console.log(‘fail‘);
},
‘complete‘:function(res){
console.log(‘complete‘);
}
});
},
fail:function(res){
console.log(res.data)
}
});

后端部分:laravel5:

<?php

namespace App\Http\Controllers\Api\Order;
use App\Models\Order;
use App\Models\OrderGoods;
use Illuminate\Http\Request;
use App\Http\Controllers\Controller;
use App\Models\Member;
use App\Models\MemberAddress;
use App\Models\Product;
use App\Http\Requests;
use App\Http\Requests\Interfaces\MemberCheck;
use Carbon\Carbon;
class OrderController extends Controller
{
    use MemberCheck;
    public function __construct()
    {
        $config = array(
            ‘appid‘         => env(‘WECHAT_APPID‘),//小程序appid
            ‘pay_mchid‘     => env(‘WECH_ID‘),//商户号
            ‘pay_apikey‘ =>env(‘WECHAT_KEY‘),//可在微信商户后台生成支付秘钥
        );
        $this->config = $config;
    }

    /**
     * 生成订单号
     * @author      lxhui<[email protected]>
     * @since 1.0
     * @return array
     */
    private static function trade_no() {
        list($usec, $sec) = explode(" ", microtime());
        $usec = substr(str_replace(‘0.‘, ‘‘, $usec), 0 ,4);
        $str  = rand(10,99);
        return date("YmdHis").$usec.$str;
    }

    /**
     * order
     * @author      lxhui<[email protected]>
     * @since 1.0
     * @return array
     */
    public function store(Request $request )
    {
        \DB::beginTransaction();
        try{
            if( !$this->checkMember([‘openid‘=>$request->openid]))
                return response()->json([‘code‘=>200,‘status‘=>0,‘message‘=>‘该openid未注册‘]);

            $config = $this->config;
            $result =json_decode($request->apiParams,true);
            if( !$result[‘addressInfo‘])
                return response()->json([‘code‘=>200,‘status‘=>0,‘message‘=>‘收货地址不能为空‘]);
            if( !$result[‘orderInfo‘])
                return response()->json([‘code‘=>200,‘status‘=>0,‘message‘=>‘订单商品不能为空‘]);

            $address_data = [
                ‘openid‘=>$request->openid,
                ‘realname‘=>$result[‘addressInfo‘][‘userName‘],
                ‘mobile‘=>$result[‘addressInfo‘][‘telNumber‘],
                ‘province‘=>$result[‘addressInfo‘][‘provinceName‘],
                ‘city‘=>$result[‘addressInfo‘][‘cityName‘],
                ‘area‘=>$result[‘addressInfo‘][‘countyName‘],
                ‘address‘=>$result[‘addressInfo‘][‘detailInfo‘],
                ‘zipcode‘=>$result[‘addressInfo‘][‘postalCode‘],
            ];
            $model = MemberAddress::firstOrCreate($address_data);
            $productArray =$result[‘orderInfo‘];
            $products = [];
            $orderProducts = [];
            $productsFee = 0.0; //支付商品总价
            foreach ($productArray as $val) {
                $product = Product::find($val[‘goodsid‘]);
                $productsFee += $product->marketprice * $val[‘total‘];
            }
            // 计算价格
            $shippingFee = 10.0;
            $totalFee = $productsFee + $shippingFee;
            // 创建订单
            $ordersn = self::trade_no();
            $order = new Order();
            $order->ordersn = $ordersn;
            $order->openid = $request->openid;
            $order->price = $totalFee;
            $order->goodsprice = $totalFee;
            $order->createtime =time();
            $order->addressid = $model->id;
            $order->storeid = $request->storeid;
            $order->save();
            $orderid = $order->id;

            $order_goods = new OrderGoods();
            foreach ($productArray as $val) {
                $product = Product::find($val[‘goodsid‘]);
                $product=[‘orderid‘=>$orderid,‘goodsid‘=>$val[‘goodsid‘],‘price‘=>$product->marketprice,‘total‘=>$val[‘total‘],‘openid‘=>$request->openid];
                array_push($products, $product);
            }
            /* 生产预订单参数 */
            $openid = $request->openid;
            $body = ‘城里网商城订单‘;
            $order_sn = $ordersn;
            $total_fee = $totalFee;

            //统一下单参数构造
            $unifiedorder = array(
                ‘appid‘            => $config[‘appid‘],
                ‘mch_id‘        => $config[‘pay_mchid‘],
                ‘nonce_str‘        => self::getNonceStr(),
                ‘body‘            => $body,
                ‘out_trade_no‘    => $order_sn,
                ‘total_fee‘        => $total_fee * 100,
                ‘spbill_create_ip‘    => $request->getClientIp(),
                ‘notify_url‘    => ‘https://‘.$_SERVER[‘HTTP_HOST‘].‘/Api/Wxpay/notify‘,
                ‘trade_type‘    => ‘JSAPI‘,
                ‘openid‘        => $request->openid,//‘oIXoL0ZpfG3NdSE8Qa-S1GcEHJGY‘//测试openid
            );
            $unifiedorder[‘sign‘] = self::makeSign($unifiedorder);
            //请求数据
            $xmldata = self::array2xml($unifiedorder);
            $url = ‘https://api.mch.weixin.qq.com/pay/unifiedorder‘;
            $res = self::curl_post_ssl($url, $xmldata);
            if(!$res){
                return response()->json([‘code‘=>200,‘status‘=>0,‘message‘=>‘无法连接服务器‘]);
                //self::return_err("Can‘t connect the server");
            }
             //file_put_contents是用来查看服务器返回的结果 测试完可以删除了
             //file_put_contents(‘/Statics/log1.txt‘,$res,FILE_APPEND);

            $content = self::xml2array($res);
            $result_code= isset($content[‘result_code‘]) ? $content[‘result_code‘] : ‘‘;
            $return_code = isset($content[‘return_code‘]) ? $content[‘return_code‘] : ‘‘;
            if(strval($result_code) == ‘FAIL‘){
                return self::return_err(strval($content[‘err_code_des‘]));
            }
            if(strval($return_code) == ‘FAIL‘){
                return self::return_err(strval($content[‘return_msg‘]));
            }
            $data = $this->pay($content[‘prepay_id‘]);
            \DB::table("eshop_order_goods")->insert($products);
            \DB::commit();
            return response()->json([‘code‘=>200,‘status‘=>1,‘message‘=>‘提交成功‘,‘data‘=>$data]);
        } catch (\Exception $e){
            \DB::rollback();//事务回滚
            return response()->json([‘code‘=>200,‘status‘=>0,‘message‘=>$e->getMessage()]);
        }
    }

    /**
     * 进行支付接口签名
     * @param string $prepay_id 预支付ID(调用prepay()方法之后的返回数据中获取)
     * @return  json的数据
     */
    public function pay($prepay_id){
        $config = $this->config;
        $data = array(
            ‘appId‘        => $config[‘appid‘],
            ‘timeStamp‘    => time(),
            ‘nonceStr‘    => self::getNonceStr(),
            ‘package‘    => ‘prepay_id=‘.$prepay_id,
            ‘signType‘    => ‘MD5‘
        );

        $data[‘paySign‘] = self::makeSign($data);

        return $data;
    }

    //微信支付回调验证
    public function notify(){
        $xml = $GLOBALS[‘HTTP_RAW_POST_DATA‘];

        // 这句file_put_contents是用来查看服务器返回的XML数据 测试完可以删除了
        //file_put_contents(APP_ROOT.‘/Statics/log2.txt‘,$res,FILE_APPEND);

        //将服务器返回的XML数据转化为数组
        $data = self::xml2array($xml);
        // 保存微信服务器返回的签名sign
        $data_sign = $data[‘sign‘];
        // sign不参与签名算法
        unset($data[‘sign‘]);
        $sign = self::makeSign($data);

        // 判断签名是否正确  判断支付状态
        if ( ($sign===$data_sign) && ($data[‘return_code‘]==‘SUCCESS‘) && ($data[‘result_code‘]==‘SUCCESS‘) ) {
            $result = $data;
            //获取服务器返回的数据
            $order_sn = $data[‘out_trade_no‘];            //订单单号
            $openid = $data[‘openid‘];                    //付款人openID
            $total_fee = $data[‘total_fee‘];            //付款金额
            $transaction_id = $data[‘transaction_id‘];     //微信支付流水号

            //更新数据库
            $this->updateDB($order_sn,$openid,$total_fee,$transaction_id);

        }else{
            $result = false;
        }
        // 返回状态给微信服务器
        if ($result) {
            $str=‘<xml><return_code><![CDATA[SUCCESS]]></return_code><return_msg><![CDATA[OK]]></return_msg></xml>‘;
        }else{
            $str=‘<xml><return_code><![CDATA[FAIL]]></return_code><return_msg><![CDATA[签名失败]]></return_msg></xml>‘;
        }
        echo $str;
        return $result;
    }

//---------------------------------------------------------------用到的函数------------------------------------------------------------
    /**
     * 错误返回提示
     * @param string $errMsg 错误信息
     * @param string $status 错误码
     * @return  json的数据
     */
    protected function return_err($errMsg=‘error‘,$status=0){
        return response()->json([‘code‘=>200,‘result‘=>‘fail‘,‘status‘=>$status,‘errmsg‘=>$errMsg]);
    }

    /**
     * 正确返回
     * @param     array $data 要返回的数组
     * @return  json的数据
     */
    protected function return_data($data=array()){
        return response()->json([‘code‘=>200,‘result‘=>‘success‘,‘status‘=>1,‘data‘=>$data]);
    }

    /**
     * 将一个数组转换为 XML 结构的字符串
     * @param array $arr 要转换的数组
     * @param int $level 节点层级, 1 为 Root.
     * @return string XML 结构的字符串
     */
    protected function array2xml($arr, $level = 1) {
        $s = $level == 1 ? "<xml>" : ‘‘;
        foreach($arr as $tagname => $value) {
            if (is_numeric($tagname)) {
                $tagname = $value[‘TagName‘];
                unset($value[‘TagName‘]);
            }
            if(!is_array($value)) {
                $s .= "<{$tagname}>".(!is_numeric($value) ? ‘<![CDATA[‘ : ‘‘).$value.(!is_numeric($value) ? ‘]]>‘ : ‘‘)."</{$tagname}>";
            } else {
                $s .= "<{$tagname}>" . $this->array2xml($value, $level + 1)."</{$tagname}>";
            }
        }
        $s = preg_replace("/([\x01-\x08\x0b-\x0c\x0e-\x1f])+/", ‘ ‘, $s);
        return $level == 1 ? $s."</xml>" : $s;
    }

    /**
     * 将xml转为array
     * @param  string     $xml xml字符串
     * @return array    转换得到的数组
     */
    protected function xml2array($xml){
        //禁止引用外部xml实体
        libxml_disable_entity_loader(true);
        $result= json_decode(json_encode(simplexml_load_string($xml, ‘SimpleXMLElement‘, LIBXML_NOCDATA)), true);
        return $result;
    }

    /**
     *
     * 产生随机字符串,不长于32位
     * @param int $length
     * @return 产生的随机字符串
     */
    protected function getNonceStr($length = 32) {
        $chars = "abcdefghijklmnopqrstuvwxyz0123456789";
        $str ="";
        for ( $i = 0; $i < $length; $i++ )  {
            $str .= substr($chars, mt_rand(0, strlen($chars)-1), 1);
        }
        return $str;
    }

    /**
     * 生成签名
     * @return 签名
     */
    protected function makeSign($data){
        //获取微信支付秘钥
        $key = $this->config[‘pay_apikey‘];
        // 去空
        $data=array_filter($data);
        //签名步骤一:按字典序排序参数
        ksort($data);
        $string_a=http_build_query($data);
        $string_a=urldecode($string_a);
        //签名步骤二:在string后加入KEY
        //$config=$this->config;
        $string_sign_temp=$string_a."&key=".$key;
        //签名步骤三:MD5加密
        $sign = md5($string_sign_temp);
        // 签名步骤四:所有字符转为大写
        $result=strtoupper($sign);
        return $result;
    }

    /**
     * 微信支付发起请求
     */
    protected function curl_post_ssl($url, $xmldata, $second=30,$aHeader=array()){
        $ch = curl_init();
        //超时时间
        curl_setopt($ch,CURLOPT_TIMEOUT,$second);
        curl_setopt($ch,CURLOPT_RETURNTRANSFER, 1);
        //这里设置代理,如果有的话
        //curl_setopt($ch,CURLOPT_PROXY, ‘10.206.30.98‘);
        //curl_setopt($ch,CURLOPT_PROXYPORT, 8080);
        curl_setopt($ch,CURLOPT_URL,$url);
        curl_setopt($ch,CURLOPT_SSL_VERIFYPEER,false);
        curl_setopt($ch,CURLOPT_SSL_VERIFYHOST,false);

        if( count($aHeader) >= 1 ){
            curl_setopt($ch, CURLOPT_HTTPHEADER, $aHeader);
        }

        curl_setopt($ch,CURLOPT_POST, 1);
        curl_setopt($ch,CURLOPT_POSTFIELDS,$xmldata);
        $data = curl_exec($ch);
        if($data){
            curl_close($ch);
            return $data;
        }
        else {
            $error = curl_errno($ch);
            echo "call faild, errorCode:$error\n";
            curl_close($ch);
            return false;
        }
    }

}

返回结果:

{
    "code": 200,
    "status": 1,
    "message": "提交成功",
    "data": {
        "appId": "wxaxxxxxxxxxxxxx",
        "timeStamp": 1499497335,
        "nonceStr": "qqejagkybrh9rxmyfosze71qs49ppcub",
        "package": "prepay_id=wx2017070815022850d11adcb80558793405",
        "signType": "MD5",
        "paySign": "285BCE1F0B93701D927FA5F6CBE10E86"
    }
}

代码没有完美封装,博友们如果有一套好的封装,请留下共享地址哈啊

时间: 2024-10-14 12:47:03

微信小程序实现支付功能的相关文章

【微信小程序】支付过程详解

一.介绍 今天跟大家分享微信小程序集成支付. 二.分析 1.小程序支付API 地址:https://mp.weixin.qq.com/debug/wxadoc/dev/api/api-pay.html#wxrequestpaymentobject 注释: 通过接口的查看,我们知道,小程序端要想使用支付的接口,必须要拿到相应的值 timeStamp:Linux时间戳,可获取本地时间. nonceStr:随机字符串,从服务器统一下单接口返回. package:统一下单接口返回的prepay_id参数

微信小程序篇(微信小程序的支付)

微信小程序的支付和微信公众号的支付是类似的,对比起来还比公众号支付简单了一些,我们只需要调用微信的统一下单接口获取prepay_id之后我们在调用微信的支付即可. 今天我们来封装一般node的支付接口!!! 首先调用统一下单接口我们需要知道一些信息 var bookingNo = 'davdian' + this.createNonceStr() + this.createTimeStamp() var deferred = Q.defer() var appid = config.appId

微信小程序篇(微信小程序的支付)转载 “微信小程序联盟”

微信小程序的支付和微信公众号的支付是类似的,对比起来还比公众号支付简单了一些,我们只需要调用微信的统一下单接口获取prepay_id之后我们在调用微信的支付即可. 今天我们来封装一般node的支付接口!!! 首先调用统一下单接口我们需要知道一些信息 var bookingNo = 'davdian' + this.createNonceStr() + this.createTimeStamp() var deferred = Q.defer() var appid = config.appId

微信小程序开发-蓝牙功能开发

0. 前言 这两天刚好了解了一下微信小程序的蓝牙功能.主要用于配网功能.发现微信的小程序蓝牙API已经封装的很好了.编程起来很方便.什么蓝牙知识都不懂的情况下,不到两天就晚上数据的收发了,剩下的就是数据帧格式的定义,当然这部分就不是本次博客的重点.1. 准备硬件 这里我准备了CH341SER这个作为USB转串口.用sscom5.13.1 串口工具.由于我不太懂硬件开发.硬件部分都是由公司其他人开发的.我只是负责把环境搭建起来.然后负责我的微信小程序开发. 2. 开发小程序简单讲解 onLoad

一号旺铺微信小程序三大新功能:可相互跳转、批量管理门店

可相互跳转.批量管理门店,小程序是抓住零售新机会的一把利器吗?一号旺铺专业.专注开发小程序,就微信小程序的三个功能与大家分享. 同一个公众号下关联的 10 个同主体小程序和 3 个非同主体小程序之间,可以调用接口直接相互跳转. 微信小程序新动向:可相互跳转.批量管理门店 可相互跳转.批量管理门店,小程序是抓住零售新机会的一把利器吗? 小程序到底能不能抓住零售新机会,成效还在验证中.不过为了实现这一目标,小程序又释放了一个重大功能. 6 月 21 日深夜,微信官方表示,与同一公众号关联的小程序之间

微信小程序发起支付流程

小程序调起支付API 需要参数 邮件中参数 API参数名 详细说明 APPID appid appid是微信小程序后台APP的唯一标识,在小程序后台申请小程序账号后,微信会自动分配对应的appid,用于标识该应用.可在小程序-->设置-->开发设置中查看. 微信支付商户号 mch_id 商户申请微信支付后,由微信支付分配的商户收款账号. API密钥 key 交易过程生成签名的密钥,仅保留在商户系统和微信支付后台,不会在网络中传播.商户妥善保管该Key,切勿在网络中传输,不能在其他客户端中存储,

微信小程序人脸识别功能

本文档以微信小程序为例,微信小程序开发人脸核身功能需要两个申请:腾讯云申请人脸识别服务和小程序申请人脸识别服务(注意:只有符合以下条件的机构才能在小程序端申请) 政务:政府机构或事业单位金融:银行.保险医疗:公立医疗机构运营商:电信运营商教育:公立教育机构交通:航空.客运.网约车.交通卡.共享交通.轨道交通.租车旅游:酒店物流:快递.邮政.物流 主要参考文档: 1.腾讯云人脸核身文档 :https://cloud.tencent.com/product/faceid/developer 2.腾讯

微信小程序 五星评价功能

要实现的效果:点击到第几颗星,就要显示到第几颗星,///////////////// 接下来直接查看源码: <view class="l-evalbox row"> <text class="l-evaltxt">满意度:</text> <view class="l-evalist flex-1" bindtap="chooseicon"> <icon class=&quo

NopCommerce 4.2 之微信小程序 - 每日签到功能

/// <summary> /// 调用接口自动签到,并返回签到结果 /// </summary> /// <returns></returns> [Route("index")] [APIFilter(true, true)] public APIResult<object> index() { //设置为驼峰命名(即将实体全部首字母小写处理) var serializerSettings = new JsonSeriali