mirror of
https://github.com/nezhahq/nezha.git
synced 2025-01-22 20:58:14 -05:00
182 lines
4.0 KiB
Go
182 lines
4.0 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"io"
|
|
"log"
|
|
"os"
|
|
"time"
|
|
|
|
"github.com/blang/semver"
|
|
"github.com/p14yground/go-github-selfupdate/selfupdate"
|
|
"github.com/spf13/cobra"
|
|
"google.golang.org/grpc"
|
|
|
|
"github.com/naiba/nezha/model"
|
|
pb "github.com/naiba/nezha/proto"
|
|
"github.com/naiba/nezha/service/dao"
|
|
"github.com/naiba/nezha/service/monitor"
|
|
"github.com/naiba/nezha/service/rpc"
|
|
)
|
|
|
|
var (
|
|
clientID string
|
|
server string
|
|
clientSecret string
|
|
debug bool
|
|
version string
|
|
|
|
rootCmd = &cobra.Command{
|
|
Use: "nezha-agent",
|
|
Short: "「哪吒面板」监控、备份、站点管理一站式服务",
|
|
Long: `哪吒面板
|
|
================================
|
|
监控、备份、站点管理一站式服务
|
|
啦啦啦,啦啦啦,我是 mjj 小行家`,
|
|
Run: run,
|
|
Version: version,
|
|
}
|
|
)
|
|
|
|
var (
|
|
reporting bool
|
|
client pb.NezhaServiceClient
|
|
ctx = context.Background()
|
|
delayWhenError = time.Second * 10
|
|
updateCh = make(chan struct{}, 0)
|
|
)
|
|
|
|
func doSelfUpdate() {
|
|
defer func() {
|
|
time.Sleep(time.Minute * 20)
|
|
updateCh <- struct{}{}
|
|
}()
|
|
v := semver.MustParse(version)
|
|
log.Println("check update", v)
|
|
latest, err := selfupdate.UpdateSelf(v, "naiba/nezha")
|
|
if err != nil {
|
|
log.Println("Binary update failed:", err)
|
|
return
|
|
}
|
|
if latest.Version.Equals(v) {
|
|
// latest version is the same as current version. It means current binary is up to date.
|
|
log.Println("Current binary is the latest version", version)
|
|
} else {
|
|
log.Println("Successfully updated to version", latest.Version)
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
func main() {
|
|
// 来自于 GoReleaser 的版本号
|
|
dao.Version = version
|
|
|
|
rootCmd.PersistentFlags().StringVarP(&server, "server", "s", "localhost:5555", "客户端ID")
|
|
rootCmd.PersistentFlags().StringVarP(&clientID, "id", "i", "", "客户端ID")
|
|
rootCmd.PersistentFlags().StringVarP(&clientSecret, "secret", "p", "", "客户端Secret")
|
|
rootCmd.PersistentFlags().BoolVarP(&debug, "debug", "d", false, "开启Debug")
|
|
if err := rootCmd.Execute(); err != nil {
|
|
fmt.Println(err)
|
|
os.Exit(1)
|
|
}
|
|
}
|
|
|
|
func run(cmd *cobra.Command, args []string) {
|
|
dao.Conf = &model.Config{
|
|
Debug: debug,
|
|
}
|
|
auth := rpc.AuthHandler{
|
|
ClientID: clientID,
|
|
ClientSecret: clientSecret,
|
|
}
|
|
|
|
// 上报服务器信息
|
|
go reportState()
|
|
|
|
if version != "" {
|
|
go func() {
|
|
for range updateCh {
|
|
go doSelfUpdate()
|
|
}
|
|
}()
|
|
updateCh <- struct{}{}
|
|
}
|
|
|
|
var err error
|
|
var conn *grpc.ClientConn
|
|
var hc pb.NezhaService_HeartbeatClient
|
|
|
|
retry := func() {
|
|
log.Println("Error to close connection ...")
|
|
if conn != nil {
|
|
conn.Close()
|
|
}
|
|
time.Sleep(delayWhenError)
|
|
log.Println("Try to reconnect ...")
|
|
}
|
|
|
|
for {
|
|
conn, err = grpc.Dial(server, grpc.WithInsecure(), grpc.WithPerRPCCredentials(&auth))
|
|
if err != nil {
|
|
log.Printf("grpc.Dial err: %v", err)
|
|
retry()
|
|
continue
|
|
}
|
|
client = pb.NewNezhaServiceClient(conn)
|
|
// 第一步注册
|
|
_, err = client.Register(ctx, monitor.GetHost().PB())
|
|
if err != nil {
|
|
log.Printf("client.Register err: %v", err)
|
|
retry()
|
|
continue
|
|
}
|
|
// 心跳接收控制命令
|
|
hc, err = client.Heartbeat(ctx, &pb.Beat{
|
|
Timestamp: fmt.Sprintf("%v", time.Now()),
|
|
})
|
|
if err != nil {
|
|
log.Printf("client.Heartbeat err: %v", err)
|
|
retry()
|
|
continue
|
|
}
|
|
err = receiveCommand(hc)
|
|
log.Printf("receiveCommand exit to main: %v", err)
|
|
retry()
|
|
}
|
|
}
|
|
|
|
func receiveCommand(hc pb.NezhaService_HeartbeatClient) error {
|
|
var err error
|
|
var action *pb.Command
|
|
defer log.Printf("receiveCommand exit %v %v => %v", time.Now(), action, err)
|
|
for {
|
|
action, err = hc.Recv()
|
|
if err == io.EOF {
|
|
return nil
|
|
}
|
|
if err != nil {
|
|
return err
|
|
}
|
|
switch action.GetType() {
|
|
default:
|
|
log.Printf("Unknown action: %v", action)
|
|
}
|
|
}
|
|
}
|
|
|
|
func reportState() {
|
|
var err error
|
|
defer log.Printf("reportState exit %v => %v", time.Now(), err)
|
|
for {
|
|
if client != nil {
|
|
monitor.TrackNetworkSpeed()
|
|
_, err = client.ReportState(ctx, monitor.GetState(dao.ReportDelay).PB())
|
|
if err != nil {
|
|
log.Printf("reportState error %v", err)
|
|
time.Sleep(delayWhenError)
|
|
}
|
|
}
|
|
}
|
|
}
|