rest框架概览

我们先通过 go-zero 自带的命令行工具 goctl 来生成一个 api service,其 main 函数如下:

func main() {
flag.Parse() var c config.Config
conf.MustLoad(*configFile, &c) ctx := svc.NewServiceContext(c)
server := rest.MustNewServer(c.RestConf)
defer server.Stop() handler.RegisterHandlers(server, ctx) fmt.Printf("Starting server at %s:%d...\n", c.Host, c.Port)
server.Start()
}
  1. 解析配置文件
  2. 将配置文件传入,初始化 serviceContext
  3. 初始化 rest server
  4. context 注入 server 中:
    1. 注册路由
    2. context 中的启动的 endpoint 同时注入到 router 当中
  5. 启动 server

接下来我们来一步步讲解其设计原理!Let's Go!

web框架

从日常开发经验来说,一个好的 web 框架大致需要满足以下特性:

  1. 路由匹配/多路由支持
  2. 支持自定义中间件
  3. 框架和业务开发完全解耦,方便开发者快速开发
  4. 参数校验/匹配
  5. 监控/日志/指标等服务自查功能
  6. 服务自保护(熔断/限流)

go-zero rest设计

https://github.com/zeromicro/go-zero/tree/master/rest

概览

  1. 借助 context (不同于 gin 的 context),将资源初始化好 → 保存在 serviveCtx 中,在 handler 中共享(至于资源池化,交给资源自己处理,serviveCtx 只是入口和共享点)
  2. 独立 router 声明文件,同时加入 router group 的概念,方便开发者整理代码结构
  3. 内置若干中间件:监控/熔断/鉴权等
  4. 利用 goctl codegen + option 设计模式,方便开发者自己控制部分中间件的接入

上图描述了 rest 处理请求的模式和大部分处理路径。

  1. 框架内置的中间件已经帮开发者解决了大部分服务自处理的逻辑
  2. 同时 go-zero 在 business logic 处也给予开发者开箱即用的组件(dq、fx 等)
  3. 从开发模式上帮助开发者只需要关注自己的 business logic 以及所需资源准备

下面我们来细说一下整个 rest 是如何启动的?

启动流程

上图描述了整体 server 启动经过的模块和大致流程。准备按照如下流程分析 rest 实现:

  1. 基于 http.server 封装以及改造:把 engine(web框架核心) 和 option 隔离开
  2. 多路由匹配采取 radix-tree 构造
  3. 中间件采用洋葱模型 → []Middleware
  4. http parse 解析以及匹配校验 → httpx.Parse()
  5. 在请求过程会收集指标 (createMetrics()) 以及监控埋点 (prometheus)

server engine封装

点开大图观看

engine 贯穿整个 server 生命周期中:

  1. router 会携带开发者定义的 path/handler,会在最后的 router.handle() 执行
  2. 注册的自定义中间件 + 框架中间件,在 router handler logic 前执行

在这里:go-zero 处理的粒度在 route 上,封装和处理都在 route 一层层执行

路由匹配

那么当 request 到来,首先是如何到路由这一层的?

首先在开发最原始的 http server ,都有这么一段代码:

type helloHandler struct{}

func (h *helloHandler) ServeHTTP(w http.ResponseWriter, r *http.Request) {
w.Write([]byte("Hello, world!"))
} func main() {
http.Handle("/", &helloHandler{})
http.ListenAndServe(":12345", nil)
}

http.ListenAndServe() 内部会执行到:server.ListenAndServe()

我们看看在 rest 里面是怎么运用的:

而传入的 handler 其实就是:router.NewRouter() 生成的 router。这个 router 承载了整个 server 的处理函数集合。

同时 http.Server 结构在初始化时,是把 handler 注入到里面的:

type Server struct {
...
Handler Handler
} func start(..., handler http.Handler, run func(srv *http.Server) error) (err error) {
server := &http.Server{
Addr: fmt.Sprintf("%s:%d", host, port),
Handler: handler,
}
...
return run(server)
}

在 http.Server 接收 req 后,最终执行的也是:handler.ServeHTTP(rw, req)

所以内置的 router 也需要实现 ServeHTTP 。至于 router 自己是怎么实现 ServeHTTP :无外乎就是寻找匹配路由,然后执行路由对应的 handle logic。

解析参数

解析参数是 http 框架需要提供的基本能力。在 goctl code gen 生成的代码中,handler 层已经集成了 req argument parse 函数:

// generate by goctl
func QueryAllTaskHandler(ctx *svc.ServiceContext) http.HandlerFunc {
return func(w http.ResponseWriter, r *http.Request) {
// custom request in .api file
var req types.QueryAllTaskRequest
// parse http request
if err := httpx.Parse(r, &req); err != nil {
httpx.Error(w, err)
return
} l := logic.NewEventLogic(r.Context(), ctx)
resp, err := l.QueryAllTask(req)
baseresponse.FormatResponseWithRequest(resp, err, w, r)
}
}

进入到 httpx.Parse() ,主要解析以下几块:

https://github.com/zeromicro/go-zero/blob/master/rest/httpx/requests.go#L32:6

  1. 解析path
  2. 解析form表单
  3. 解析http header
  4. 解析json

Parse() 中的 参数校验 的功能见:

https://go-zero.dev/cn/api-grammar.html 中的 tag修饰符

Tips

学习源码推荐 fork 出来边看边写注释和心得,可以加深理解,以后用到这块功能的时候也可以回头翻阅。

项目地址

https://github.com/zeromicro/go-zero

欢迎使用 go-zerostar 支持我们!

多图详解万星 Restful 框架原理与实现的更多相关文章

  1. SPI总线协议及SPI时序图详解

    SPI,是英语Serial Peripheral Interface的缩写,顾名思义就是串行外围设备接口.SPI,是一种高速的,全双工,同步的通信总线,并且在芯片的管脚上只占用四根线,节约了芯片的管脚 ...

  2. (转)CAS (4) —— CAS浏览器SSO访问顺序图详解(CAS Web Flow Diagram by Example)

    CAS (4) —— CAS浏览器SSO访问顺序图详解(CAS Web Flow Diagram by Example) tomcat版本: tomcat-8.0.29 jdk版本: jdk1.8.0 ...

  3. SPI总线协议及SPI时序图详解【转】

    转自:https://www.cnblogs.com/adylee/p/5399742.html SPI,是英语Serial Peripheral Interface的缩写,顾名思义就是串行外围设备接 ...

  4. 十图详解tensorflow数据读取机制(附代码)转知乎

    十图详解tensorflow数据读取机制(附代码) - 何之源的文章 - 知乎 https://zhuanlan.zhihu.com/p/27238630

  5. CAS (6) —— Nginx代理模式下浏览器访问CAS服务器网络顺序图详解

    CAS (6) -- Nginx代理模式下浏览器访问CAS服务器网络顺序图详解 tomcat版本: tomcat-8.0.29 jdk版本: jdk1.8.0_65 nginx版本: nginx-1. ...

  6. CAS (4) —— CAS浏览器SSO访问顺序图详解(CAS Web Flow Diagram by Example)

    CAS (4) -- CAS浏览器SSO访问顺序图详解(CAS Web Flow Diagram by Example) tomcat版本: tomcat-8.0.29 jdk版本: jdk1.8.0 ...

  7. UML类图详解_关联关系_一对多

    对于一对多的示例,可以想象一个账户可以多次申购.在申购的时候没有固定上限,下限为0,那么就可以使用容器类(container class)来搞,最常见的就是vector了. 下面我们来看一个“一对多” ...

  8. UML类图详解_关联关系_多对一

    首先先来明确一个概念,即多重性.什么是多重性呢?多重性是指两个对象之间的链接数目,表示法是“下限...上限”,最小数据为零(0),最大数目为没有设限(*),如果仅标示一个数目级上下限相同. 实际在UM ...

  9. UML简单介绍—类图详解

    类图详解 阅读本文前请先阅读:UML简单介绍—类图这么看就懂了 1.泛化关系 一个动物类: /** * 动物类 */ public class Animal { public String name; ...

随机推荐

  1. ES6 promise的应用

    html部分 <!DOCTYPE html> <html> <head> <meta charset="utf-8"> <me ...

  2. 存储系统管理(三)——磁盘配额及lvm逻辑卷管理

    Linux是一个多用户的操作系统,系统有很多用户,就必须限制每个用户的保存空间,配额就是来管理用户空间的,配额只是针对与设备而言. 1.新建一个分区 2.格式化分区为xfs文件系统 3.将其以配额的方 ...

  3. MyBatis学习总结(六)——Mybatis3.x与Spring4.x整合

    一.搭建开发环境 1.1.使用Maven创建Web项目 执行如下命令: mvn archetype:create -DgroupId=me.gacl -DartifactId=spring4-myba ...

  4. Linux下cat命令的使用

    1.普通用法-->查看文件内容 cat file_name 查看文件时的相关参数: 1.cat f1.txt,查看f1.txt文件的内容. 2.cat -n f1.txt,查看f1.txt文件的 ...

  5. leetcode数组典型题目小结

    数组与矩阵 数组与矩阵的基本知识: 1.数组:数组是在程序设计中,为了处理方便, 把具有相同类型的若干元素按有序的形式组织起来的一种形式. 首先,数组会利用索引来记录每个元素在数组中的位置,且在大多数 ...

  6. JAVA反序列化的简单探究

    JAVA反序列化的简单探究 本文主要是探究,在反序列化过程中是怎么调用到readObject.readResolve.readExternal方法的问题 新建一个需要被序列化的类ObjectA,写入r ...

  7. C# 反射 + Quartz,实现流程处理

    场景: 前不久,公司里项目经理要求我实现流程处理,比如,用户可以定义一个定时任务,每周一查看报表.定时任务很简单,用Quartz可以实现,但是用户自己选择报表就比较麻烦,因为系统的不同模块的生成报表的 ...

  8. [源码解析] 深度学习分布式训练框架 horovod (20) --- Elastic Training Operator

    [源码解析] 深度学习分布式训练框架 horovod (20) --- Elastic Training Operator 目录 [源码解析] 深度学习分布式训练框架 horovod (20) --- ...

  9. Eclipse中快速生成Javabean的方法

    总结一下: 先写出属性 无参构造器:Alt+/ 再按回车 全参构造器:Alt+Shift+S 再按字母O键 再按回车 toString方法:Alt+Shift+S 再按字母S键 再按回车 get/se ...

  10. HCNP Routing&Switching之路由过滤工具Filter-Policy

    前文我们了解了路由控制技术中路由策略和路由匹配工具IP-Prefix相关话题,回顾请参考https://www.cnblogs.com/qiuhom-1874/p/15314262.html:今天我们 ...