go web应用的部署和热更新

rod6 · · 13185 次点击 · 开始浏览    置顶
这是一个创建于 的主题,其中的信息可能已经有所发展或是发生改变。
### 序言 Golang标准库的http部分提供了强大的web应用支持,再加上negroni等中间件框架的支持,可以开发高性能的web应用(如提供Restful的api服务等)。 通常这些web应用部署在多台Linux操作系统的应用服务器上,并用Nginx等做为反向代理,实现高可用的集群服务。 当应用版本升级时,如何实现比较优雅的多态服务器的版本更新呢?我google、baidu了一下,也许我输入的关键字不对,也许这对很多人不是个问题,反正我是没有找到完整的方案。 ### 问题 Web应用的更新,我觉得可能需要考虑几个方面的问题: * 编译好的应用二进制文件、配置文件上传到服务器上; * 应用服务器能感知到有新的版本上传; * 在没有停止服务的情况下,热更新版本; * 最好所有的更新过程,可以脚本化,减少手动操作的错误。 ### 方案 其实,go社区有一些开源项目,可以自动检测web应用的改变,并实现自动的更新,但这些应用都是检测源码、资源文件的更新,启动build过程,实现自动的编译和重启,例如 [gin](https://github.com/codegangsta/gin)和 [fresh](https://github.com/pilu/fresh),个人感觉,这些应用适合应用于开发和测试阶段,并不适合应用的部署和更新,但提供了良好的思路。 #### 部署环境的目录及版本的上传 我将发布的应用二进制文件和配置文件,存放在某个目录下,如 ~/app/release,每个版本都保留在这个目录中,例如 app.1.0、app.1.1、app.2.0,一旦发现有问题,可以及时的回滚。 同时,在~/app目录下,利用软链接文件,指向到最新版本,如 ln -s ~/app/release/app.2.0 ~/app/app.bin。 此外,利用一个保存在 ~/app/release 下的文本文件,来指明当前应用的版本,如current.conf: ~~~ json { "bin.file": "~/app/release/app.2.0", "cfg.file": "~/app/release/cfg.2.0" } ~~~ 当需要更新服务器的版本时,可以通过脚本调用scp,将新版本上传到release目录下,然后更新current.conf文件。 #### 监控current.conf文件,获知版本更新 current.conf文件中是当前的版本,一旦这个文件发生变化,即表示有版本需要更新(或者回滚),我们只需要监控这个文件的变化,一旦发生变化,则做相应的处理。文件的监控,可以通过 [fsnotify](https://github.com/howeyc/fsnotify)来实现。 ~~~ go func watch() { watcher, err := fsnotify.NewWatcher() if err != nil { logger.Fatal(err) } defer watcher.Close() go func() { for { select { case event := <-watcher.Events: logger.Println("event:", event) if event.Op&fsnotify.Write == fsnotify.Write { logger.Println("notify runner to do the ln -s and restart server.") restartChan <- true } case err := <-watcher.Errors: logger.Println("error:", err) } } }() err = watcher.Add("/path/to/current.conf") if err != nil { logger.Fatal(err) } <- make(chan bool) } ~~~ #### 重启服务 监控到current.conf文件的变化后,接下来就是重启服务。 为了让服务不中断,优雅的进行重启,可以利用 [endless](https://github.com/fvbock/endless) 来替换标准库net/http的ListenAndServe: ~~~ go n := negroni.New() n.Use(middleware.NewRecovery()) n.Use(middleware.NewMaintainMiddleware()) n.Use(middleware.NewLogMiddleware()) n.Use(middleware.NewStatic(http.Dir("static"))) n.UseHandler(router.NewRouter()) log.Fatal(endless.ListenAndServe(":3000", n)) ~~~ 在current.conf变更后,首先将~/app下的软链接文件指向最新版本,然后利用kill -HUP通知应用重启。 ~~~ go func run() { for { <- restartChan c, err := ioutil.ReadFile("/path/to/current.conf") if err != nil { logger.Println("current.conf read error:", err) return } var j interface{} err = json.Unmarshal(c, &j) if err != nil { logger.Println("current.conf parse error:", err) return } parsed, ok := j.(map[string]interface{}) if !ok { logger.Println("current.conf parse error: mapping errors") return } exec.Command("rm", "app.bin").Run() exec.Command("ln", "-s", parsed["bin.file"].(string), "app.bin").Run() exec.Command("rm", "app.conf").Run() exec.Command("ln", "-s", parsed["cfg.file"].(string), "app.cfg").Run() if !started { cmd := exec.Command("./app.bin", "-c", "app.cfg") started = true } else { processes, _ := ps.Processes() for _, v := range processes { if strings.Contains(v.Executable(), parsed["bin.file"]) { process, _ := os.FindProcess(v.Pid()) process.Signal(syscall.SIGHUP) } } } } } ~~~ ### 最后 才疏学浅,只是把自己项目中一些心得与大家分享,如果有人有兴趣,我可以把代码再整理一下,开源出来。如果有更好的方案,不妨给我一些提示?

入群交流(和以上内容无关):Go中文网 QQ 交流群:729884609 或加微信入微信群:274768166 备注:入群;关注公众号:Go语言中文网

13185 次点击  ∙  2 赞  
加入收藏 微博
3 回复  |  直到 2017-01-16 10:14:54
暂无回复
添加一条新回复 (您需要 登录 后才能回复 没有账号 ?)
  • 请尽量让自己的回复能够对别人有帮助
  • 支持 Markdown 格式, **粗体**、~~删除线~~、`单行代码`
  • 支持 @ 本站用户;支持表情(输入 : 提示),见 Emoji cheat sheet
  • 图片支持拖拽、截图粘贴等方式上传