概要
基于 golang Gin 框架开发 web 服务时, 需要时不时的 go build , 然后重启服务查看运行结果.
go build 的过程集成在编辑器中(emacs), 可以通过快捷键迅速完成, 但是每次重启服务都切换到命令行中操作.
因此, 希望能够编译通过之后自动重启服务.
这里并不是部署阶段的服务重启, 所以不用过多考虑是否正常退出其中的协程.
实现方式
在开源的 illuminant 项目中, 已经将相应的代码集成到 gin 的 debug mode 中.
代码文件: https://gitee.com/wangyubin/illuminant/blob/dev/server_cmd.go
1 func setupWatcher() (chan struct{}, error) { 2 file, err := osext.Executable() 3 if err != nil { 4 return nil, err 5 } 6 log.Printf("watching %q\n", file) 7 w, err := fsnotify.NewWatcher() 8 if err != nil { 9 return nil, err10 }11 done := make(chan struct{})12 go func() {13 select {14 case e := <-w.Events:15 log.Printf("watcher received: %+v", e)16 err := syscall.Exec(file, os.Args, os.Environ())17 if err != nil {18 log.Fatal(err)19 }20 case err := <-w.Errors:21 log.Printf("watcher error: %+v", err)22 case <-done:23 log.Print("watcher shutting down")24 return25 }26 }()27 err = w.Add(file)28 if err != nil {29 return nil, err30 }31 return done, nil32 }
在 gin debug mode 下, 使用此方法自动重启服务
1 if c.Bool("prod") { 2 gin.SetMode(gin.ReleaseMode) 3 // start route 4 return routes.Routes(cnf.Server.Port) 5 } else { 6 gin.SetMode(gin.DebugMode) 7 watcher, err := setupWatcher() 8 if err != nil { 9 // do something sensible10 log.Fatal(err)11 }12 defer close(watcher)13 return routes.Routes(cnf.Server.Port)14 }
补充
上面函数的核心有以下两点:
- w, err := fsnotify.NewWatcher(): 创建监控文件变化的 watcher, err = w.Add(file) 并将当前二进制文件加入到监控文件列表中
- err := syscall.Exec(file, os.Args, os.Environ()) 接受到文件变化的事件时, 重新调用一次自己, 使用上次一样的参数和环境变量
syscall.Exec
对于这个函数, 一般可能用的比较少, 这里稍微介绍下. 它有 3 个参数:
- args[0]: 可执行文件的路径(相对路径, 绝对路径或者 PATH 中的路径都可以)
- args[1]: 命令的参数
- args[2]: 命令的执行的环境变量, os.Environ() 表示继承 caller 的环境变量
当 syscall.Exec 执行时, 在它之前的所有未执行完的程序都会被中止(包括在 go routine 中执行的程序),
然后执行 syscall.Exec 调用的命令, 该命令还保持在之前程序的 PID 下执行.
syscall.Exec 是最后一条执行的代码, 重启时在它之后可以有代码, 但是都不会被执行到, 包括 defer 中的代码.
下面是个小例子(通过这个例子可以验证上面的结论):
1 package main 2 3 import ( 4 "fmt" 5 "log" 6 "os" 7 "syscall" 8 "time" 9 10 "github.com/fsnotify/fsnotify"11 "github.com/kardianos/osext"12 )13 14 func syscallExec() {15 watcher, err := setupWatcher()16 if err != nil {17 log.Fatal(err)18 }19 defer finally(watcher)20 21 fmt.Printf("current pid: %d\n", os.Getpid())22 var count = 023 24 go func(count int) {25 for {26 fmt.Printf(">>> count in GO ROUTINE: %d\n", count)27 count++28 time.Sleep(1 * time.Second)29 }30 }(count)31 32 for {33 fmt.Printf(">>> count in MAIN: %d\n", count)34 count++35 time.Sleep(1 * time.Second)36 }37 }38 39 func finally(watcher chan struct{}) {40 // 重启时没有执行此函数41 fmt.Println("exit original exec")42 close(watcher)43 }44 45 func setupWatcher() (chan struct{}, error) {46 file, err := osext.Executable()47 if err != nil {48 return nil, err49 }50 log.Printf("watching %q\n", file)51 w, err := fsnotify.NewWatcher()52 if err != nil {53 return nil, err54 }55 done := make(chan struct{})56 go func() {57 select {58 case e := <-w.Events:59 log.Printf("watcher received: %v", e)60 err := syscall.Exec(file, os.Args, os.Environ())61 if err != nil {62 log.Fatal(err)63 }64 case err := <-w.Errors:65 log.Printf("watcher error: %+v", err)66 case <-done:67 log.Print("watcher shutting down")68 return69 }70 }()71 err = w.Add(file)72 if err != nil {73 return nil, err74 }75 return done, nil76 }