[golang gin框架] 34.Gin 商城项目- 集成支付宝微信支付、生成支付二维码、监听处理异步通知跳转到订单页面

52 篇文章 28 订阅
10 篇文章 2 订阅

一.界面展示

当用户点击'去支付'时,请求支付界面,并 展示对应订单相关数据,以及 支付方式相关操作,点击对应的支付方式,进行支付操作
该界面对应的功能:
1.进入该界面,后台逻辑判断: 是否存在该订单,如果不存在,则跳转到购物车页面;如果存在,则获取对应订单相关数据,并 渲染到页面
2.选择支付方式,根据不同支付方式进行支付操作
支付宝支付: 点击 '支付宝'支付方式,会请求后台,通过支付宝相关配置以及方法,生成一个 支付url,重定向到对应的支付url,用户扫描支付,支付完成后,跳转到服务端设置的 返回页面,然后支付宝服务器会请求配置的服务端 回调地址,进行订单校验以及对应的 订单逻辑处理
微信支付: 点击'微信支付'支付方式,也会请求后台,通过微信相关配置以及方法,生成一个支付 二维码,展示到界面,用户扫描支付完成后,微信服务器会请求配置的服务端 回调地址,进行订单校验以及对应的 订单逻辑处理(修改订单支付状态等),然后前端在这个支付页面会每个几秒(一般方法是定时 ajax轮询,或者 web_socket请求)请求一个接口,判断该订单是否支付,如果支付了,就跳转到支付成功页面

去支付页面

微信支付页面

点击'微信支付'支付方式,也会请求后台,通过微信相关配置以及方法,生成一个支付二维码,展示到界面,用户扫描支付完成后,微信服务器会请求配置的服务端回调地址,进行订单校验以及对应的订单逻辑处理(修改订单支付状态等),然后前端在这个支付页面会每个几秒(一般方法是定时ajax轮询,或者web_socket请求)请求一个接口,判断该订单是否支付,如果支付了,就跳转到支付成功页面

微信支付页面

点击 '支付宝'支付方式,会请求后台,通过支付宝相关配置以及方法,生成一个支付url,重定向到对应的支付url,用户扫描支付,支付完成后,跳转到服务端设置的返回页面,然后支付宝服务器会请求配置的服务端回调地址,进行订单校验以及对应的订单逻辑处理

二.代码展示

  1. 集成支付宝支付控制器代码

要使用 github.com/smartwalle/alipay/v3需要先引入:
直接在import中引入 github.com/smartwalle/alipay/v3,然后go mod tidy就可以了
package frontend

import (
    "fmt"
    "net/http"
    "time"
    "github.com/gin-gonic/gin"
    "github.com/smartwalle/alipay/v3"
)

type AlipayController struct{}

//支付宝支付请求接口
func (con AlipayController) Alipay(c *gin.Context) {
    //1、获取订单号 判断此订单号是否值当前用户的
    //2、获取订单里面的支付信息
    var privateKey = "xxx" // 必须,上一步中使用 RSA签名验签工具 生成的私钥
    var client, err = alipay.New("2021xxx588", privateKey, true)
    client.LoadAppPublicCertFromFile("crt/appCertPublicKey_2021xxx588.crt") // 加载应用公钥证书
    client.LoadAliPayRootCertFromFile("crt/alipayRootCert.crt")   // 加载支付宝根证书
    client.LoadAliPayPublicCertFromFile("crt/alipayCertPublicKey_RSA2.crt")  // 加载支付宝公钥证书

    // 将 key 的验证调整到初始化阶段
    if err != nil {
        fmt.Println(err)
        return
    }

    //支付宝PC扫描方式支付
    var p = alipay.TradePagePay{}
    p.NotifyURL = "http://xxx/v3/alipayNotify"  // 回调地址
    p.ReturnURL = "http://xxx5/v3/alipayReturn"  //支付后跳转地址
    p.Subject = "测试 公钥证书模式-这是一个gin订单"
    template := "2006-01-02 15:04:05"
    p.OutTradeNo = time.Now().Format(template)
    p.TotalAmount = "0.1"  //支付金额(元)
    p.ProductCode = "FAST_INSTANT_TRADE_PAY"  //根据支付方式调整

    var url, err4 = client.TradePagePay(p)
    if err4 != nil {
        fmt.Println(err4)
    }

    //支付url
    var payURL = url.String()

    //重定向到支付url
    c.Redirect(http.StatusFound, payURL)
}

//支付回调
func (con AlipayController) AlipayNotify(c *gin.Context) {
    var privateKey = "xxx" // 必须,上一步中使用 RSA签名验签工具 生成的私钥
    var client, err = alipay.New("202xxx588", privateKey, true)
    client.LoadAppPublicCertFromFile("crt/appCertPublicKey_202xxx588.crt") // 加载应用公钥证书
    client.LoadAliPayRootCertFromFile("crt/alipayRootCert.crt")  // 加载支付宝根证书
    client.LoadAliPayPublicCertFromFile("crt/alipayCertPublicKey_RSA2.crt") // 加载支付宝公钥证书

    if err != nil {
        fmt.Println(err)
        return
    }

    req := c.Request
    req.ParseForm()
    //支付校验
    ok, _ := client.VerifySign(req.Form)

    fmt.Println(ok)

    fmt.Println(req.Form)
    //订单逻辑处理...
    c.String(200, "ok")
}

//支付后跳转页面
func (con AlipayController) AlipayReturn(c *gin.Context) {
    c.String(200, "支付成功")
}
  1. 集成微信支付控制器代码

要使用 github.com/objcoding/wxpay需要先引入:
直接在import中引入github.com/objcoding/wxpay,然后go mod tidy就可以了, qrcode的引入方式也是一样的
package frontend

import (
    "fmt"
    "net/http"
    "time"
    "github.com/gin-gonic/gin"
    "github.com/objcoding/wxpay"
    qrcode "github.com/skip2/go-qrcode"
)

type WxpayController struct{}

//微信支付
func (con WxpayController) Wxpay(c *gin.Context) {
    //1、获取订单号 判断此订单号是否值当前用户的
    //2、获取订单里面的支付信息
    //1、配置基本信息
    account := wxpay.NewAccount(
        "wx7xxx6e4",  // appId
        "150xxx",  //商户id
        "zhongyxxx66",  //密钥
        false,
    )
    client := wxpay.NewClient(account)

    //2、获取ip地址,订单号等信息
    ip := c.ClientIP()
    template := "200601021504"
    tradeNo := time.Now().Format(template)
    //3、调用统一下单
    params := make(wxpay.Params)
    params.SetString("body", "Gin微信支付").
        SetString("out_trade_no", tradeNo).
        SetInt64("total_fee", 1). //1分
        SetString("spbill_create_ip", ip).
        SetString("notify_url", "http://xxx/wxpay/notify"). //必须在商户平台的Native支付回调链接里面配置
        // SetString("trade_type", "APP")
        SetString("trade_type", "NATIVE") //网站支付需要改为NATIVE

    p, err1 := client.UnifiedOrder(params)
    if err1 != nil {
        c.String(http.StatusOK, "生成二维码失败")
        return
    }
    //4、获取code_url 生成支付二维码
    png, err := qrcode.Encode(p["code_url"], qrcode.Medium, 256)
    if err != nil {
        c.String(http.StatusOK, "生成二维码失败")
        return
    }
    c.String(http.StatusOK, string(png))
}

/*
支付回调:
1、发布到服务器测试
2、必须在商户平台的Native支付回调链接里面配置
3、如何接收XML的数据  c.GetRawData()
4、如何获取数据
5、如何验证数据
6、更新数据
*/
func (con WxpayController) WxpayNotify(c *gin.Context) {
    //1、获取表单传过来的xml数据
    xmlByte, _ := c.GetRawData()
    xmlStr := string(xmlByte)
    postParams := wxpay.XmlToMap(xmlStr)
    //2、校验签名
    account := wxpay.NewAccount(
        "wx7xxxe4",
        "150xxx1",
        "zhongxxx66",
        false,
    )
    client := wxpay.NewClient(account)
    isValidate := client.ValidSign(postParams)

    fmt.Println(isValidate)
    fmt.Println("-----更新订单-----")
    fmt.Println(postParams)
    //3、更新订单
    c.String(200, "ok")
}
  1. 查看订单支付状态方法

在controllers/frontend/BuyController.go下增加查看订单支付状态方法:前端支付页面,微信支付会每个几秒请求该接口,根据返回结果,判断是否跳转到订单支付成功页面
//查看订单支付状态
func (con BuyController) OrderPayStatus(c *gin.Context) {
    id, err := models.Int(c.Query("id"))
    if err != nil {
        c.JSON(http.StatusOK, gin.H{
            "success": false,
            "message": "传入参数错误",
        })
        return
    }
    //获取用户信息
    user := models.User{}
    models.Cookie.Get(c, "user", &user)

    //获取主订单信息
    order := models.Order{}
    models.DB.Where("id = ?", id).Find(&order)

    //判断当前数据是否合法
    if user.Id != order.Uid {
        c.JSON(http.StatusOK, gin.H{
            "success": false,
            "message": "非法请求",
        })
        return
    }

    //判断是否支付
    if order.PayStatus == 1 && order.OrderStatus == 1 {
        c.JSON(http.StatusOK, gin.H{
            "success": true,
            "message": "支付成功",
        })
        return
    } else {
        c.JSON(http.StatusOK, gin.H{
            "success": false,
            "message": "支付成功",
        })
        return
    }
}
  1. 路由

把微信支付,支付宝支付相关路由写入routers/frontendRouters.go中
//查看订单支付状态
defaultRouters.GET("/buy/orderPayStatus", middlewares.InitUserAuthMiddleware, frontend.BuyController{}.OrderPayStatus)
//支付宝支付
defaultRouters.GET("/alipay", middlewares.InitUserAuthMiddleware, frontend.AlipayController{}.Alipay)
//支付宝支付回调
defaultRouters.POST("/alipayNotify", frontend.AlipayController{}.AlipayNotify)
//支付宝支付完成后跳转
defaultRouters.GET("/alipayReturn", middlewares.InitUserAuthMiddleware, frontend.AlipayController{}.AlipayReturn)
//微信支付
defaultRouters.GET("/wxpay", middlewares.InitUserAuthMiddleware, frontend.WxpayController{}.Wxpay)
//微信支付回调
defaultRouters.POST("/wxpay/notify", frontend.WxpayController{}.WxpayNotify)
  1. html

微信,支付宝支付操作处理
{{ define "frontend/buy/pay.html" }}

{{ template "frontend/public/page_header.html" .}}
<!--end header -->
<link rel="stylesheet" href="/static/frontend/css/pay.css" />
<script src="/static/frontend/js/bootstrap.js"></script>
<link rel="stylesheet" href="/static/frontend/css/bootstrap.css">
<!-- start banner_x -->
<div class="banner_x center clearfix">
    <a href="/" target="_blank">
        <div class="logo fl"></div>
    </a>
    <div class="wdgwc fl ml40">支付页面 </div>
</div>

<div class="page-main">
    <div class="checkout-box">
        <div class="section section-order">
            <div class="order-info clearfix">
                <div class="fl">
                    <h2 class="title">订单提交成功!去付款咯~</h2>
                    <p class="order-time" id="J_deliverDesc"></p>
                    <p class="order-time">请在<span class="pay-time-tip">47小时59分</span>内完成支付, 超时后将取消订单</p>
                    <p class="post-info" id="J_postInfo">
                        收货信息: {{.order.Name}} &nbsp;&nbsp;{{.order.Phone}} &nbsp;&nbsp; {{.order.Address}} </p>
                </div>
                <div class="fr">
                    <p class="total">
                        应付总额:<span class="money"><em>1000元</em>元</span>
                    </p>
                    <br>
                    <br>
                    <a href="javascript:void(0);" class="show-detail" id="J_showDetail"
                        data-stat-id="db85b2885a2fdc53">订单详情</a>
                </div>
            </div>
            <i class="iconfont icon-right">√</i>
            <div class="order-detail">
                <ul>
                    <li class="clearfix">
                        <div class="label">订单号:</div>
                        <div class="content">
                            <span class="order-num">{{.order.OrderId}}</span>
                        </div>
                    </li>
                    <li class="clearfix">
                        <div class="label">收货信息:</div>
                        <div class="content">
                            收货信息: {{.order.Name}} &nbsp;&nbsp;{{.order.Phone}} &nbsp;&nbsp; {{.order.Address}} </div>
                    </li>
                    <li class="clearfix">
                        <div class="label">商品:</div>
                        <div class="content">
                            {{range $key,$value:=.orderItems}}
                            <p>{{$value.ProductTitle}} {{$value.GoodsVersion}} {{$value.GoodsColor}}
                                数量:{{$value.ProductNum}} 价格:{{$value.ProductPrice}}</p>
                            {{end}}
                        </div>
                    </li>
                    <li class="clearfix hide">
                        <div class="label">配送时间:</div>
                        <div class="content">
                            不限送货时间 </div>
                    </li>
                    <li class="clearfix">
                        <div class="label">发票信息:</div>
                        <div class="content">
                            电子发票 个人 </div>
                    </li>
                </ul>
            </div>
        </div>

        <div class="section section-payment">
            <div class="cash-title" id="J_cashTitle">
                选择以下支付方式付款
            </div>

            <div class="payment-box ">

                <div class="payment-body">
                    <ul class="clearfix payment-list J_paymentList J_linksign-customize">
                        <ul class="clearfix payment-list J_paymentList J_linksign-customize">
                            <li id="weixinPay">
                                <img src="/static/frontend/image/weixinpay0701.png" alt="微信支付" />
                            </li>
                            <li id="alipay">
                                <a href="/alipay?id={{.order.Id}}" target="_blank">
                                    <img src="/static/frontend/image/payOnline_zfb.png" alt="支付宝" /></a>
                            </li>
    
                        </ul>

                    </ul>
                </div>
            </div>

        </div>
    </div>
</div>

<!-- 微信支付 Dialog -->
<div class="modal fade" id="weixinPayModel" tabindex="-1" role="dialog">
    <div class="modal-dialog" role="document">
        <div class="modal-content">
            <div class="modal-header">
                <button type="button" class="close" data-dismiss="modal" aria-label="Close"><span
                        aria-hidden="true">&times;</span></button>
                <h4 class="modal-title" id="myModalLabel">微信支付</h4>
            </div>
            <div class="modal-body">
                <img class="lcode" src="/wxpay?id={{.order.Id}}" />
                <img class="rphone" src="/static/frontend/image/phone.png" />
            </div>
        </div>
    </div>
</div>
<!-- 支付宝支付Dialog -->

<div class="modal fade" id="alipayModel" tabindex="-1" role="dialog">
    <div class="modal-dialog" role="document">
        <div class="modal-content">
            <div class="modal-header">
                <button type="button" class="close" data-dismiss="modal" aria-label="Close"><span
                        aria-hidden="true">&times;</span></button>
                <h4 class="modal-title" id="myModalLabel">在线支付提醒</h4>
            </div>
            <div class="modal-body">

                感谢您的支持,支付完成后,页面会自动跳转到订单页面,若需要重新支付请点击“继续支付”按钮,若已完成支付请点击“已完成支付”
                <div class="modal-footer">

                    <button type="button" class="btn btn-default" data-dismiss="modal">继续支付</button>

                    <button type="button" class="btn btn-primary" id="alipayDone">已完成支付</button>
                </div>
            </div>

        </div>
    </div>
</div>

<script>
    $(function () {
        $(".show-detail").click(function () {
            $(".order-detail").slideToggle();
        })
        $("#weixinPay").click(function () {
            $('#weixinPayModel').modal('show');
        })
        $("#alipay").click(function () {
            $('#alipayModel').modal('show');
        })
        $("#alipayDone").click(function () {
            location.href = "/user/order"
        })
        //微信支付:弹出二维码模态框后,每个3秒定时定期支付状态,根据状态,判断逻辑
        setInterval(function () {
            $.get('/buy/orderPayStatus?id={{.order.Id}}', function (response) {            
                if (response.success) { //跳转到支付成功订单页面
                    location.href = '/user/order'
                }
            })
        }, 3000);
    })
</script>
<!-- footer -->
{{ template "frontend/public/page_footer.html" .}}
</body>
</html>
{{end}}

[上一节][golang gin框架] 33.Gin 商城项目- 微信支付操作相关功能讲解

[下一节][golang gin框架] 35.Gin 商城项目- 用户中心制作以及订单列表数据渲染(分页展示,订单状态,筛选订单 搜索订单,订单详情, 以及后台订单管理功能实现逻辑 )

  • 1
    点赞
  • 3
    收藏
    觉得还不错? 一键收藏
  • 0
    评论
好的,我可以为您提供一些关于使用 Golang Gin 框架进行项目实战的建议。 首先,您需要安装 GolangGin 框架。安装过程可以参考官方文档进行操作。然后,您可以按照以下步骤进行项目实战: 1. 创建项目结构 在您的 GOPATH 目录下创建一个新的项目文件夹。通常情况下,您需要为项目创建以下子文件夹: ``` . ├── main.go ├── config │ ├── config.go │ └── ... ├── controller │ ├── home_controller.go │ └── ... ├── middleware │ ├── auth.go │ └── ... ├── model │ ├── user.go │ └── ... ├── router │ ├── router.go │ └── ... └── service ├── user_service.go └── ... ``` 2. 创建路由 在 router 文件夹中创建一个 router.go 文件,并编写以下代码: ```go package router import ( "github.com/gin-gonic/gin" "your-project/controller" "your-project/middleware" ) func SetupRouter() *gin.Engine { r := gin.Default() // 中间件 r.Use(middleware.AuthMiddleware()) // 路由 r.GET("/", controller.HomePageHandler) return r } ``` 在上面的代码中,我们使用了 gin 框架中的 gin.Default() 函数来创建一个默认的 gin.Engine 实例,并使用中间件来进行身份验证。然后我们定义了一个路由,将根路径映射到 HomePageHandler 方法。 3. 创建控制器 在 controller 文件夹中创建 home_controller.go 文件,并编写以下代码: ```go package controller import ( "github.com/gin-gonic/gin" "your-project/service" ) func HomePageHandler(c *gin.Context) { users := service.GetAllUsers() c.JSON(200, users) } ``` 在上面的代码中,我们定义了一个 HomePageHandler 方法,它接收一个 gin.Context 对象作为参数,并使用 service 层中的 GetAllUsers 函数获取所有用户的数据。然后我们使用 JSON 方法将数据以 JSON 格式返回给客户端。 4. 创建服务层 在 service 文件夹中创建 user_service.go 文件,并编写以下代码: ```go package service import "your-project/model" func GetAllUsers() []*model.User { // TODO: 查询数据库获取所有用户数据 return []*model.User{} } ``` 在上面的代码中,我们定义了一个 GetAllUsers 函数,它返回一个包含所有用户数据的切片。在实际项目中,我们需要查询数据库来获取这些数据。 5. 创建模型 在 model 文件夹中创建 user.go 文件,并编写以下代码: ```go package model type User struct { ID int `json:"id"` Name string `json:"name"` Age int `json:"age"` } ``` 在上面的代码中,我们定义了一个 User 结构体,它包含三个字段:ID、Name 和 Age。这些字段将用于存储每个用户的信息。 6. 创建配置文件 在 config 文件夹中创建 config.go 文件,并编写以下代码: ```go package config const ( DB_HOST = "localhost" DB_PORT = "3306" DB_USERNAME = "root" DB_PASSWORD = "password" DB_NAME = "your-database-name" ) ``` 在上面的代码中,我们定义了一些常量,用于存储数据库配置信息。这些常量将在连接数据库时使用。 7. 连接数据库 在 service 层中的 GetAllUsers 函数中,我们需要连接数据库来获取所有用户数据。可以使用以下代码来连接 MySQL 数据库: ```go package service import ( "database/sql" "fmt" "your-project/config" "your-project/model" ) func GetAllUsers() []*model.User { // 连接数据库 db, err := sql.Open("mysql", fmt.Sprintf("%s:%s@tcp(%s:%s)/%s", config.DB_USERNAME, config.DB_PASSWORD, config.DB_HOST, config.DB_PORT, config.DB_NAME, )) if err != nil { panic(err) } defer db.Close() // 查询所有用户数据 rows, err := db.Query("SELECT * FROM users") if err != nil { panic(err) } // 解析查询结果 users := []*model.User{} for rows.Next() { user := &model.User{} err := rows.Scan(&user.ID, &user.Name, &user.Age) if err != nil { panic(err) } users = append(users, user) } return users } ``` 在上面的代码中,我们使用 sql.Open 函数来打开 MySQL 数据库连接。然后我们使用 db.Query 函数来查询所有用户数据,并使用 rows.Scan 函数将查询结果解析为 User 结构体的切片。 以上就是使用 Golang Gin 框架进行项目实战的流程。当然,在实际项目中,您需要根据自己的需求进行相应的修改和扩展。

“相关推荐”对你有帮助么?

  • 非常没帮助
  • 没帮助
  • 一般
  • 有帮助
  • 非常有帮助
提交
评论
添加红包

请填写红包祝福语或标题

红包个数最小为10个

红包金额最低5元

当前余额3.43前往充值 >
需支付:10.00
成就一亿技术人!
领取后你会自动成为博主和红包主的粉丝 规则
hope_wisdom
发出的红包
实付
使用余额支付
点击重新获取
扫码支付
钱包余额 0

抵扣说明:

1.余额是钱包充值的虚拟货币,按照1:1的比例进行支付金额的抵扣。
2.余额无法直接购买下载,可以购买VIP、付费专栏及课程。

余额充值