【测试平台系列】第一章 手撸压力机(十)-定义场景

本文涉及的产品
公共DNS(含HTTPDNS解析),每月1000万次HTTP解析
云解析 DNS,旗舰版 1个月
全局流量管理 GTM,标准版 1个月
简介: 上一章,咱们对http请求进行了一些优化,本章节我们将组成场景去运行。首先场景就是一连串的http接口的请求,我们使用list(列表)来组装成一个场景

上一章,咱们对http请求进行了一些优化,本章节我们将组成场景去运行。首先场景就是一连串的http接口的请求,我们使用list(列表)来组装成一个场景。
首先我们在model目录下,新建test_scene.go文件。

// Package model -----------------------------
// @file      : test_scene.go
// @author    : 被测试耽误的大厨
// @contact   : 13383088061@163.com
// @time      : 2023/7/6 15:53
// -------------------------------------------
package model

import "sync"

// TestScene 业务流程组成的测试场景
type TestScene struct {
        Name        string       `json:"name"`         // 场景名称
        Id          string       `json:"id"`           // 唯一id
        ParentId    string       `json:"parent_id"`    // 父id
        ItemId      string       `json:"item_id"`      // 项目Id
        TeamId      string       `json:"team_id"`      // 团队Id
        SourceId    string       `json:"source_id"`    // 源Id
        ChannelId   string       `json:"channel_id"`   // 渠道Id比如YApi,postman等
        ChannelType string       `json:"channel_type"` // 渠道类型
        TestObjects []TestObject `json:"test_objects"` // 场景
}

/*
 处理测试场景
*/

func (testScene TestScene) Dispose() {
        // 从testScene.TestObjects的最外层开始循环
        for _, testObject := range testScene.TestObjects {
                // 声明一个waitGroup控制等待内层循环请求内部完成
                wg := sync.WaitGroup{}
                // 从一层级的list中读取每个TestObject对象
                wg.Add(1)
                go func(object TestObject) {
                        defer wg.Done()
                        response := TestObjectResponse{
                                Name:        object.Name,
                                Id:          object.Id,
                                SceneId:     testScene.Id,
                ItemId:      object.ItemId,
                                ObjectType:  object.ObjectType,

                        }
                        // 开始进行请求处理
                        object.Dispose(&response)
                }(testObject)
                // 等待内层的list完成后,再开始下一次外层循环
                wg.Wait()
        }
}

上面的内容,我们定义了一个TestScene结构体,来表示我们的测试场景,然后定义了TestScene的结构体来处理场景。
然后我们新建一个reponse_data.go文件,用来声明我们需要的测试结果信息


// Package model -----------------------------
// @file      : response_data.go
// @author    : 被测试耽误的大厨
// @contact   : 13383088061@163.com
// @time      : 2023/7/6 16:21
// -------------------------------------------
package model

// TestObjectResponse 响应信息, 用于返回给调用方
type TestObjectResponse struct {
        Name            string            `json:"name"`         // 对象名称
        Id              string            `json:"id"`           // 对象id
        SceneId         string            `json:"scene_id"`     // 场景id
        ObjectType      string            `json:"object_type"`  // 对象类型http、websocket、dubbo等
        ItemId          string            `json:"item_id"`      // 项目Id
        TeamId          string            `json:"team_id"`      // 团队Id
        Code            int               `json:"code"`
        RequestHeaders  map[string]string `json:"request_headers"`
        ResponseHeaders map[string]string `json:"response_headers"`
        Response        string            `json:"response"`
        RequestTime     int64             `json:"request_time"`
}

我们将原来test_object.go中的结构体单独提取到reponse_data.go文件中,并对一些无用的字段进行优化。

// Package model -----------------------------
// @file      : test_object.go
// @author    : 被测试耽误的大厨
// @contact   : 13383088061@163.com
// @time      : 2023/6/11 20:38
// -------------------------------------------
package model

// TestObject 测试对象结构体
type TestObject struct {
        Name        string      `json:"name"`        // 对象名称
        Id          string      `json:"id"`          // 唯一id
        ObjectType  string      `json:"object_type"` // 对象类型http、websocket、dubbo等
        ItemId      string      `json:"item_id"`     // 项目Id
        TeamId      string      `json:"team_id"`     // 团队Id
        HttpRequest HttpRequest `json:"http_request"`
}

// Dispose 测试对象的处理函数,在go语言中 Dispose方法是TestObject对象的方法,其他对象不能使用
func (to TestObject) Dispose(response *TestObjectResponse) {
        switch to.ObjectType {
        case HTTP1: // 由于我们有个多类型,为了方便统计,我们定义好变量,直接进行比对即可
                to.HttpRequest.Request(response)
                return
        }
        return
}

然后我们在service目录下新建scene_api.go文件,新建RunTestScene方法来实现请求读取。

// Package service -----------------------------
// @file      : scene_api.go
// @author    : 被测试耽误的大厨
// @contact   : 13383088061@163.com
// @time      : 2023/7/6 16:06
// -------------------------------------------
package service

import (
        "encoding/json"
        "fmt"
        "github.com/gin-gonic/gin"
        "github.com/google/uuid"
        "kitchen-engine/global/common"
        "kitchen-engine/global/log"
        "kitchen-engine/model"
        "net/http"
)

/*
        RunTestScene 实现run/testObject/接口
*/

func RunTestScene(c *gin.Context) {

        // 声明一个TO对象
        var testScene model.TestScene

        // 接收json格式的请求数据
        err := c.ShouldBindJSON(&testScene)
        id := uuid.New().String()
        // 如果请求格式错误
        if err != nil {
                log.Logger.Error("请求数据格式错误", err.Error())
                common.ReturnResponse(c, http.StatusBadRequest, id, "请求数据格式错误!", err.Error())
                return
        }

        // 使用json包解析以下TO对象, 解析出来为[]byte类型
        requestJson, _ := json.Marshal(testScene)
        // 打印以下日志, 使用fmt.Sprintf包格式花数据,%s 表示string(requestJson)为字符串类型,如果不确定类型,可以使用%v表示
        log.Logger.Debug(fmt.Sprintf("测试对象: %s", string(requestJson)))

        // 如果场景为空,直接返回
        if testScene.TestObjects == nil || len(testScene.TestObjects) == 0 {
                common.ReturnResponse(c, http.StatusBadRequest, id, "请求错误!", "场景不能为空")
                return
        }

        // 开始处理TO
        testScene.Dispose()
        // 返回响应消息
        common.ReturnResponse(c, http.StatusOK, id, "请求成功!", "")
        return
}

同时,将service目录下的api.go更名为object_api.go,并对其中的无用字段进行优化:

// Package service -----------------------------
// @file      : object_api.go
// @author    : 被测试耽误的大厨
// @contact   : 13383088061@163.com
// @time      : 2023/7/4 18:05
// -------------------------------------------
package service

import (
        "encoding/json"
        "fmt"
        "github.com/gin-gonic/gin"
        "github.com/google/uuid"
        "kitchen-engine/global/common"
        "kitchen-engine/global/log"
        "kitchen-engine/model"
        "net/http"
)

/*
        RunTestObject 实现run/testObject/接口
*/

func RunTestObject(c *gin.Context) {

        // 声明一个TO对象
        var testObject model.TestObject

        // 接收json格式的请求数据
        err := c.ShouldBindJSON(&testObject)
        id := uuid.New().String()
        // 如果请求格式错误
        if err != nil {
                log.Logger.Error("请求数据格式错误", err.Error())
                common.ReturnResponse(c, http.StatusBadRequest, id, "请求数据格式错误!", err.Error())
                return
        }

        // 使用json包解析以下TO对象, 解析出来为[]byte类型
        requestJson, _ := json.Marshal(testObject)
        // 打印以下日志, 使用fmt.Sprintf包格式花数据,%s 表示string(requestJson)为字符串类型,如果不确定类型,可以使用%v表示
        log.Logger.Debug(fmt.Sprintf("测试对象: %s", string(requestJson)))

        response := model.TestObjectResponse{
                Name:       testObject.Name,
                Id:         testObject.Id,
                ObjectType: testObject.ObjectType,
                ItemId:     testObject.ItemId,
        }

        // 开始处理TO
        testObject.Dispose(&response)
        // 返回响应消息
        common.ReturnResponse(c, http.StatusOK, id, "请求成功!", response)
        return
}

我们再对Http请求添加一个timeout字段,并在发送请求是设置超时时间。

// HttpRequest http请求的结构
type HttpRequest struct {
        Url                string             `json:"url"`                  // 接口uri
        Method             string             `json:"method"`               // 接口方法,Get Post Update...
        RequestTimeout     int64              `json:"request_timeout"`      // 请求超时时长,默认为30秒,防止请求阻塞
        Headers            []Header           `json:"headers"`              // 接口请求头
        Querys             []Query            `json:"querys"`               // get请求时的url
        Cookies            []Cookie           `json:"cookies"`              // cookie
        Body               *Body              `json:"body"`                 // 请求提
        HttpClientSettings HttpClientSettings `json:"http_client_settings"` // http客户端配置
}
在http_request.go文件中,新建HttpRequest结构体的setReqTime方法
// 设置请求超时时间
func (hr *HttpRequest) setReqTimeout(req *fasthttp.Request) {
        if hr.RequestTimeout <= 0 {
                hr.RequestTimeout = 30
        }
        req.SetTimeout(time.Duration(hr.RequestTimeout) * time.Second)
}

并对HttpRequest结构体的Request方法进行优化
func (hr *HttpRequest) Request(response *TestObjectResponse) {

        // 使用fasthttp 协程池

        // 新建一个http请求
        req := fasthttp.AcquireRequest()
        defer fasthttp.ReleaseRequest(req)
        // 新建一个http响应接受服务端的返回
        resp := fasthttp.AcquireResponse()
        defer fasthttp.ReleaseResponse(resp)

        // 新建一个http的客户端, newHttpClient是一个方法,在下面
        client := newHttpClient(hr.HttpClientSettings)

        // 设置请求方法
        hr.methodInit(req)

        // 设置header
        hr.headerInit(req)
        // 设置query
        hr.queryInit(req)
        // 设置cookie
        hr.cookieInit(req)

        // 设置url
        hr.urlInit(req)

        // 设置body
        hr.bodyInit(req)

        // 设置请求超时时间
        hr.setReqTimeout(req)

        // 记录开始时间
        startTime := time.Now()
        // 开始请求
        err := client.Do(req, resp)
        // 计算响应时间差值
        requestTime := time.Since(startTime)
        response.RequestTime = requestTime.Milliseconds()
        response.Code = resp.StatusCode()
        if err != nil {
                response.Response = err.Error()
                return
        }

        // 如果响应中有压缩类型,防止压缩类型出现乱码
        switch string(resp.Header.ContentEncoding()) {
        case "br", "deflate", "gzip":
                b, _ := resp.BodyUncompressed()
                response.Response = string(b)
        default:
                response.Response = string(resp.Body())
        }

}

同时,我们添加run/testScene接口,修改router.go中的initRouters方法。


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

        }

}

这样,我们对场景的简单封装就完成了,下一步我们将对场景以及请求进行更全面的适配。

相关文章
|
3月前
|
Kubernetes 测试技术 Perl
混沌测试平台 Chaos Mesh
混沌测试平台 Chaos Mesh
105 1
|
8天前
|
网络协议 关系型数据库 应用服务中间件
【项目场景】请求数据时测试环境比生产环境多花了1秒是怎么回事?
这是一位粉丝(谢同学)给V哥的留言,描述了他在优化系统查询时遇到的问题:测试环境优化达标,但生产环境响应时间多出1秒。通过抓包分析,发现MySQL请求和响应之间存在500毫秒的延迟,怀疑是网络传输开销。V哥给出了以下优化建议:
|
8天前
|
监控 安全 测试技术
构建高效的精准测试平台:设计与实现指南
在软件开发过程中,精准测试是确保产品质量和性能的关键环节。一个精准的测试平台能够自动化测试流程,提高测试效率,缩短测试周期,并提供准确的测试结果。本文将分享如何设计和实现一个精准测试平台,从需求分析到技术选型,再到具体的实现步骤。
40 1
|
4月前
|
传感器 数据采集 监控
LabVIEW电池管理系统测试平台
LabVIEW电池管理系统测试平台
61 4
|
26天前
|
人工智能 监控 测试技术
云应用开发平台测试
云应用开发平台测试
42 2
|
8天前
|
监控 安全 测试技术
构建高效精准测试平台:设计与实现全攻略
在软件开发过程中,精准测试是确保产品质量的关键环节。一个高效、精准的测试平台能够自动化测试流程,提高测试覆盖率,缩短测试周期。本文将分享如何设计和实现一个精准测试平台,从需求分析到技术选型,再到具体的实现步骤。
28 0
|
2月前
|
设计模式 SQL 安全
PHP中的设计模式:单例模式的深入探索与实践在PHP的编程实践中,设计模式是解决常见软件设计问题的最佳实践。单例模式作为设计模式中的一种,确保一个类只有一个实例,并提供全局访问点,广泛应用于配置管理、日志记录和测试框架等场景。本文将深入探讨单例模式的原理、实现方式及其在PHP中的应用,帮助开发者更好地理解和运用这一设计模式。
在PHP开发中,单例模式通过确保类仅有一个实例并提供一个全局访问点,有效管理和访问共享资源。本文详细介绍了单例模式的概念、PHP实现方式及应用场景,并通过具体代码示例展示如何在PHP中实现单例模式以及如何在实际项目中正确使用它来优化代码结构和性能。
44 2
|
2月前
|
JavaScript 前端开发 数据库
数据库测试场景实践总结
本文介绍了数据库超时和应用锁表SSDB测试场景的验证方法,通过锁定数据表模拟写入失败情况,并利用SSDB进行重试。测试需开发人员配合验证功能。同时,提供了SSDB服务器登录、查询队列数量及重启服务等常用命令。适用于验证和解决数据库写入问题。
34 7
|
3月前
|
测试技术 Android开发 iOS开发
Appium 是一个开源的自动化测试框架,它支持多种平台和多种编程语言
Appium是一款开源自动化测试框架,支持iOS和Android多平台及多种编程语言。通过WebDriver协议,开发者可编写自动化测试脚本。在iPhone上实现屏幕点击等操作需安装Appium及其依赖,启动服务器,并设置所需的测试环境参数。利用Python等语言编写测试脚本,模拟用户交互行为,最后运行测试脚本来验证应用功能。对于iPhone测试,需准备真实设备或Xcode模拟器。
110 1
|
3月前
|
运维 Kubernetes 监控

热门文章

最新文章