[Pitaya Demo解读笔记]5.pipeline demo

项目目录:examples/demo/custom_metrics

本例展示了如何使用结构体验证 (struct validation)和处理函数钩子(handler hooks)

运行

服务器开起来

sh 复制代码
 go run main.go

服务监听了 3251 端口,使用命令行跑一下基本功能

有一点需要注意一下,之前的 demo 都设置了 component.WithNameFunc(strings.ToLower),所以之前的 router 都是全小写,这个示例程序并没有设置,访问的时候是大小写敏感的,否则会找不到服务器或 handler:

sh 复制代码
 > pitaya-cli
 Pitaya REPL Client
 >>> request metagameDemo.metagamehandler.createplayercheat {"name":"test", "email":"test@gmail.com", "softCurrency":10, "hardCurrency":20}
 >>> sv->{"code":"PIT-404","msg":"pitaya/handler: metagameDemo.metagamehandler.createplayercheat not found"}
 >>> request metagameDemo.metagameHandler.CreatePlayerCheat {"name":"test", "email":"test@gmail.com", "softCurrency":10, "hardCurrency":20}
 >>> sv->{"msg":"ok"}

服务器端输出日志如下:

sh 复制代码
 time="2023-10-16T16:13:59+08:00" level=info msg="Simple Before exec" requestId=BJIiSpBZ1Pue7CIr9md6iz route=metagameDemo.metagameHandler.CreatePlayerCheat source=pitaya userId=
 time="2023-10-16T16:13:59+08:00" level=info msg="Name: test" requestId=BJIiSpBZ1Pue7CIr9md6iz route=metagameDemo.metagameHandler.CreatePlayerCheat source=pitaya userId=
 time="2023-10-16T16:13:59+08:00" level=info msg="Email: test@gmail.com" requestId=BJIiSpBZ1Pue7CIr9md6iz route=metagameDemo.metagameHandler.CreatePlayerCheat source=pitaya userId=
 time="2023-10-16T16:13:59+08:00" level=info msg="SoftCurrency: 10" requestId=BJIiSpBZ1Pue7CIr9md6iz route=metagameDemo.metagameHandler.CreatePlayerCheat source=pitaya userId=
 time="2023-10-16T16:13:59+08:00" level=info msg="HardCurrency: 20" requestId=BJIiSpBZ1Pue7CIr9md6iz route=metagameDemo.metagameHandler.CreatePlayerCheat source=pitaya userId=
 time="2023-10-16T16:13:59+08:00" level=debug msg="SID=4, Data={"name":"test","email":"test@gmail.com","softCurrency":10,"hardCurrency":20}" requestId=BJIiSpBZ1Pue7CIr9md6iz route=metagameDemo.metagameHandler.CreatePlayerCheat source=pitaya userId=
 time="2023-10-16T16:13:59+08:00" level=info msg="CreatePlayerChest called" requestId=BJIiSpBZ1Pue7CIr9md6iz route=metagameDemo.metagameHandler.CreatePlayerCheat source=pitaya userId=
 time="2023-10-16T16:13:59+08:00" level=info msg="Simple After exec - response: &{ok} , error: <nil>" requestId=BJIiSpBZ1Pue7CIr9md6iz route=metagameDemo.metagameHandler.CreatePlayerCheat source=pitaya userId=

钩子方法输出了 Simple Before 和 Simple After...

如果我们给一个不满足验证的值,hardCurrency = 2000,服务器会验证输入,返回错误信息:

sh 复制代码
 >>> request metagameDemo.metagameHandler.CreatePlayerCheat {"name":"test", "email":"test@gmail.com", "softCurrency":10, "hardCurrency":2000}
 >>> sv->{"code":"PIT-000","msg":"Key: 'CreatePlayerCheatArgs.HardCurrency' Error:Field validation for 'HardCurrency' failed on the 'lte' tag"}

代码分析

结构体验证

开启结构体验证:

go 复制代码
 // main.go:101
 config.DefaultPipelines.StructValidation.Enabled = true

使用 tag validate 设置验证规则:

go 复制代码
 type CreatePlayerCheatArgs struct {
     Name         string `json:"name"`
     Email        string `json:"email" validate:"email"`
     SoftCurrency int    `json:"softCurrency" validate:"gte=0,lte=1000"`
     HardCurrency int    `json:"hardCurrency" validate:"gte=0,lte=200"`
 }

Pitaya 默认的验证器是这个包:github.com/go-playgrou...

这个功能很强大啊,在游戏开发里,默认客户端是不可信的,需要对大部分上行参数做数据验证,自己手动写就很麻烦,在 pitaya 框架里,我们可以使用 StructValidation 来实现,优雅!

结构体验证的实现用到了我们接下来要说的钩子方法,开启验证,就是在 BeforeHandler 添加了验证方法:

go 复制代码
 // builder.go: 314
 func configureDefaultPipelines(handlerHooks *pipeline.HandlerHooks) {
     handlerHooks.BeforeHandler.PushBack(defaultpipelines.StructValidatorInstance.Validate)
 }

钩子方法 BeforeHandler AfterHandler

pitaya 提供了钩子方法集:

  • BeforeHandler 在消息处理方法前执行,结构体验证就是在这里调用的
  • AfterHandler 在消息处理方法后执行

这两种钩子方法集在 ProcessHandlerMessage 里被调用:

go 复制代码
 // handler_pool.go:43
 func (h *HandlerPool) ProcessHandlerMessage(
     ctx context.Context,
     rt *route.Route,
     serializer serialize.Serializer,
     handlerHooks *pipeline.HandlerHooks,
     session session.Session,
     data []byte,
     msgTypeIface interface{},
     remote bool,
 ) ([]byte, error) {
     ...
     ctx, arg, err = handlerHooks.BeforeHandler.ExecuteBeforePipeline(ctx, arg)
     if err != nil {
         return nil, err
     }
     ...
     resp, err := util.Pcall(handler.Method, args)
     ...
     resp, err = handlerHooks.AfterHandler.ExecuteAfterPipeline(ctx, resp, err)
     ...
 }

ExecuteBeforePipelineExecuteAfterPipeline 就是遍历切片,依次调用注册的方法

相关推荐
白露与泡影8 小时前
Spring Boot 完整流程
java·spring boot·后端
Mr.Rice.Fool9 小时前
rust面试经验1
后端·面试·职场和发展·rust
审判长烧鸡9 小时前
GO错误处理【5】显式错误处理
go·错误处理·报错链条
jeff聊企业数字化9 小时前
私有化即时通讯选型指南:兼顾安全与高效
go·业界资讯·即时通讯
北风toto10 小时前
Spring Boot / Spring Cloud 配置文件加密详解:使用 jasypt-spring-boot 实现 ENC() 加密
spring boot·后端·spring cloud
代码羊羊10 小时前
Rust 格式化输出完全攻略:从入门到精通
开发语言·后端·rust
Rust研习社10 小时前
Rust + PostgreSQL 极简技术栈应用开发
开发语言·数据库·后端·http·postgresql·rust
geovindu10 小时前
go:Template Method Pattern
开发语言·后端·设计模式·golang·模板方法模式
审判长烧鸡10 小时前
GO错误处理【6】显式哲学
go·显式哲学
白晨并不是很能熬夜10 小时前
【RPC】第 4 篇:服务发现 — Zookeeper + 缓存容错
java·后端·程序人生·缓存·zookeeper·rpc·服务发现