golang API開發(fā)過程的中的自動(dòng)重啟方式(基于gin框架)
概要
基于 golang Gin 框架開發(fā) web 服務(wù)時(shí), 需要時(shí)不時(shí)的 go build , 然后重啟服務(wù)查看運(yùn)行結(jié)果.
go build 的過程集成在編輯器中(emacs), 可以通過快捷鍵迅速完成, 但是每次重啟服務(wù)都切換到命令行中操作.
因此, 希望能夠編譯通過之后自動(dòng)重啟服務(wù).
這里并不是部署階段的服務(wù)重啟, 所以不用過多考慮是否正常退出其中的協(xié)程.
實(shí)現(xiàn)方式
在開源的 illuminant 項(xiàng)目中, 已經(jīng)將相應(yīng)的代碼集成到 gin 的 debug mode 中.
代碼文件: https://gitee.com/wangyubin/illuminant/blob/dev/server_cmd.go
func setupWatcher() (chan struct{}, error) { file, err := osext.Executable() if err != nil { return nil, err } log.Printf("watching %q\n", file) w, err := fsnotify.NewWatcher() if err != nil { return nil, err } done := make(chan struct{}) go func() { select { case e := <-w.Events: log.Printf("watcher received: %+v", e) err := syscall.Exec(file, os.Args, os.Environ()) if err != nil { log.Fatal(err) } case err := <-w.Errors: log.Printf("watcher error: %+v", err) case <-done: log.Print("watcher shutting down") return } }() err = w.Add(file) if err != nil { return nil, err } return done, nil }
在 gin debug mode 下, 使用此方法自動(dòng)重啟服務(wù)
if c.Bool("prod") { gin.SetMode(gin.ReleaseMode) // start route return routes.Routes(cnf.Server.Port) } else { gin.SetMode(gin.DebugMode) watcher, err := setupWatcher() if err != nil { // do something sensible log.Fatal(err) } defer close(watcher) return routes.Routes(cnf.Server.Port) }
補(bǔ)充
上面函數(shù)的核心有以下兩點(diǎn):
- w, err := fsnotify.NewWatcher(): 創(chuàng)建監(jiān)控文件變化的 watcher, err = w.Add(file) 并將當(dāng)前二進(jìn)制文件加入到監(jiān)控文件列表中
- err := syscall.Exec(file, os.Args, os.Environ()) 接受到文件變化的事件時(shí), 重新調(diào)用一次自己, 使用上次一樣的參數(shù)和環(huán)境變量
syscall.Exec
對(duì)于這個(gè)函數(shù), 一般可能用的比較少, 這里稍微介紹下. 它有 3 個(gè)參數(shù):
- args[0]: 可執(zhí)行文件的路徑(相對(duì)路徑, 絕對(duì)路徑或者 PATH 中的路徑都可以)
- args[1]: 命令的參數(shù)
- args[2]: 命令的執(zhí)行的環(huán)境變量, os.Environ() 表示繼承 caller 的環(huán)境變量
當(dāng) syscall.Exec 執(zhí)行時(shí), 在它之前的所有未執(zhí)行完的程序都會(huì)被中止(包括在 go routine 中執(zhí)行的程序),
然后執(zhí)行 syscall.Exec 調(diào)用的命令, 該命令還保持在之前程序的 PID 下執(zhí)行.
syscall.Exec 是最后一條執(zhí)行的代碼, 重啟時(shí)在它之后可以有代碼, 但是都不會(huì)被執(zhí)行到, 包括 defer 中的代碼.
下面是個(gè)小例子(通過這個(gè)例子可以驗(yàn)證上面的結(jié)論):
package main import ( "fmt" "log" "os" "syscall" "time" "github.com/fsnotify/fsnotify" "github.com/kardianos/osext" ) func syscallExec() { watcher, err := setupWatcher() if err != nil { log.Fatal(err) } defer finally(watcher) fmt.Printf("current pid: %d\n", os.Getpid()) var count = 0 go func(count int) { for { fmt.Printf(">>> count in GO ROUTINE: %d\n", count) count++ time.Sleep(1 * time.Second) } }(count) for { fmt.Printf(">>> count in MAIN: %d\n", count) count++ time.Sleep(1 * time.Second) } } func finally(watcher chan struct{}) { // 重啟時(shí)沒有執(zhí)行此函數(shù) fmt.Println("exit original exec") close(watcher) } func setupWatcher() (chan struct{}, error) { file, err := osext.Executable() if err != nil { return nil, err } log.Printf("watching %q\n", file) w, err := fsnotify.NewWatcher() if err != nil { return nil, err } done := make(chan struct{}) go func() { select { case e := <-w.Events: log.Printf("watcher received: %v", e) err := syscall.Exec(file, os.Args, os.Environ()) if err != nil { log.Fatal(err) } case err := <-w.Errors: log.Printf("watcher error: %+v", err) case <-done: log.Print("watcher shutting down") return } }() err = w.Add(file) if err != nil { return nil, err } return done, nil }
到此這篇關(guān)于golang API開發(fā)過程的中的自動(dòng)重啟方式(基于gin框架)的文章就介紹到這了,更多相關(guān)golang API 自動(dòng)重啟內(nèi)容請(qǐng)搜索腳本之家以前的文章或繼續(xù)瀏覽下面的相關(guān)文章希望大家以后多多支持腳本之家!
相關(guān)文章
GoLang中的iface?和?eface?的區(qū)別解析
iface 和 eface 都是 Go 中描述接口的底層結(jié)構(gòu)體,區(qū)別在于 iface 描述的接口包含方法,而 eface 則是不包含任何方法的空接口:interface{},這篇文章主要介紹了GoLang之iface?和?eface?的區(qū)別,需要的朋友可以參考下2022-09-09Go語(yǔ)言基礎(chǔ)知識(shí)點(diǎn)介紹
在本篇文章里小編給大家整理的是一篇關(guān)于Go語(yǔ)言基礎(chǔ)知識(shí)點(diǎn)介紹內(nèi)容,有興趣的朋友們可以跟著學(xué)習(xí)參考下。2021-07-07go?doudou開發(fā)單體RESTful服務(wù)快速上手教程
這篇文章主要為大家介紹了go?doudou開發(fā)單體RESTful服務(wù)快速上手教程,有需要的朋友可以借鑒參考下,希望能夠有所幫助,祝大家多多進(jìn)步,早日升職加薪2022-12-12golang踩坑實(shí)戰(zhàn)之channel的正確使用方式
Golang?channel是Go語(yǔ)言中一個(gè)非常重要的特性,除了用來處理并發(fā)編程的任務(wù)中,它還可以用來進(jìn)行消息傳遞和事件通知,這篇文章主要給大家介紹了關(guān)于golang踩坑實(shí)戰(zhàn)之channel的正確使用方式,需要的朋友可以參考下2023-06-06詳解Go語(yǔ)言運(yùn)用廣度優(yōu)先搜索走迷宮
廣度優(yōu)先搜索是從圖中的某一頂點(diǎn)出發(fā),遍歷每一個(gè)頂點(diǎn)時(shí),依次遍歷其所有的鄰接點(diǎn),再?gòu)倪@些鄰接點(diǎn)出發(fā),依次訪問它們的鄰接點(diǎn),直到圖中所有被訪問過的頂點(diǎn)的鄰接點(diǎn)都被訪問到。然后查看圖中是否存在尚未被訪問的頂點(diǎn),若有,則以該頂點(diǎn)為起始點(diǎn),重復(fù)上述遍歷的過程2021-06-06