Go基础学习记录之部署和维护(三)

{app.params.name}}{app.params.name}}{app.params.name}}

部署

当我们的Web应用程序最终生产就绪时,部署它所需的步骤是什么? 在Go中,在编译程序之后创建封装我们的应用程序的可执行文件。 用C编写的程序可以完美地作为后台守护程序进程运行,但是Go还没有对守护进程的本机支持。 好消息是我们可以使用第三方工具来帮助我们管理Go应用程序的部署,其中包括Supervisord,upstart和daemontools等。 本节将向您介绍Supervisord过程控制系统的一些基础知识。

守护进程

目前,Go程序不能作为守护进程运行(有关其他信息,请参阅github上的未解决问题)。 在Go中分叉现有线程很困难,因为无法确保所有已使用的线程中的一致状态。

但是,我们可以看到许多在线实现守护进程的尝试,例如以下两种方式;

  • MarGo使用Command部署应用程序概念的一个实现。 如果您真的想要对应用程序进行守护,建议使用类似于以下内容的代码:
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中实现了守护进程,但我仍然不建议您使用任何一种方法,因为Go中没有对守护进程的正式支持。 尽管如此,第一个选项是更可行的选项,并且目前被一些着名的开源项目(如skynet)用于实现守护进程。

Supervisord

上面,我们看了两个常用于在Go中实现守护进程的方案,但这两种方法都缺乏官方支持。 因此,建议您使用第三方工具来管理应用程序部署。 在这里,我们来看看用Python实现的Supervisord项目,该项目为流程管理提供了广泛的工具。 Supervisord将帮助您对Go应用程序进行守护,还允许您使用一些简单的命令以及许多其他操作来执行诸如启动,关闭和重新启动应用程序之类的操作。 此外,Supervisord托管进程可以自动重启已崩溃的进程,确保程序可以从任何中断中恢复。

安装Supervisord

可以使用sudo easy_install supervisor轻松安装Supervisord。 当然,还可以选择直接从官方网站下载,解压缩,进入文件夹,然后运行setup.py install手动安装。

  • 如果您要使用easy_install路由,则需要先安装setuptools

转到http://pypi.python.org/pypi/setuptools#files并下载相应的文件,具体取决于系统的python版本。 输入目录并执行sh setuptoolsxxxx.egg。 当脚本完成后,您将能够使用easy_install命令安装Supervisord。

配置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 management interface settings
port=9001
username = admin
password = yourpassword

[supervisorctl]
; Must 'unix_http_server' match the settings inside
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
; Manage the configuration of a single process, you can add multiple program
[program: blogdemon]
command =/data/blog/blogdemon
autostart = true
startsecs = 5
user = root
redirect_stderr = true
stdout_logfile =/var/log/supervisord/blogdemon.log

Supervisord管理

安装完成后,可以在命令行上使用两个Supervisord命令:supervisor和supervisorctl。 命令如下:

  • supervisord:初始启动,启动和流程配置管理。
  • supervisorctl stop programxxx:停止programxxx进程,其中programxxx是supervisord.conf文件中配置的值。 例如,如果您配置了[program:blogdemon],则可以使用supervisorctl stop blogdemon命令终止该进程。
  • supervisorctl start programxxx:启动programxxx进程
  • supervisorctl restart programxxx:重启programxxx进程
  • supervisorctl stop all:停止所有进程; 注意:启动,重启,停止不会加载最新的配置文件。
  • supervisorctl reload:加载最新的配置文件,启动它们,并使用新配置管理所有进程。

小结

在本次分享中,我们描述了如何在Go中实现守护进程。 我们了解到Go本身并不支持守护进程,我们需要使用第三方工具来帮助我们管理守护进程。 其中一个工具是Supervisord过程控制系统,我们可以使用它来轻松部署和管理我们的Go程序。

版权声明

durban创作并维护的 小绒毛的足迹博客采用创作共用保留署名-非商业-禁止演绎4.0国际许可证。

本文首发于 博客( https://www.xiaorongmao.com ),版权所有,侵权必究。

本文永久链接: https://www.xiaorongmao.com/blog/121



版权声明

durban创作并维护的 小绒毛的足迹博客采用创作共用保留署名-非商业-禁止演绎4.0国际许可证。

本文首发于 小绒毛的足迹博客( https://www.xiaorongmao.com ),版权所有,侵权必究。

本文永久链接: https://www.xiaorongmao.com/blog/121