Kitex源码阅读——脚手架代码是如何通过命令行生成的(一)
前言
Kitex是字节跳动内部的Golang微服务RPC框架,先已开源。
Kitex文档:https://www.cloudwego.io/zh/docs/kitex/getting-started/
Kitex体验:https://juejin.cn/post/7098966260502921230
在Kitex体验的文章中,我们使用Kitex从零构建了自己的服务,只要定义好IDL
(接口描述语言),按照Kitex提供的命令行规则,就可以生成支持Thrift
、Protobuf
的客户端和服务端相关的脚手架代码,使得我们可以直接着手编写服务端的响应实现和客户端的请求发起逻辑。
那么Kitex究竟是怎么生成脚手架代码的?这系列文章将围绕此展开源码阅读,并最终解答这个疑问。
源码分析
初览kitex命令行工具
在最初安装或者更新Kitex的时候,用到了下面这条命令下载了Kitex可执行文件(用于脚手架生成):
go install github.com/cloudwego/kitex/tool/cmd/kitex
kitex
是一个可执行文件,因为go install
做了两件事(编译+安装),它将github.com/cloudwego/kitex/tool/cmd/kitex
目录下的main.go
及其依赖库编译成了一个可执行文件,再将其下载到本地的$GOPATH/bin
路径下。
换句话说,你完全可以通过下面这命令将整个kitex
依赖库全部下载下来:
go get github.com/cloudwego/kitex@latest
然后进入github.com/cloudwego/kitex/tool/cmd/kitex
目录去手动执行go build
命令,根据目录名(包名)将其编译成一个可执行文件kitex
,再将其移动到$GOPATH/bin
目录下,就能复现上面go install
的工作。
go build -o ~/go/bin/kitexx # 使用-o参数可以将编译的可执行文件指定位置和名称
比如我构建了一个功能强大的kitexx
工具!(可以在终端中调用,只是它还没有接受命令行参数的能力,别担心!随着源码的分析我们将会扩展kitexx的功能! )
先回归Kitex,go install
之后,我们在命令行中输入下面的命令就可以实现项目脚手架代码的生成:
kitex -module example -service example echo.thrift
kitex
就指代$GOPATH/bin
下的可执行文件kitex
,后面的-module xxx..
都是指定的命令行参数。
下面让我们看一下kitex负责脚手架代码生成的可执行文件编译前的代码:
# 使用tree命令查看$GOPATH/pkg/mod/github.com/cloudwego/kitex@v0.2.1/tool/cmd/kitex的目录结构,也就是这两个文件中编写了接受命令行参数、创建服务脚手架的核心代码
.
└── kitex
├── args.go
└── main.go
分析main.go的初始化函数
下面这是main.go
的init函数
,看到初始化过程就是args调用了addExtraFlag
方法,并且传入了一个extraFlag
。
那么我们来看一下extraFlag
的结构,通过首行注释得知,这个结构是用于添加与代码生成无关的flag
的(每一个flag可以理解成kitex工具命令行需要解析的参数,后面会讲)。
结构体有两个成员函数,第二个用于检查需要添加的flag的合法性,第一个用于添加flags到FlagSet
,FlagSet
出自于Go标准命令行解析库flag。
看到这你大致能猜测解析命令行的工作最终还是落到了Go标准库头上,只是kitex在此基础上定制了自己需要的功能。
这篇文章介绍了命令行解析库flag的使用:https://segmentfault.com/a/1190000021143456。
那么让我们看一下FlagSet结构
的源码(这里没放出来),注释描述FlagSet
是一个用于存放flags的集合,并介绍了Usage
的作用和触发条件。
这时候我们已经深入源码第三层了,先不急着深入,容易迷失方向。先回到最初init函数
中,我们已经知道apply
方法用于添加flag
到FlagSet
中,那么是如何添加的呢?
我们来看一下FlagSet
的BoolVar
方法源码,newBoolValue
的作用是将value
的值赋给p
然后返回(可以点进去看源码),BoolVar
方法的作用由注释得知,为了定义一个bool类型的flag(由name、value、usage定义)
然后调用了f.Var
方法,猜测是用于将这个定义的bool类型
的flag添加入FlagSet
集合,看一下源码。首先检测要添加的flag的name不能以-或者=开头,然后判断map中是否存在相同名称的flag,如果有则panic,然后按步添加flag到f.formal
中(map[string]*Flag
)
现在大致明白,init函数的作用就是调用了args.addExtraFlag
方法,添加了一个额外的不是用于代码生成的flag,至于check部分就是当遇到指定错误的时候需要终止程序。
为os.Exit()
指定状态码,0表示成功,1表示内部错误,2表示无效参数
到目前为止,init函数部分基本已经分析了一遍,你可能好奇,既然在初始化阶段已经为FlagSet添加了一个和version
相关flag
(其实并没有完成添加,这里先卖个关子!下面解释),那么FlagSet
本身是在哪里初始化的?
这里我们留意到init
函数上方的全局变量args
,并且留意到args.addExtraFlag
也是调用自args
,那么势必要看一下arguments
的源码,看看能否找到FlagSet
的初始化工作。
我们的目标是找到一个类似NewFlagSet的函数,那么就进入args.go使用command+f吧!果然找到了,而且只有一处!
那么再看这个buildFlags函数在哪被调用的,没办法,看来还得接着查一下parseArgs函数的调用情况。
终于你在main.go
的主函数里找到了,但是问题来了,main.go
文件的init()
初始化函数你分析了之后是给FlagSet
添加flag的,而且应该是先于主函数体执行的,那此时FlagSet还没初始化啊?这不是惊天大BUG? 事实上这就是上面我卖的那个关子
我们再看一下args.addExtraFlag
方法和args的结构体
,事实上,初始化的时候只是将extraFlag
创建了出来,加入了一个切片,真正为FlagSet
添加flag必然是等到flag.NewFlagSet
方法初始化FlagSet之后。
上面这个乌龙其实在阅读源码的时候很可能遇到,因为我们在没有全面的视角的情况下,往往很多问题的出现只是缺少对源码的熟悉,只有反复推敲,才能逐渐梳理清楚。
丰富kitexx框架的功能
事实上,main.go文件的init函数初始化只添加了一个flag,说明了这个flag的name、value还有usage,但是并没有涉及到自动化构建脚手架的工作,当然这部分我相信通过继续阅读main函数的其余部分可以得到解答。但是考虑到篇幅原因,我打算将其放在下一篇文章中。
先来丰富一下我们的kitexx框架,为其添加解析命令行的功能。(现阶段只是简单使用flag标准库的一些API,后续再作更多的解释)。
flag库也可以看这篇文章:https://segmentfault.com/a/1190000021143456
将上述代码手动编译到$GOPATH/bin
目录下,并且尝试通过命令行运行kitexx
,输入事先添加好的两个flag
(bool类型的flag后面可以不加参数),实现用我们输入的参数值替换b和s的默认值并打印。
小结
通过这篇文章,我们初步分析了kitex框架的脚手架代码生成工具的源代码的init函数。并且体验了一下实现自己的命令行解析框架kitexx。
在后续的文章中将继续分析main函数的剩余部分,并继续扩展kitexx框架的功能。
关注公众号【程序员白泽】,将同步文章更新。
Kitex源码阅读——脚手架代码是如何通过命令行生成的(一)的更多相关文章
- [Go] gocron源码阅读-通过第三方cli包实现命令行参数获取和管理
gocron源码中使用的是下面这个第三方包来实现的,下面就单独的拿出来测试以下效果,和官方flag包差不多 go get github.com/urfave/cli package main impo ...
- django源码(2.0.2)粗解之命令行执行
前言 django的命令行在整个的django web开发中都会经常用到,而且是必须得用到.所以,能够了解下django的命令行实现其实是非常有帮助的. 如果大家比较关心django命令的详细说明和使 ...
- 【原】FMDB源码阅读(三)
[原]FMDB源码阅读(三) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 FMDB比较优秀的地方就在于对多线程的处理.所以这一篇主要是研究FMDB的多线程处理的实现.而 ...
- 【原】FMDB源码阅读(二)
[原]FMDB源码阅读(二) 本文转载请注明出处 -- polobymulberry-博客园 1. 前言 上一篇只是简单地过了一下FMDB一个简单例子的基本流程,并没有涉及到FMDB的所有方方面面,比 ...
- 【原】FMDB源码阅读(一)
[原]FMDB源码阅读(一) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 说实话,之前的SDWebImage和AFNetworking这两个组件我还是使用过的,但是对于 ...
- 【原】AFNetworking源码阅读(六)
[原]AFNetworking源码阅读(六) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 这一篇的想讲的,一个就是分析一下AFSecurityPolicy文件,看看AF ...
- 【原】AFNetworking源码阅读(五)
[原]AFNetworking源码阅读(五) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 上一篇中提及到了Multipart Request的构建方法- [AFHTTP ...
- 【原】AFNetworking源码阅读(四)
[原]AFNetworking源码阅读(四) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 上一篇还遗留了很多问题,包括AFURLSessionManagerTaskDe ...
- 【原】AFNetworking源码阅读(三)
[原]AFNetworking源码阅读(三) 本文转载请注明出处 —— polobymulberry-博客园 1. 前言 上一篇的话,主要是讲了如何通过构建一个request来生成一个data tas ...
随机推荐
- BFC理解
Block formatting context (块级格式化上下文) 页面文档由块block构成 每个block在页面上占据自己的位置 使用新的元素构建BFC overflow:hidden | a ...
- Python 图_系列之基于邻接炬阵实现广度、深度优先路径搜索算法
图是一种抽象数据结构,本质和树结构是一样的. 图与树相比较,图具有封闭性,可以把树结构看成是图结构的前生.在树结构中,如果把兄弟节点之间或子节点之间横向连接,便构建成一个图. 树适合描述从上向下的一对 ...
- 在Nginx或Tengine服务器上安装证书
阿里云SSL证书服务支持下载证书并安装到Nginx.Tengine服务器上,本文介绍了证书安装的具体操作. 前提条件 已准备远程登录工具,例如PuTTY或者Xshell. 背景信息 本文档以CentO ...
- 如何得到个性化banner
介绍 有时候用一些脚本工具,会有一些由其他字符组成的字符.(如下面这个我还在写的) 使用 kali自带了这个工具 -- figlet. figlet AuToIP 就可以得到上面的字符啦! 另外如果想 ...
- LC-数组-二分查找-704
二分查找 [left, right] 方式 [left, mid -1] [mid + 1, right] int left = 0, right = nums.length - 1; while ( ...
- eclipse 下 SpringBoot 工程使用Maven打包
eclipse 下 SpringBoot 工程使用Maven打包 1. pom.xml 添加打包配置 点击查看代码 <!-- 打包使用 --> <build> <plug ...
- 面渣逆袭:RocketMQ二十三问
基础 1.为什么要使用消息队列呢? 消息队列主要有三大用途,我们拿一个电商系统的下单举例: 解耦:引入消息队列之前,下单完成之后,需要订单服务去调用库存服务减库存,调用营销服务加营销数据--引入消息队 ...
- event 事件对象
关于event对象 在触发的事件的函数里面我们会接收到一个event对象,通过该对象我们可以得到需要的一些参数,比如说我们需要知道此事件作用到谁身上了,就可以通过event的属性target来获取到( ...
- 不仅仅是一把瑞士军刀 —— Apifox的野望和不足
声明:本文内容不涉及任何 Apifox 的功能介绍,一来网上这方面的文章已经汗牛充栋,二来 Apifox 本身的用户体验做的非常好,对于开发者而言学习成本基本为零. 阮一峰:不管你是前端开发还是后端开 ...
- Linux内核--链表结构(一)
一.前言 Linux内核链表结构是一种双向循环链表结构,与传统的链表结构不同,Linux内核链表结构仅包含前驱和后继指针,不包含数据域.使用链表结构,仅需在结构体成员中包含list_head*成员就行 ...