前往小程序,Get更优阅读体验!
立即前往
首页
学习
活动
专区
工具
TVP
发布
社区首页 >专栏 >静态分析Golang语言生成函数调用关系的利器——go-callvis

静态分析Golang语言生成函数调用关系的利器——go-callvis

作者头像
方亮
发布2024-03-19 08:24:56
810
发布2024-03-19 08:24:56
举报
文章被收录于专栏:方亮方亮

不同于之前分析C语言项目的工具,go-callvis还是很方便使用。只要把两项工作做好就能顺利的使用。

  1. go使用1.19及以上版本。
  2. 要在被分析工程的根目录执行分析指令。

我的测试环境是Ubuntu 22 TLS版,默认的Golang是1.18。这会导致go-callvis安装失败。如果版本匹配,可以忽略下面Golang升级的步骤。

升级go

删除旧版本

代码语言:javascript
复制
sudo apt remove golang-go golang-1.18-doc golang-1.18-go golang-1.18-src

安装新版本

直接上1.21版本。

代码语言:javascript
复制
sudo apt install golang-1.21

配置环境变量

代码语言:javascript
复制
sudo vim /etc/profile

在文件末尾另起新行填入以下内容

代码语言:javascript
复制
export GOROOT=/usr/lib/go-1.21
export GOPATH=$HOME/gowork
export GOBIN=$GOPATH/bin
export PATH=$GOPATH:$GOBIN:$GOROOT/bin:$PATH

载入环境

修改当前环境
代码语言:javascript
复制
source /etc/profile
修改之后进入的环境
代码语言:javascript
复制
sudo vim ~/.bashrc

在文件末尾另起一行新增

代码语言:javascript
复制
source /etc/profile

分析

我们以gorush的源码为例。它是一套基于Gin实现的消息推送框架。

代码语言:javascript
复制
https://github.com/appleboy/gorush.git

安装go-callvis

进入待分析工程目录

代码语言:javascript
复制
cd gorush

然后安装

代码语言:javascript
复制
go install github.com/ofabry/go-callvis@master

最后执行分析命令

代码语言:javascript
复制
go-callvis ./

2024/01/28 07:55:35 http serving at http://localhost:7878 2024/01/28 07:55:35 OpenURL error: exec: “xdg-open,x-www-browser,www-browser”: executable file not found in $PATH

这样我们就在浏览器中打开http://localhost:7878看到分析结果

go-callvis默认是寻找main包名,并以此为入口分析代码调用关系的。而且每个方法是以包名聚合的。如上图中,nats就是一个包名,其下的方法都被聚合在一起。这个特性由-group指定,默认是pkg(包),还可以是type。

分析其他包

如果我们希望分析其他包名,则需要使用-focus指定,而且路径要制定到包所在的目录(注意不是根目录),并且还要添加-algo static。如果路径不对,go-callvis会找不到对应的包(不会遍历子目录)。比如我们需要查看router包,它实际位于~/gorush/router目录下。如果执行

代码语言:javascript
复制
go-callvis -focus router -algo static ./

则会报错

focus failed, could not find package: router

正确的指令是

代码语言:javascript
复制
 go-callvis -algo static -focus router ./router/
总结
代码语言:javascript
复制
 go-callvis -algo static -focus 【package_name】 ./【package_path】

导出文件

上述方法都会启动一个服务器,然后在浏览器中查看。如果网络环境不允许,则我们可以让其产出图片。-format用于指定文件格式;-file用于指定导出的文件名。

代码语言:javascript
复制
 go-callvis -algo static \
  -focus router \
  -format=png \
  -file=gorush_router \
  ./router/

2024/01/28 08:14:51 writing dot output… 2024/01/28 08:14:51 converting dot to png…

如果觉得这个图太长,则可以使用-rankdir转变成其他绘制方向([LR | RL | TB | BT] (default “LR”))。

代码语言:javascript
复制
go-callvis -algo static \
 -focus router \
 -format=png \
 -file=gorush_router_tb \
 -rankdir TB \
 ./router/
总结
代码语言:javascript
复制
 go-callvis -algo static \
  -focus 【package_name】\
  -format=【[svg | png | jpg | ...] (default "svg")】 \
  -file=【file_name】\
  ./【package_path】

清晰主体脉络

上面的图片还是很大,提供了大量信息。但是大量的信息也会带来一些干扰,我们可以通过增加一些参数,忽略一些不太重要的信息,来让调用框架清晰起来。

-nointer Omit calls to unexported functions. -nostd Omit calls to/from packages in standard library.

-nostd表示标准库的包就不用展现了,这个可以省去大量最后一层的调用关系;-nointer则表示忽略非导出方法,这样我们就关注于包之间调用关系即可。

代码语言:javascript
复制
go-callvis -algo static \
 -focus router \
 -format=png \
 -file=gorush_router_noiter_nostd \
 -nointer -nostd \
 ./router/
总结
代码语言:javascript
复制
 go-callvis -algo static \
  -focus 【package_name】\
  -format=【[svg | png | jpg | ...] (default "svg")】 \
  -file=【file_name】\
  -nointer -nostd \
  ./【package_path】

其他

我还对gin的源码做了测试

代码语言:javascript
复制
git clone https://github.com/gin-gonic/gin.git
cd gin
go install github.com/ofabry/go-callvis@master
go-callvis -algo static -focus gin -format=png -file=gin -nointer -nostd ./

参考资料

本文参与 腾讯云自媒体分享计划,分享自作者个人站点/博客。
原始发表:2024-01-28,如有侵权请联系 cloudcommunity@tencent.com 删除

本文分享自 作者个人站点/博客 前往查看

如有侵权,请联系 cloudcommunity@tencent.com 删除。

本文参与 腾讯云自媒体分享计划  ,欢迎热爱写作的你一起参与!

评论
登录后参与评论
0 条评论
热度
最新
推荐阅读
目录
  • 升级go
    • 删除旧版本
      • 安装新版本
        • 配置环境变量
          • 载入环境
            • 修改当前环境
            • 修改之后进入的环境
        • 分析
          • 安装go-callvis
            • 分析其他包
              • 总结
            • 导出文件
              • 总结
            • 清晰主体脉络
              • 总结
            • 其他
            • 参考资料
            领券
            问题归档专栏文章快讯文章归档关键词归档开发者手册归档开发者手册 Section 归档