用 go 写了个查看 html 文档的命令行工具

dart 和 rust 的语言都有文档生成功能,可以从注释中生成 html 格式的文档。但是很不巧的是,它们都没有提供 http 访问的工具。本地开发时只能对着 index.html 干瞪眼。虽说借助 docker 也能启动一个文档服务器,比如 docker run -it --rm -p 80:80 -v ./doc/api/:/usr/share/nginx/html nginx 。但敲这么长的命令确实有点考验记忆力。所以我写个脚本美滋滋 开发了个轮子,用于查看 html 文档(文档链接)。

其实 go 写这个超简单的,下面这行代码一行就能搞定。

go 复制代码
_ = http.ListenAndServe(":9090", http.FileServer(http.Dir("doc/api")))

但要将其写成工具的话,考虑的事情就多了。返回的 error 要处理;端口和本地目录要写成活的;要打印实际监听的端口(传入 ":0" 代表随机选择端口监听所以不能直接打印传入的参数)......最后就写成了这样。

go 复制代码
var (
	addr string // serve address
	dir  string // root path
)

func init() {
	flag.StringVar(&addr, "addr", "localhost:9090", "serve address")
	flag.StringVar(&dir, "dir", ".", "root path")
}

func main() {
	flag.Parse() // 标准库 flag 包自带 -h 和 --help 适配,不用单独写。

	l, err := net.Listen("tcp", addr)
	if err != nil {
		log.Fatal(err)
	}
	defer l.Close()

	ctx := context.Background()
	ctx, stop := signal.NotifyContext(ctx, os.Interrupt)
	defer stop()

	s := http.Server{
		Handler: http.FileServer(http.Dir(dir)),
	}

	var wg sync.WaitGroup // 虽然有点画蛇添足,但加了个 kill 适配。这种写法可以作为其他项目的参考。
	wg.Add(1)
	go func() {
		defer wg.Done()
    log.Printf("Serve at http://%s\n", l.Addr()) // 打印的是实际监听的端口,避免 ":0" 的问题
		err = s.Serve(l)
	}()

	wg.Add(1)
	go func() {
		defer wg.Done()
		<-ctx.Done()
		log.Println("Closing server...")
		s.Close()
	}()

	wg.Wait()
	if !errors.Is(err, http.ErrServerClosed) {
		log.Fatal(err)
	}
}

有了这个工具后就能在本地启动工具查看文档了,以上面的 dart 工程为例:

sh 复制代码
# 克隆工程,生成文档
git clone https://github.com/kvii/kratos-plugin.git
cd kratos-plugin
dart doc

# 下载工具,启动服务
go install github.com/kvii/doc@latest
doc

# 浏览器打开 http://127.0.0.1:9090 即可访问文档

其实各个系统都能直接从命令行打开浏览器页面,比如 mac 就是 open "http://127.0.0.1:9090"。用 exec 包可以直接执行命令行命令,再适配下系统就能实现服务启动后自动打开浏览器的功能了。没加这功能的原因有二:

  1. 有的环境里可能没有浏览器,比如 linux 服务器和 devcontainer(开发容器)。
  2. 笔者开发用 vscode,它的终端对 url 有适配。ctrl + 鼠标点击就能直接浏览器里打开终端打印的 url。
相关推荐
IUGEI11 小时前
【MySQL】SQL慢查询如何排查?从慢查询排查到最终优化完整流程
java·数据库·后端·mysql·go
Daydreamer16 小时前
Trpc配置插件
go
诗意地回家16 小时前
niuhe.conf 配置文件说明
vscode·go
yagamiraito_1 天前
757. 设置交集大小至少为2 (leetcode每日一题)
算法·leetcode·go
Code_Artist2 天前
robfig/cron定时任务库快速入门
分布式·后端·go
川白2 天前
用 Go 写多线程粒子动画:踩坑终端显示与跨平台编译【含 Windows Terminal 配置 + Go 交叉编译脚本】
go
zhuyasen3 天前
Go 实战:在 Gin 基础上上构建一个生产级的动态反向代理
nginx·go·gin
Tsblns3 天前
从Go http.HandleFunc()函数 引出"函数到接口"的适配思想
go
Schuyler20253 天前
年轻人的第一个 GO 桌面应用:用 Wails 做个学习搭子计时器
go
狼爷4 天前
Go 重试机制终极指南:基于 go-retry 打造可靠容错系统
架构·go