• 企业400电话
  • 微网小程序
  • AI电话机器人
  • 电商代运营
  • 全 部 栏 目

    企业400电话 网络优化推广 AI电话机器人 呼叫中心 网站建设 商标✡知产 微网小程序 电商运营 彩铃•短信 增值拓展业务
    golang API开发过程的中的自动重启方式(基于gin框架)

    概要

    基于 golang Gin 框架开发 web 服务时, 需要时不时的 go build , 然后重启服务查看运行结果.
    go build 的过程集成在编辑器中(emacs), 可以通过快捷键迅速完成, 但是每次重启服务都切换到命令行中操作.
    因此, 希望能够编译通过之后自动重启服务.

    这里并不是部署阶段的服务重启, 所以不用过多考虑是否正常退出其中的协程.

    实现方式

    在开源的 illuminant 项目中, 已经将相应的代码集成到 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 下, 使用此方法自动重启服务

    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)
     }

    补充

    上面函数的核心有以下两点:

    syscall.Exec

    对于这个函数, 一般可能用的比较少, 这里稍微介绍下. 它有 3 个参数:

    当 syscall.Exec 执行时, 在它之前的所有未执行完的程序都会被中止(包括在 go routine 中执行的程序),
    然后执行 syscall.Exec 调用的命令, 该命令还保持在之前程序的 PID 下执行.

    syscall.Exec 是最后一条执行的代码, 重启时在它之后可以有代码, 但是都不会被执行到, 包括 defer 中的代码.

    下面是个小例子(通过这个例子可以验证上面的结论):

    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{}) {
      // 重启时没有执行此函数
      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
     }

    到此这篇关于golang API开发过程的中的自动重启方式(基于gin框架)的文章就介绍到这了,更多相关golang API 自动重启内容请搜索脚本之家以前的文章或继续浏览下面的相关文章希望大家以后多多支持脚本之家!

    您可能感兴趣的文章:
    • 使用Golang简单实现七牛图片处理API
    • Laravel框架执行原生SQL语句及使用paginate分页的方法
    • Flask框架Flask-Login用法分析
    • Go Web框架gin的入门教程
    • Python的Flask框架及Nginx实现静态文件访问限制功能
    • PHP框架Laravel插件Pagination实现自定义分页
    • 使用Nginx+uWsgi实现Python的Django框架站点动静分离
    上一篇:解决Goland中利用HTTPClient发送请求超时返回EOF错误DEBUG
    下一篇:golang 设置web请求状态码操作
  • 相关文章
  • 

    © 2016-2020 巨人网络通讯 版权所有

    《增值电信业务经营许可证》 苏ICP备15040257号-8

    golang API开发过程的中的自动重启方式(基于gin框架) golang,API,开发,过程,的,中的,