溫馨提示×

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

密碼登錄×
登錄注冊(cè)×
其他方式登錄
點(diǎn)擊 登錄注冊(cè) 即表示同意《億速云用戶服務(wù)條款》

golang-grpc如何實(shí)現(xiàn)平滑重啟

發(fā)布時(shí)間:2021-12-15 09:38:55 來(lái)源:億速云 閱讀:450 作者:小新 欄目:云計(jì)算

這篇文章給大家分享的是有關(guān)golang-grpc如何實(shí)現(xiàn)平滑重啟的內(nèi)容。小編覺(jué)得挺實(shí)用的,因此分享給大家做個(gè)參考,一起跟隨小編過(guò)來(lái)看看吧。

package goo

import (
	"fmt"
	"github.com/facebookgo/grace/gracenet"
	"google.golang.org/grpc"
	"io/ioutil"
	"log"
	"os"
	"os/signal"
	"syscall"
)

type GRPCGraceful struct {
	nett string
	addr string
	s    *grpc.Server
	net  *gracenet.Net
}

func NewGRPCGraceful(nett, addr string, s *grpc.Server) *GRPCGraceful {
	return &GRPCGraceful{
		nett: nett,
		addr: addr,
		s:    s,
		net:  &gracenet.Net{},
	}
}

func (g *GRPCGraceful) Serve() error {
	lis, err := g.net.Listen(g.nett, g.addr)
	if err != nil {
		return err
	}

	errs := make(chan error)

	// 啟動(dòng)serve
	AsyncFunc(func() {
		errs <- g.s.Serve(lis)
	})
	// 判斷并關(guān)閉舊進(jìn)程
	AsyncFunc(g.killPPID)
	// 存儲(chǔ)pid
	AsyncFunc(g.storePID)

	// 監(jiān)聽(tīng)信號(hào)
	quit := g.handleSignal(errs)

	// 監(jiān)聽(tīng)退出信號(hào),錯(cuò)誤信息
	select {
	case err := <-errs:
		return err
	case <-quit:
		return nil
	}
}

// 監(jiān)聽(tīng)信號(hào)
func (g *GRPCGraceful) handleSignal(errs chan error) <-chan struct{} {
	// 通道,是否退出
	quit := make(chan struct{})

	AsyncFunc(func() {
		ch := make(chan os.Signal)
		signal.Notify(ch, syscall.SIGHUP, syscall.SIGINT, syscall.SIGTERM, syscall.SIGQUIT, syscall.SIGUSR1, syscall.SIGUSR2)

		for sig := range ch {
			switch sig {
			// 監(jiān)聽(tīng)退出
			case syscall.SIGHUP, syscall.SIGINT, syscall.SIGTERM, syscall.SIGQUIT:
				signal.Stop(ch)
				g.s.GracefulStop()
				close(quit)
				return

			// 監(jiān)聽(tīng)重啟
			case syscall.SIGUSR1, syscall.SIGUSR2:
				if _, err := g.net.StartProcess(); err != nil {
					errs <- err
				}
			}
		}
	})

	return quit
}

// 記錄進(jìn)程號(hào)到.pid文件
func (g *GRPCGraceful) storePID() {
	pid := fmt.Sprintf("%d", os.Getpid())
	ioutil.WriteFile(".pid", []byte(pid), 0644)
	log.Println(fmt.Sprintf("server is running, address=%s, pid=%s", g.addr, pid))
}

// 判斷進(jìn)程是否繼承進(jìn)程,平滑重啟時(shí),關(guān)閉舊進(jìn)程
func (g *GRPCGraceful) killPPID() {
	inherit := os.Getenv("LISTEN_FDS") != ""
	if !inherit {
		return
	}
	ppid := os.Getppid()
	if ppid == 1 {
		return
	}
	syscall.Kill(ppid, syscall.SIGTERM)
}

感謝各位的閱讀!關(guān)于“golang-grpc如何實(shí)現(xiàn)平滑重啟”這篇文章就分享到這里了,希望以上內(nèi)容可以對(duì)大家有一定的幫助,讓大家可以學(xué)到更多知識(shí),如果覺(jué)得文章不錯(cuò),可以把它分享出去讓更多的人看到吧!

向AI問(wèn)一下細(xì)節(jié)

免責(zé)聲明:本站發(fā)布的內(nèi)容(圖片、視頻和文字)以原創(chuàng)、轉(zhuǎn)載和分享為主,文章觀點(diǎn)不代表本網(wǎng)站立場(chǎng),如果涉及侵權(quán)請(qǐng)聯(lián)系站長(zhǎng)郵箱:is@yisu.com進(jìn)行舉報(bào),并提供相關(guān)證據(jù),一經(jīng)查實(shí),將立刻刪除涉嫌侵權(quán)內(nèi)容。

AI