【测试平台系列】第一章 手撸压力机(七)- 使用gin

今天,我们使用gin框架将压力机做成一个web服务后端。

我们引入gin框架:

golang 复制代码
go get github.com/gin-gonic/gin

在项目根目录新建一个routers目录,并在routers目录下新建:cors.go router.go router_group.go三个文件。

router.go文件主要是我们编辑路由时使用,目的是统一对web服务的接口进行管理。

golang 复制代码
router.go

package routers

import "github.com/gin-gonic/gin"

/*
  路由配置
*/
func initRouters(groups *gin.RouterGroup) {
        {
                groups.POST("/run/testObject/") //运行测试对象接口, url: http://*****/engine/run/testObject/
        }

}

cors.go主要是进行跨域配置,如:前端调用我们的后端服务
cors.go


package routers

import (
        "github.com/gin-gonic/gin"
        "net/http"
)

/*
  跨域配置
  需要在路由使用前进行使用
*/
func cors() gin.HandlerFunc {
        return func(ctx *gin.Context) {
                method := ctx.Request.Method
                ctx.Header("Access-Control-Allow-Origin", "*")
                ctx.Header("Access-Control-Allow-Headers", "Content-Type,AccessToken,X-CSRF-Token, Authorization, Token, x-token")
                ctx.Header("Access-Control-Allow-Methods", "POST, GET, OPTIONS, DELETE, PATCH, PUT")
                ctx.Header("Access-Control-Expose-Headers", "Content-Length, Access-Control-Allow-Origin, Access-Control-Allow-Headers, Content-Type")
                ctx.Header("Access-Control-Allow-Credentials", "true")

                if method == "OPTIONS" {
                        ctx.AbortWithStatus(http.StatusNoContent)
                }
                ctx.Next()
        }
}

router_group.go主要是实现一个路由组,我们可以定义一个路由的统一前缀,对接口进行分类管理。

golang 复制代码
router_group.go

package routers

import "github.com/gin-gonic/gin"

/*
 路由组
*/

func NewRouterGroup() (handler *gin.Engine) {
        handler = gin.Default()
        handler.Use(cors()) // 使用跨域配置,在路由使用前进行使用跨域配置
        group := handler.Group("engine")
        initRouters(group)
        return
}

目前总体项目结构如下:

这样我们就把我们web服务的路由表配置好了,下面我们使用go自带的http,开启并进行服务监听。

在main.go文件中新建runService()函数,并在main方法中调用。

golang 复制代码
package main

import (
        "context"
        "fmt"
        "kitchen-engine/global/config"
        "kitchen-engine/global/log"
        "kitchen-engine/routers"
        "net/http"
        "os"
        "os/signal"
        "syscall"
        "time"
)

// 初始化函数,在main函数之前运行
func init() {
        // 初始化配置信息
        config.InitConfig()
        // 初始化日志配置
        log.InitLogger(config.YC)

}

/*
 启动服务并监听
*/
func runService() {

        handler := routers.NewRouterGroup()
        engineServer := &http.Server{
        // 我们的服务地址是本机的ip:8000端口,这个可以根据自己的需求修改
                Addr:    "0.0.0.0:8000",
                Handler: handler,
                // 控制服务器解析请求报头的键和值(包括请求行)时将读取的最大字节数。它不限制请求体的大小。如果为零,则使用DefaultMaxHeaderBytes。
                MaxHeaderBytes: 1 << 20,
        }

        go func() {
                // 启动http服务
                if err := engineServer.ListenAndServe(); err != nil {
                        log.Logger.Error(fmt.Sprintln("engineServer出错: ", err.Error()))
                        return
                }
                log.Logger.Debug("engine服务启动成功")
        }()

        /// 接收终止信号
        quit := make(chan os.Signal)
        signal.Notify(quit, syscall.SIGINT, syscall.SIGTERM)
        <-quit

        ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
        defer cancel()

        if err := engineServer.Shutdown(ctx); err != nil {
                log.Logger.Info("注销成功")
        }

}

func main() {
        log.Logger.Debug("yc:   ", config.YC)
        runService()

        //// 一个类型中的字段,可以重置,也可以使用默认值,在go中,所有的类型的初始值,都是字段类型的0值,比如string的初始值是""空字符串,int类型的初始值是0等等
        //httpClientSettings := model.HttpClientSettings{
        //        Name:                     "测试厨房",
        //        NoDefaultUserAgentHeader: true,
        //        MaxConnDuration:          1000,
        //        AdvancedOptions: model.AdvancedOptions{
        //                Tls: model.Tls{
        //                        IsVerify:   true,
        //                        VerifyType: 1,
        //                },
        //        },
        //}
        //
        //headers := []model.Header{
        //        model.Header{
        //                Field: "name",
        //                Value: "你好",
        //        },
        //}
        //
        //httpRequest := model.HttpRequest{
        //        Url:                "https://www.baidu.com",
        //        Method:             "GET",
        //        HttpClientSettings: httpClientSettings,
        //        Headers:            headers,
        //}
        //
        //client.RequestHttp(httpRequest)
        log.Logger.Info("欢迎使用zap日志")
}

解释一下为什么我们使用了gin还要在使用net/http进行服务启动,是因为我们需要接受控制台的终止信号,使用gin很难完成,所以我们使用http的Shutdown函数。

golang 复制代码
运行结果如下:
配置文件:D:\workspace\go\planet\seven-day\kitchen-engine\open.yaml初始化成功
2023-06-29T16:24:15.151+0800    DEBUG   kitchen-engine/main.go:62       yc:   {{test /data/logs/runner-go-info.log /data/logs/runner-go-err.log true}}
[GIN-debug] [WARNING] Creating an Engine instance with the Logger and Recovery middleware already attached.

[GIN-debug] [WARNING] Running in "debug" mode. Switch to "release" mode in production.
using env:   export GIN_MODE=release
using code:  gin.SetMode(gin.ReleaseMode)

[GIN-debug] POST   /engine/run/testObject/   --> kitchen-engine/routers.cors.func1 (3 handlers)

好了,本次章节结束。下章,我们将我们的testObject接口进行实现。

相关推荐
song_ly0016 天前
深入理解软件测试覆盖率:从概念到实践
笔记·学习·测试
试着10 天前
【AI面试准备】掌握常规的性能、自动化等测试技术,并在工作中熟练应用
面试·职场和发展·自动化·测试
waves浪游10 天前
论坛系统测试报告
测试工具·测试用例·bug·测试
灰色人生qwer11 天前
使用JMeter 编写的测试计划的多个线程组如何生成独立的线程组报告
jmeter·测试
.格子衫.11 天前
powershell批处理——io校验
测试·powershell
试着12 天前
【AI面试准备】TensorFlow与PyTorch构建缺陷预测模型
人工智能·pytorch·面试·tensorflow·测试
waves浪游12 天前
博客系统测试报告
测试工具·测试用例·bug·测试
智云软件测评服务14 天前
数字化时代下,软件测试中的渗透测试是如何保障安全的?
渗透·测试·漏洞
试着15 天前
【AI面试准备】XMind拆解业务场景识别AI赋能点
人工智能·面试·测试·xmind
waves浪游16 天前
性能测试工具篇
测试工具·测试用例·bug·测试