mirror of
https://github.com/duanhf2012/origin.git
synced 2026-02-03 14:34:47 +08:00
520 lines
12 KiB
Go
520 lines
12 KiB
Go
package node
|
||
|
||
import (
|
||
"errors"
|
||
"fmt"
|
||
"github.com/duanhf2012/origin/v2/cluster"
|
||
"github.com/duanhf2012/origin/v2/console"
|
||
"github.com/duanhf2012/origin/v2/log"
|
||
"github.com/duanhf2012/origin/v2/profiler"
|
||
"github.com/duanhf2012/origin/v2/service"
|
||
"github.com/duanhf2012/origin/v2/util/buildtime"
|
||
"github.com/duanhf2012/origin/v2/util/sysprocess"
|
||
"github.com/duanhf2012/origin/v2/util/timer"
|
||
"go.uber.org/zap/zapcore"
|
||
"io"
|
||
"net/http"
|
||
_ "net/http/pprof"
|
||
"os"
|
||
"os/signal"
|
||
"strconv"
|
||
"strings"
|
||
"syscall"
|
||
"time"
|
||
)
|
||
|
||
var sig chan os.Signal
|
||
var nodeId string
|
||
var preSetupService []service.IService //预安装
|
||
var preSetupTemplateService []func() service.IService
|
||
var profilerInterval time.Duration
|
||
var configDir = "./config/"
|
||
var NodeIsRun = false
|
||
|
||
const (
|
||
SingleStop syscall.Signal = 10
|
||
SignalRetire syscall.Signal = 12
|
||
)
|
||
|
||
type BuildOSType = int8
|
||
|
||
const (
|
||
Windows BuildOSType = 0
|
||
Linux BuildOSType = 1
|
||
Mac BuildOSType = 2
|
||
)
|
||
|
||
func init() {
|
||
sig = make(chan os.Signal, 4)
|
||
signal.Notify(sig, syscall.SIGINT, syscall.SIGTERM, SingleStop, SignalRetire)
|
||
|
||
console.RegisterCommandBool("help", false, "<-help> This help.", usage)
|
||
console.RegisterCommandString("name", "", "<-name nodeName> node's name.", setName)
|
||
console.RegisterCommandString("start", "", "<-start nodeid=nodeid> Run originserver.", startNode)
|
||
console.RegisterCommandString("stop", "", "<-stop nodeid=nodeid> Stop originserver process.", stopNode)
|
||
console.RegisterCommandString("retire", "", "<-retire nodeid=nodeid> retire originserver process.", retireNode)
|
||
console.RegisterCommandString("config", "", "<-config path> Configuration file path.", setConfigPath)
|
||
console.RegisterCommandString("console", "", "<-console true|false> Turn on or off screen log output.", openConsole)
|
||
console.RegisterCommandString("loglevel", "debug", "<-loglevel debug|info|warn|error|stackerror|fatal> Set loglevel.", setLevel)
|
||
console.RegisterCommandString("logpath", "", "<-logpath path> Set log file path.", setLogPath)
|
||
console.RegisterCommandInt("logsize", 0, "<-logsize size> Set log size(MB).", setLogSize)
|
||
console.RegisterCommandInt("logchanlen", 0, "<-logchanlen len> Set log channel len.", setLogChanLen)
|
||
console.RegisterCommandString("pprof", "", "<-pprof ip:port> Open performance analysis.", setPprof)
|
||
}
|
||
|
||
func notifyAllServiceRetire() {
|
||
service.NotifyAllServiceRetire()
|
||
}
|
||
|
||
func usage(val interface{}) error {
|
||
ret := val.(bool)
|
||
if ret == false {
|
||
return nil
|
||
}
|
||
|
||
if len(buildtime.GetBuildDateTime()) > 0 {
|
||
fmt.Fprintf(os.Stderr, "Welcome to Origin(build info: %s)\nUsage: originserver [-help] [-start node=1] [-stop] [-config path] [-pprof 0.0.0.0:6060]...\n", buildtime.GetBuildDateTime())
|
||
} else {
|
||
fmt.Fprintf(os.Stderr, "Welcome to Origin\nUsage: originserver [-help] [-start node=1] [-stop] [-config path] [-pprof 0.0.0.0:6060]...\n")
|
||
}
|
||
|
||
console.PrintDefaults()
|
||
return nil
|
||
}
|
||
|
||
func setName(_ interface{}) error {
|
||
return nil
|
||
}
|
||
|
||
func setPprof(val interface{}) error {
|
||
listenAddr := val.(string)
|
||
if listenAddr == "" {
|
||
return nil
|
||
}
|
||
|
||
go func() {
|
||
err := http.ListenAndServe(listenAddr, nil)
|
||
if err != nil {
|
||
panic(fmt.Errorf("%+v", err))
|
||
}
|
||
}()
|
||
|
||
return nil
|
||
}
|
||
|
||
func setConfigPath(val interface{}) error {
|
||
configPath := val.(string)
|
||
if configPath == "" {
|
||
return nil
|
||
}
|
||
_, err := os.Stat(configPath)
|
||
if err != nil {
|
||
return fmt.Errorf("cannot find file path %s", configPath)
|
||
}
|
||
|
||
cluster.SetConfigDir(configPath)
|
||
configDir = configPath
|
||
return nil
|
||
}
|
||
|
||
func getRunProcessPid(nodeId string) (int, error) {
|
||
f, err := os.OpenFile(fmt.Sprintf("%s_%s.pid", os.Args[0], nodeId), os.O_RDONLY, 0600)
|
||
defer f.Close()
|
||
if err != nil {
|
||
return 0, err
|
||
}
|
||
|
||
pidByte, errs := io.ReadAll(f)
|
||
if errs != nil {
|
||
return 0, errs
|
||
}
|
||
|
||
return strconv.Atoi(string(pidByte))
|
||
}
|
||
|
||
func writeProcessPid(nodeId string) {
|
||
//pid
|
||
f, err := os.OpenFile(fmt.Sprintf("%s_%s.pid", os.Args[0], nodeId), os.O_WRONLY|os.O_TRUNC|os.O_CREATE, 0600)
|
||
defer f.Close()
|
||
if err != nil {
|
||
fmt.Println(err.Error())
|
||
os.Exit(-1)
|
||
} else {
|
||
_, err = f.Write([]byte(fmt.Sprintf("%d", os.Getpid())))
|
||
if err != nil {
|
||
fmt.Println(err.Error())
|
||
os.Exit(-1)
|
||
}
|
||
}
|
||
}
|
||
|
||
func GetNodeId() string {
|
||
return nodeId
|
||
}
|
||
|
||
func initNode(id string) {
|
||
//1.初始化集群
|
||
nodeId = id
|
||
err := cluster.GetCluster().Init(GetNodeId(), Setup)
|
||
if err != nil {
|
||
log.Error("Init cluster fail", log.ErrorField("error", err))
|
||
os.Exit(1)
|
||
}
|
||
|
||
err = initLog()
|
||
if err != nil {
|
||
log.Error("Init log fail", log.ErrorField("error", err))
|
||
os.Exit(1)
|
||
return
|
||
}
|
||
|
||
//2.顺序安装服务
|
||
serviceOrder := cluster.GetCluster().GetLocalNodeInfo().ServiceList
|
||
for _, serviceName := range serviceOrder {
|
||
bSetup := false
|
||
|
||
//判断是否有配置模板服务
|
||
splitServiceName := strings.Split(serviceName, ":")
|
||
if len(splitServiceName) == 2 {
|
||
serviceName = splitServiceName[0]
|
||
templateServiceName := splitServiceName[1]
|
||
for _, newSer := range preSetupTemplateService {
|
||
ser := newSer()
|
||
ser.OnSetup(ser)
|
||
if ser.GetName() == templateServiceName {
|
||
ser.SetName(serviceName)
|
||
ser.Init(ser, cluster.GetRpcClient, cluster.GetRpcServer, cluster.GetCluster().GetServiceCfg(ser.GetName()))
|
||
service.Setup(ser)
|
||
|
||
bSetup = true
|
||
break
|
||
}
|
||
}
|
||
|
||
if bSetup == false {
|
||
log.Error("Template service not found", log.String("service name", serviceName), log.String("template service name", templateServiceName))
|
||
os.Exit(1)
|
||
}
|
||
}
|
||
|
||
for _, s := range preSetupService {
|
||
if s.GetName() != serviceName {
|
||
continue
|
||
}
|
||
bSetup = true
|
||
pServiceCfg := cluster.GetCluster().GetServiceCfg(s.GetName())
|
||
s.Init(s, cluster.GetRpcClient, cluster.GetRpcServer, pServiceCfg)
|
||
|
||
service.Setup(s)
|
||
}
|
||
|
||
if bSetup == false {
|
||
log.Fatal("Service name " + serviceName + " configuration error")
|
||
}
|
||
}
|
||
|
||
//3.service初始化
|
||
log.Info("Start running server.")
|
||
service.Init()
|
||
}
|
||
|
||
func initLog() error {
|
||
logger := log.GetLogger()
|
||
if log.LogPath == "" {
|
||
err := setLogPath("./log")
|
||
if err != nil {
|
||
return err
|
||
}
|
||
}
|
||
|
||
localNodeInfo := cluster.GetCluster().GetLocalNodeInfo()
|
||
fileName := fmt.Sprintf("%s.log", localNodeInfo.NodeId)
|
||
logger.FileName = fileName
|
||
|
||
logger.Init()
|
||
return nil
|
||
}
|
||
|
||
func Start() {
|
||
err := console.Run(os.Args)
|
||
if err != nil {
|
||
fmt.Printf("%+v\n", err)
|
||
return
|
||
}
|
||
}
|
||
|
||
func retireNode(args interface{}) error {
|
||
//1.解析参数
|
||
param := args.(string)
|
||
if param == "" {
|
||
return nil
|
||
}
|
||
|
||
sParam := strings.Split(param, "=")
|
||
if len(sParam) != 2 {
|
||
return fmt.Errorf("invalid option %s", param)
|
||
}
|
||
if sParam[0] != "nodeid" {
|
||
return fmt.Errorf("invalid option %s", param)
|
||
}
|
||
nId := strings.TrimSpace(sParam[1])
|
||
if nId == "" {
|
||
return fmt.Errorf("invalid option %s", param)
|
||
}
|
||
|
||
processId, err := getRunProcessPid(nId)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
|
||
RetireProcess(processId)
|
||
return nil
|
||
}
|
||
|
||
func stopNode(args interface{}) error {
|
||
//1.解析参数
|
||
param := args.(string)
|
||
if param == "" {
|
||
return nil
|
||
}
|
||
|
||
sParam := strings.Split(param, "=")
|
||
if len(sParam) != 2 {
|
||
return fmt.Errorf("invalid option %s", param)
|
||
}
|
||
if sParam[0] != "nodeid" {
|
||
return fmt.Errorf("invalid option %s", param)
|
||
}
|
||
nId := strings.TrimSpace(sParam[1])
|
||
if nId == "" {
|
||
return fmt.Errorf("invalid option %s", param)
|
||
}
|
||
|
||
processId, err := getRunProcessPid(nId)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
|
||
KillProcess(processId)
|
||
return nil
|
||
}
|
||
|
||
func startNode(args interface{}) error {
|
||
//1.解析参数
|
||
param := args.(string)
|
||
if param == "" {
|
||
return nil
|
||
}
|
||
|
||
sParam := strings.Split(param, "=")
|
||
if len(sParam) != 2 {
|
||
return fmt.Errorf("invalid option %s", param)
|
||
}
|
||
if sParam[0] != "nodeid" {
|
||
return fmt.Errorf("invalid option %s", param)
|
||
}
|
||
strNodeId := strings.TrimSpace(sParam[1])
|
||
if strNodeId == "" {
|
||
return fmt.Errorf("invalid option %s", param)
|
||
}
|
||
|
||
for {
|
||
processId, pErr := getRunProcessPid(strNodeId)
|
||
if pErr != nil {
|
||
break
|
||
}
|
||
|
||
name, cErr := sysprocess.GetProcessNameByPID(int32(processId))
|
||
myName, mErr := sysprocess.GetMyProcessName()
|
||
//当前进程名获取失败,不应该发生
|
||
if mErr != nil {
|
||
log.Error("get my process's name is error", log.ErrorField("err", mErr))
|
||
os.Exit(-1)
|
||
}
|
||
|
||
//进程id存在,而且进程名也相同,被认为是当前进程重复运行
|
||
if cErr == nil && name == myName {
|
||
log.Error("repeat runs are not allowed", log.String("nodeId", strNodeId), log.Int("processId", processId))
|
||
os.Exit(-1)
|
||
}
|
||
break
|
||
}
|
||
|
||
//2.记录进程id号
|
||
writeProcessPid(strNodeId)
|
||
timer.StartTimer(10*time.Millisecond, 1000000)
|
||
|
||
//3.初始化node
|
||
defer log.GetLogger().Logger.Sync()
|
||
initNode(strNodeId)
|
||
|
||
//4.运行service
|
||
service.Start()
|
||
|
||
//5.运行集群
|
||
err := cluster.GetCluster().Start()
|
||
if err != nil {
|
||
log.Error(err.Error())
|
||
os.Exit(-1)
|
||
}
|
||
|
||
//6.监听程序退出信号&性能报告
|
||
var pProfilerTicker *time.Ticker = &time.Ticker{}
|
||
if profilerInterval > 0 {
|
||
pProfilerTicker = time.NewTicker(profilerInterval)
|
||
}
|
||
|
||
NodeIsRun = true
|
||
for NodeIsRun {
|
||
select {
|
||
case s := <-sig:
|
||
signal := s.(syscall.Signal)
|
||
if signal == SignalRetire {
|
||
log.Info("receipt retire signal.")
|
||
notifyAllServiceRetire()
|
||
} else {
|
||
NodeIsRun = false
|
||
log.Info("receipt stop signal.")
|
||
}
|
||
case <-pProfilerTicker.C:
|
||
profiler.Report()
|
||
}
|
||
}
|
||
|
||
//7.退出
|
||
service.StopAllService()
|
||
cluster.GetCluster().Stop()
|
||
|
||
log.Info("Server is stop.")
|
||
|
||
return nil
|
||
}
|
||
|
||
type templateServicePoint[T any] interface {
|
||
*T
|
||
service.IService
|
||
}
|
||
|
||
func Setup(s ...service.IService) {
|
||
for _, sv := range s {
|
||
sv.OnSetup(sv)
|
||
preSetupService = append(preSetupService, sv)
|
||
}
|
||
}
|
||
|
||
func SetupTemplateFunc(fs ...func() service.IService) {
|
||
for _, f := range fs {
|
||
preSetupTemplateService = append(preSetupTemplateService, f)
|
||
}
|
||
}
|
||
|
||
func SetupTemplate[T any, P templateServicePoint[T]]() {
|
||
SetupTemplateFunc(func() service.IService {
|
||
var t T
|
||
return P(&t)
|
||
})
|
||
}
|
||
|
||
func GetService(serviceName string) service.IService {
|
||
return service.GetService(serviceName)
|
||
}
|
||
|
||
func SetConfigDir(cfgDir string) {
|
||
configDir = cfgDir
|
||
cluster.SetConfigDir(cfgDir)
|
||
}
|
||
|
||
func GetConfigDir() string {
|
||
return configDir
|
||
}
|
||
|
||
func OpenProfilerReport(interval time.Duration) {
|
||
profilerInterval = interval
|
||
}
|
||
|
||
func openConsole(args interface{}) error {
|
||
if args == "" {
|
||
return nil
|
||
}
|
||
strOpen := strings.ToLower(strings.TrimSpace(args.(string)))
|
||
if strOpen == "false" {
|
||
bOpenConsole := false
|
||
log.OpenConsole = &bOpenConsole
|
||
} else if strOpen == "true" {
|
||
bOpenConsole := true
|
||
log.OpenConsole = &bOpenConsole
|
||
} else {
|
||
return errors.New("parameter console error")
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func setLevel(args interface{}) error {
|
||
if args == "" {
|
||
return nil
|
||
}
|
||
|
||
strlogLevel := strings.TrimSpace(args.(string))
|
||
switch strlogLevel {
|
||
case "debug":
|
||
log.LogLevel = zapcore.DebugLevel
|
||
case "info":
|
||
log.LogLevel = zapcore.InfoLevel
|
||
case "warn":
|
||
log.LogLevel = zapcore.WarnLevel
|
||
case "error":
|
||
log.LogLevel = zapcore.ErrorLevel
|
||
case "stackerror":
|
||
log.LogLevel = zapcore.ErrorLevel
|
||
case "fatal":
|
||
log.LogLevel = zapcore.FatalLevel
|
||
default:
|
||
return errors.New("unknown level: " + strlogLevel)
|
||
}
|
||
return nil
|
||
}
|
||
|
||
func setLogPath(args interface{}) error {
|
||
if args == "" {
|
||
return nil
|
||
}
|
||
logPath := strings.TrimSpace(args.(string))
|
||
_, err := os.Stat(logPath)
|
||
if err != nil {
|
||
err = os.MkdirAll(logPath, os.ModePerm)
|
||
if err != nil {
|
||
return errors.New("Cannot create dir " + logPath)
|
||
}
|
||
}
|
||
|
||
log.LogPath = logPath
|
||
return nil
|
||
}
|
||
|
||
func setLogSize(args interface{}) error {
|
||
logSize, ok := args.(int)
|
||
if ok == false {
|
||
return errors.New("param logsize is error")
|
||
}
|
||
if logSize == 0 {
|
||
return nil
|
||
}
|
||
|
||
log.MaxSize = logSize
|
||
|
||
return nil
|
||
}
|
||
|
||
func setLogChanLen(args interface{}) error {
|
||
logChanLen, ok := args.(int)
|
||
if ok == false {
|
||
return errors.New("param logsize is error")
|
||
}
|
||
if logChanLen == 0 {
|
||
return nil
|
||
}
|
||
|
||
log.LogChanLen = logChanLen
|
||
return nil
|
||
}
|