中文字幕av专区_日韩电影在线播放_精品国产精品久久一区免费式_av在线免费观看网站

溫馨提示×

溫馨提示×

您好,登錄后才能下訂單哦!

密碼登錄×
登錄注冊×
其他方式登錄
點擊 登錄注冊 即表示同意《億速云用戶服務條款》

golang中怎么重啟httpserver

發布時間:2021-07-06 15:39:53 來源:億速云 閱讀:311 作者:Leah 欄目:編程語言

這篇文章給大家介紹golang中怎么重啟httpserver,內容非常詳細,感興趣的小伙伴們可以參考借鑒,希望對大家能有所幫助。

原理上來說是這樣一個過程:

    1)發布新的bin文件去覆蓋老的bin文件

    2)發送一個信號量,告訴正在運行的進程,進行重啟

    3)正在運行的進程收到信號后,會以子進程的方式啟動新的bin文件

    4)新進程接受新請求,并處理

    5)老進程不再接受請求,但是要等正在處理的請求處理完成,所有在處理的請求處理完之后,便自動退出

    6)新進程在老進程退出之后,由init進程收養,但是會繼續服務。

所以一步一步來看,關鍵是從第2步開始之后怎么做,所以我們先來看看第2步的實現,這個應該說很簡單,發送信號量到一個進程,使用kill命令即可,在facebook這個項目中發送的信號量有3個:SIGINT,SIGTERM,SIGUSR2,前面兩個信號收到后程序會直接退出,后面一個信號SIGUSR2才會執行所謂的優雅重啟。

第3步,正在運行的進程收到SIGUSR2信號后,會以子進程的方式啟動新的bin文件。先直接上代碼看:https://github.com/facebookgo/grace/blob/master/gracehttp/http.go

func (a *app) signalHandler(wg *sync.WaitGroup) {
 ch := make(chan os.Signal, 10)
 signal.Notify(ch, syscall.SIGINT, syscall.SIGTERM, syscall.SIGUSR2)
 for {
  sig := <-ch
  switch sig {
  case syscall.SIGINT, syscall.SIGTERM: 
   // this ensures a subsequent INT/TERM will trigger standard go behaviour of
   // terminating. 執行標準的go終止行為,程序就結束了
   signal.Stop(ch)
   a.term(wg)
   return
  case syscall.SIGUSR2: // 這里開始執行優雅重啟
   err := a.preStartProcess() 
   // 這個函數在源代碼中沒有具體實現功能,只是預留了一個鉤子函數,用戶可以注冊自己的函數,可以在重啟之前做些自定義的事情。一般情況下也沒有什么可以做的,除非有些特殊的服務環境或是狀態保存之類的,至少目前,我們的server還沒有遇到
   if err != nil {
    a.errors <- err
   }
   // we only return here if there's an error, otherwise the new process
   // will send us a TERM when it's ready to trigger the actual shutdown.
   if _, err := a.net.StartProcess(); err != nil { // 這里開始正式所謂的優雅重啟   
    a.errors <- err
   }
  }
 }
}

a.net.StartProcess的過程我們來看看基本過程:

func (n *Net) StartProcess() (int, error) {
 listeners, err := n.activeListeners() // 獲取目前在監聽的端口,這塊也是重點,下面重點介紹
 if err != nil {
  return 0, err
 }
 
 // Extract the fds from the listeners. 從監聽端口中把文件描述符取出來
 files := make([]*os.File, len(listeners))
 for i, l := range listeners {
  files[i], err = l.(filer).File()
  if err != nil {
   return 0, err
  }
  defer files[i].Close()
 }
 
 // Use the original binary location. This works with symlinks such that if
 // the file it points to has been changed we will use the updated symlink.
 // 獲取可執行bin文件的路勁,也可以是鏈接路勁,會使用最新的鏈接路徑作為啟動文件路勁的
 argv0, err := exec.LookPath(os.Args[0])
 if err != nil {
  return 0, err
 }
 
 // Pass on the environment and replace the old count key with the new one.
 // 獲取 LISTEN_FDS 換進變量值 
 var env []string
 for _, v := range os.Environ() {
  if !strings.HasPrefix(v, envCountKeyPrefix) {
   env = append(env, v)
  }
 }
 env = append(env, fmt.Sprintf("%s%d", envCountKeyPrefix, len(listeners)))
 
 allFiles := append([]*os.File{os.Stdin, os.Stdout, os.Stderr}, files...)
 // 這里調用一個golang底層的進程啟動函數,來指定,上面獲取的參數來啟動進程
 process, err := os.StartProcess(argv0, os.Args, &os.ProcAttr{
  Dir: originalWD,
  Env: env,
  Files: allFiles,
 })
 if err != nil {
  return 0, err
 }
 // 返回新進程id。
 return process.Pid, nil 
}

以上是啟動新進程,并且接管監聽端口的過程, 一般情況下端口是不可以重復監聽的,所以這里就要需要使用比較特別的辦法,從上面的代碼來看就是讀取監聽端口的文件描述符,并且把監聽端口的文件描述符傳遞給子進程,子進程里從這個文件描述符實現對端口的監聽

另外還有一個比較特別的地方就是老的接口怎么關閉的問題,關閉必須要把已經收到的請求處理完成之后再關閉。為此facebook的同學另外開了一個項目httpdown,繼承了原始的httpserver,但是多了對各種鏈接狀態的維護和處理,這部分后面在分析。

關于golang中怎么重啟httpserver就分享到這里了,希望以上內容可以對大家有一定的幫助,可以學到更多知識。如果覺得文章不錯,可以把它分享出去讓更多的人看到。

向AI問一下細節

免責聲明:本站發布的內容(圖片、視頻和文字)以原創、轉載和分享為主,文章觀點不代表本網站立場,如果涉及侵權請聯系站長郵箱:is@yisu.com進行舉報,并提供相關證據,一經查實,將立刻刪除涉嫌侵權內容。

AI

南丰县| 达拉特旗| 东乌| 九台市| 辉南县| 泸定县| 溆浦县| 重庆市| 沂水县| 东兰县| 和平县| 苏尼特右旗| 慈溪市| 车险| 嘉荫县| 通州市| 南部县| 周至县| 郴州市| 昌图县| 阿勒泰市| 政和县| 伊通| 邯郸县| 内乡县| 汉源县| 祥云县| 清水河县| 神农架林区| 仙居县| 龙陵县| 健康| 安阳县| 遂平县| 涪陵区| 洛隆县| 武定县| 岱山县| 壶关县| 新巴尔虎左旗| 海宁市|