W3Cschool
恭喜您成為首批注冊用戶
獲得88經驗值獎勵
程序開發(fā)完畢之后,我們現在要部署Web應用程序了,但是我們如何來部署這些應用程序呢?因為Go程序編譯之后是一個可執(zhí)行文件,編寫過C程序的讀者一定知道采用daemon就可以完美的實現程序后臺持續(xù)運行,但是目前Go還無法完美的實現daemon,因此,針對Go的應用程序部署,我們可以利用第三方工具來管理,第三方的工具有很多,例如Supervisord、upstart、daemontools等,這小節(jié)我介紹目前自己系統(tǒng)中采用的工具Supervisord。
目前Go程序還不能實現daemon,詳細的見這個Go語言的bug:https://github.com/golang/go/issues/227,大概的意思說很難從現有的使用的線程中fork一個出來,因為沒有一種簡單的方法來確保所有已經使用的線程的狀態(tài)一致性問題。
但是我們可以看到很多網上的一些實現daemon的方法,例如下面兩種方式:
d := flag.Bool("d", false, "Whether or not to launch in the background(like a daemon)")
if *d {
cmd := exec.Command(os.Args[0],
"-close-fds",
"-addr", *addr,
"-call", *call,
)
serr, err := cmd.StderrPipe()
if err != nil {
log.Fatalln(err)
}
err = cmd.Start()
if err != nil {
log.Fatalln(err)
}
s, err := ioutil.ReadAll(serr)
s = bytes.TrimSpace(s)
if bytes.HasPrefix(s, []byte("addr: ")) {
fmt.Println(string(s))
cmd.Process.Release()
} else {
log.Printf("unexpected response from MarGo: `%s` error: `%v`\n", s, err)
cmd.Process.Kill()
}
}
package main
import (
"log"
"os"
"syscall"
)
func daemon(nochdir, noclose int) int {
var ret, ret2 uintptr
var err uintptr
darwin := syscall.OS == "darwin"
// already a daemon
if syscall.Getppid() == 1 {
return 0
}
// fork off the parent process
ret, ret2, err = syscall.RawSyscall(syscall.SYS_FORK, 0, 0, 0)
if err != 0 {
return -1
}
// failure
if ret2 < 0 {
os.Exit(-1)
}
// handle exception for darwin
if darwin && ret2 == 1 {
ret = 0
}
// if we got a good PID, then we call exit the parent process.
if ret > 0 {
os.Exit(0)
}
/* Change the file mode mask */
_ = syscall.Umask(0)
// create a new SID for the child process
s_ret, s_errno := syscall.Setsid()
if s_errno != 0 {
log.Printf("Error: syscall.Setsid errno: %d", s_errno)
}
if s_ret < 0 {
return -1
}
if nochdir == 0 {
os.Chdir("/")
}
if noclose == 0 {
f, e := os.OpenFile("/dev/null", os.O_RDWR, 0)
if e == nil {
fd := f.Fd()
syscall.Dup2(fd, os.Stdin.Fd())
syscall.Dup2(fd, os.Stdout.Fd())
syscall.Dup2(fd, os.Stderr.Fd())
}
}
return 0
}
上面提出了兩種實現Go的daemon方案,但是我還是不推薦大家這樣去實現,因為官方還沒有正式的宣布支持daemon,當然第一種方案目前來看是比較可行的,而且目前開源庫skynet也在采用這個方案做daemon。
上面已經介紹了Go目前是有兩種方案來實現他的daemon,但是官方本身還不支持這一塊,所以還是建議大家采用第三方成熟工具來管理我們的應用程序,這里我給大家介紹一款目前使用比較廣泛的進程管理軟件:Supervisord。Supervisord是用Python實現的一款非常實用的進程管理工具。supervisord會幫你把管理的應用程序轉成daemon程序,而且可以方便的通過命令開啟、關閉、重啟等操作,而且它管理的進程一旦崩潰會自動重啟,這樣就可以保證程序執(zhí)行中斷后的情況下有自我修復的功能。
我前面在應用中踩過一個坑,就是因為所有的應用程序都是由Supervisord父進程生出來的,那么當你修改了操作系統(tǒng)的文件描述符之后,別忘記重啟Supervisord,光重啟下面的應用程序沒用。當初我就是系統(tǒng)安裝好之后就先裝了Supervisord,然后開始部署程序,修改文件描述符,重啟程序,以為文件描述符已經是100000了,其實Supervisord這個時候還是默認的1024個,導致他管理的進程所有的描述符也是1024.開放之后壓力一上來系統(tǒng)就開始報文件描述符用光了,查了很久才找到這個坑。
Supervisord可以通過sudo easy_install supervisor
安裝,當然也可以通過Supervisord官網下載后解壓并轉到源碼所在的文件夾下執(zhí)行setup.py install
來安裝。
使用easy_install必須安裝setuptools
打開http://pypi.python.org/pypi/setuptools#files
,根據你系統(tǒng)的python的版本下載相應的文件,然后執(zhí)行sh setuptoolsxxxx.egg
,這樣就可以使用easy_install命令來安裝Supervisord。
Supervisord默認的配置文件路徑為/etc/supervisord.conf,通過文本編輯器修改這個文件,下面是一個示例的配置文件:
;/etc/supervisord.conf
[unix_http_server]
file = /var/run/supervisord.sock
chmod = 0777
chown= root:root
[inet_http_server]
# Web管理界面設定
port=9001
username = admin
password = yourpassword
[supervisorctl]
; 必須和'unix_http_server'里面的設定匹配
serverurl = unix:///var/run/supervisord.sock
[supervisord]
logfile=/var/log/supervisord/supervisord.log ; (main log file;default $CWD/supervisord.log)
logfile_maxbytes=50MB ; (max main logfile bytes b4 rotation;default 50MB)
logfile_backups=10 ; (num of main logfile rotation backups;default 10)
loglevel=info ; (log level;default info; others: debug,warn,trace)
pidfile=/var/run/supervisord.pid ; (supervisord pidfile;default supervisord.pid)
nodaemon=true ; (start in foreground if true;default false)
minfds=1024 ; (min. avail startup file descriptors;default 1024)
minprocs=200 ; (min. avail process descriptors;default 200)
user=root ; (default is current user, required if root)
childlogdir=/var/log/supervisord/ ; ('AUTO' child log dir, default $TEMP)
[rpcinterface:supervisor]
supervisor.rpcinterface_factory = supervisor.rpcinterface:make_main_rpcinterface
; 管理的單個進程的配置,可以添加多個program
[program:blogdemon]
command=/data/blog/blogdemon
autostart = true
startsecs = 5
user = root
redirect_stderr = true
stdout_logfile = /var/log/supervisord/blogdemon.log
Supervisord安裝完成后有兩個可用的命令行supervisor和supervisorctl,命令使用解釋如下:
這小節(jié)我們介紹了Go如何實現daemon化,但是由于目前Go的daemon實現的不足,需要依靠第三方工具來實現應用程序的daemon管理的方式,所以在這里介紹了一個用python寫的進程管理工具Supervisord,通過Supervisord可以很方便的把我們的Go應用程序管理起來。
Copyright©2021 w3cschool編程獅|閩ICP備15016281號-3|閩公網安備35020302033924號
違法和不良信息舉報電話:173-0602-2364|舉報郵箱:jubao@eeedong.com
掃描二維碼
下載編程獅App
編程獅公眾號
聯系方式:
更多建議: