2019-12-05 09:36:58 -05:00
|
|
|
|
package main
|
|
|
|
|
|
|
|
|
|
import (
|
|
|
|
|
"context"
|
2021-01-16 01:11:51 -05:00
|
|
|
|
"crypto/tls"
|
2021-01-15 11:45:49 -05:00
|
|
|
|
"errors"
|
2021-01-30 04:10:51 -05:00
|
|
|
|
"flag"
|
2019-12-07 05:14:40 -05:00
|
|
|
|
"fmt"
|
2019-12-05 09:36:58 -05:00
|
|
|
|
"log"
|
2021-01-15 11:45:49 -05:00
|
|
|
|
"net"
|
|
|
|
|
"net/http"
|
2021-05-08 04:11:13 -04:00
|
|
|
|
"net/url"
|
2019-12-08 10:49:38 -05:00
|
|
|
|
"os"
|
2021-01-18 20:59:04 -05:00
|
|
|
|
"os/exec"
|
2019-12-07 05:14:40 -05:00
|
|
|
|
"time"
|
2019-12-05 09:36:58 -05:00
|
|
|
|
|
2020-11-29 11:07:27 -05:00
|
|
|
|
"github.com/blang/semver"
|
2021-01-15 11:45:49 -05:00
|
|
|
|
"github.com/genkiroid/cert"
|
|
|
|
|
"github.com/go-ping/ping"
|
2020-12-12 12:31:22 -05:00
|
|
|
|
"github.com/p14yground/go-github-selfupdate/selfupdate"
|
2019-12-05 09:36:58 -05:00
|
|
|
|
"google.golang.org/grpc"
|
2019-12-05 10:42:20 -05:00
|
|
|
|
|
2021-03-20 02:53:10 -04:00
|
|
|
|
"github.com/naiba/nezha/cmd/agent/monitor"
|
2020-11-10 21:07:45 -05:00
|
|
|
|
"github.com/naiba/nezha/model"
|
2021-01-28 10:19:59 -05:00
|
|
|
|
"github.com/naiba/nezha/pkg/utils"
|
2020-11-10 21:07:45 -05:00
|
|
|
|
pb "github.com/naiba/nezha/proto"
|
|
|
|
|
"github.com/naiba/nezha/service/dao"
|
|
|
|
|
"github.com/naiba/nezha/service/rpc"
|
2019-12-05 09:36:58 -05:00
|
|
|
|
)
|
|
|
|
|
|
2021-05-21 22:56:30 -04:00
|
|
|
|
func init() {
|
|
|
|
|
cert.TimeoutSeconds = 30
|
|
|
|
|
http.DefaultClient.Timeout = time.Second * 5
|
|
|
|
|
}
|
|
|
|
|
|
2019-12-08 10:49:38 -05:00
|
|
|
|
var (
|
2020-11-29 11:07:27 -05:00
|
|
|
|
server string
|
|
|
|
|
clientSecret string
|
2020-12-18 23:43:03 -05:00
|
|
|
|
version string
|
2019-12-08 10:49:38 -05:00
|
|
|
|
)
|
|
|
|
|
|
2020-11-30 01:24:00 -05:00
|
|
|
|
var (
|
2021-05-21 22:56:30 -04:00
|
|
|
|
client pb.NezhaServiceClient
|
|
|
|
|
ctx = context.Background()
|
|
|
|
|
updateCh = make(chan struct{}) // Agent 自动更新间隔
|
|
|
|
|
httpClient = &http.Client{
|
2021-01-20 20:37:29 -05:00
|
|
|
|
Transport: &http.Transport{
|
|
|
|
|
TLSClientConfig: &tls.Config{InsecureSkipVerify: true},
|
|
|
|
|
},
|
|
|
|
|
CheckRedirect: func(req *http.Request, via []*http.Request) error {
|
|
|
|
|
return http.ErrUseLastResponse
|
|
|
|
|
},
|
2021-05-21 22:56:30 -04:00
|
|
|
|
Timeout: time.Second * 30,
|
2021-01-20 20:37:29 -05:00
|
|
|
|
}
|
2020-11-30 01:24:00 -05:00
|
|
|
|
)
|
|
|
|
|
|
2021-05-21 22:56:30 -04:00
|
|
|
|
const (
|
|
|
|
|
delayWhenError = time.Second * 10 // Agent 重连间隔
|
|
|
|
|
)
|
2021-01-17 02:43:34 -05:00
|
|
|
|
|
2019-12-05 09:36:58 -05:00
|
|
|
|
func main() {
|
2020-11-29 23:45:51 -05:00
|
|
|
|
// 来自于 GoReleaser 的版本号
|
|
|
|
|
dao.Version = version
|
2019-12-08 10:49:38 -05:00
|
|
|
|
|
2021-01-30 04:10:51 -05:00
|
|
|
|
var debug bool
|
2021-02-06 20:51:55 -05:00
|
|
|
|
flag.String("i", "", "unused 旧Agent配置兼容")
|
2021-05-10 05:30:18 -04:00
|
|
|
|
flag.BoolVar(&debug, "d", false, "开启调试信息")
|
2021-01-30 04:10:51 -05:00
|
|
|
|
flag.StringVar(&server, "s", "localhost:5555", "管理面板RPC端口")
|
|
|
|
|
flag.StringVar(&clientSecret, "p", "", "Agent连接Secret")
|
|
|
|
|
flag.Parse()
|
|
|
|
|
|
2019-12-09 03:02:49 -05:00
|
|
|
|
dao.Conf = &model.Config{
|
|
|
|
|
Debug: debug,
|
2019-12-05 09:36:58 -05:00
|
|
|
|
}
|
2021-01-30 04:10:51 -05:00
|
|
|
|
|
|
|
|
|
if server == "" || clientSecret == "" {
|
|
|
|
|
flag.Usage()
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
run()
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func run() {
|
2019-12-09 03:02:49 -05:00
|
|
|
|
auth := rpc.AuthHandler{
|
|
|
|
|
ClientSecret: clientSecret,
|
2019-12-05 09:36:58 -05:00
|
|
|
|
}
|
2019-12-10 04:57:57 -05:00
|
|
|
|
|
|
|
|
|
// 上报服务器信息
|
|
|
|
|
go reportState()
|
2021-03-20 11:50:16 -04:00
|
|
|
|
// 更新IP信息
|
|
|
|
|
go monitor.UpdateIP()
|
2019-12-10 04:57:57 -05:00
|
|
|
|
|
2021-05-21 22:56:30 -04:00
|
|
|
|
if _, err := semver.Parse(version); err == nil {
|
2020-12-19 10:27:59 -05:00
|
|
|
|
go func() {
|
|
|
|
|
for range updateCh {
|
|
|
|
|
go doSelfUpdate()
|
|
|
|
|
}
|
|
|
|
|
}()
|
|
|
|
|
updateCh <- struct{}{}
|
|
|
|
|
}
|
2020-11-29 11:07:27 -05:00
|
|
|
|
|
2019-12-10 04:57:57 -05:00
|
|
|
|
var err error
|
|
|
|
|
var conn *grpc.ClientConn
|
|
|
|
|
|
2020-02-09 09:04:59 -05:00
|
|
|
|
retry := func() {
|
2021-04-25 11:46:15 -04:00
|
|
|
|
println("Error to close connection ...")
|
2020-02-09 09:04:59 -05:00
|
|
|
|
if conn != nil {
|
|
|
|
|
conn.Close()
|
|
|
|
|
}
|
|
|
|
|
time.Sleep(delayWhenError)
|
2021-04-25 11:46:15 -04:00
|
|
|
|
println("Try to reconnect ...")
|
2020-02-09 09:04:59 -05:00
|
|
|
|
}
|
|
|
|
|
|
2019-12-09 05:14:31 -05:00
|
|
|
|
for {
|
2021-05-21 22:56:30 -04:00
|
|
|
|
timeOutCtx, cancel := context.WithTimeout(ctx, time.Second*5)
|
|
|
|
|
conn, err = grpc.DialContext(timeOutCtx, server, grpc.WithInsecure(), grpc.WithPerRPCCredentials(&auth))
|
2019-12-09 03:02:49 -05:00
|
|
|
|
if err != nil {
|
2021-04-25 11:46:15 -04:00
|
|
|
|
println("grpc.Dial err: ", err)
|
2021-05-21 22:56:30 -04:00
|
|
|
|
cancel()
|
2019-12-09 05:14:31 -05:00
|
|
|
|
retry()
|
2019-12-09 03:02:49 -05:00
|
|
|
|
continue
|
|
|
|
|
}
|
2021-05-21 22:56:30 -04:00
|
|
|
|
cancel()
|
2019-12-09 03:02:49 -05:00
|
|
|
|
client = pb.NewNezhaServiceClient(conn)
|
|
|
|
|
// 第一步注册
|
2021-01-15 11:45:49 -05:00
|
|
|
|
_, err = client.ReportSystemInfo(ctx, monitor.GetHost().PB())
|
2019-12-09 05:14:31 -05:00
|
|
|
|
if err != nil {
|
2021-04-25 11:46:15 -04:00
|
|
|
|
println("client.ReportSystemInfo err: ", err)
|
2019-12-09 05:14:31 -05:00
|
|
|
|
retry()
|
|
|
|
|
continue
|
|
|
|
|
}
|
2021-01-15 11:45:49 -05:00
|
|
|
|
// 执行 Task
|
|
|
|
|
tasks, err := client.RequestTask(ctx, monitor.GetHost().PB())
|
2019-12-09 03:02:49 -05:00
|
|
|
|
if err != nil {
|
2021-04-25 11:46:15 -04:00
|
|
|
|
println("client.RequestTask err: ", err)
|
2019-12-09 05:14:31 -05:00
|
|
|
|
retry()
|
2019-12-09 03:02:49 -05:00
|
|
|
|
continue
|
|
|
|
|
}
|
2021-01-15 11:45:49 -05:00
|
|
|
|
err = receiveTasks(tasks)
|
2021-04-25 11:46:15 -04:00
|
|
|
|
println("receiveTasks exit to main: ", err)
|
2019-12-09 05:14:31 -05:00
|
|
|
|
retry()
|
2019-12-07 05:14:40 -05:00
|
|
|
|
}
|
2019-12-09 03:02:49 -05:00
|
|
|
|
}
|
2019-12-07 05:14:40 -05:00
|
|
|
|
|
2021-01-15 11:45:49 -05:00
|
|
|
|
func receiveTasks(tasks pb.NezhaService_RequestTaskClient) error {
|
2019-12-09 03:02:49 -05:00
|
|
|
|
var err error
|
2021-04-25 11:46:15 -04:00
|
|
|
|
defer println("receiveTasks exit", time.Now(), "=>", err)
|
2019-12-09 03:02:49 -05:00
|
|
|
|
for {
|
2021-01-16 05:04:47 -05:00
|
|
|
|
var task *pb.Task
|
2021-01-15 11:45:49 -05:00
|
|
|
|
task, err = tasks.Recv()
|
2019-12-09 03:02:49 -05:00
|
|
|
|
if err != nil {
|
|
|
|
|
return err
|
|
|
|
|
}
|
2021-01-16 05:04:47 -05:00
|
|
|
|
go doTask(task)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
func doTask(task *pb.Task) {
|
|
|
|
|
var result pb.TaskResult
|
|
|
|
|
result.Id = task.GetId()
|
|
|
|
|
result.Type = task.GetType()
|
|
|
|
|
switch task.GetType() {
|
2021-01-18 20:59:04 -05:00
|
|
|
|
case model.TaskTypeHTTPGET:
|
2021-01-16 05:04:47 -05:00
|
|
|
|
start := time.Now()
|
|
|
|
|
resp, err := httpClient.Get(task.GetData())
|
|
|
|
|
if err == nil {
|
2021-04-20 07:30:34 -04:00
|
|
|
|
// 检查 HTTP Response 状态
|
|
|
|
|
result.Delay = float32(time.Since(start).Microseconds()) / 1000.0
|
2021-01-20 20:37:29 -05:00
|
|
|
|
if resp.StatusCode > 399 || resp.StatusCode < 200 {
|
2021-01-16 05:04:47 -05:00
|
|
|
|
err = errors.New("\n应用错误:" + resp.Status)
|
2021-01-15 11:45:49 -05:00
|
|
|
|
}
|
2021-01-16 05:04:47 -05:00
|
|
|
|
}
|
|
|
|
|
if err == nil {
|
2021-04-20 07:30:34 -04:00
|
|
|
|
// 检查 SSL 证书信息
|
2021-05-08 04:11:13 -04:00
|
|
|
|
serviceUrl, err := url.Parse(task.GetData())
|
|
|
|
|
if err == nil {
|
|
|
|
|
if serviceUrl.Scheme == "https" {
|
|
|
|
|
c := cert.NewCert(serviceUrl.Host)
|
|
|
|
|
if c.Error != "" {
|
|
|
|
|
result.Data = "SSL证书错误:" + c.Error
|
|
|
|
|
} else {
|
|
|
|
|
result.Data = c.Issuer + "|" + c.NotAfter
|
|
|
|
|
result.Successful = true
|
|
|
|
|
}
|
2021-01-16 05:04:47 -05:00
|
|
|
|
} else {
|
|
|
|
|
result.Successful = true
|
2021-01-15 11:45:49 -05:00
|
|
|
|
}
|
2021-01-20 20:37:29 -05:00
|
|
|
|
} else {
|
2021-05-08 04:11:13 -04:00
|
|
|
|
result.Data = "URL解析错误:" + err.Error()
|
2021-01-15 11:45:49 -05:00
|
|
|
|
}
|
2021-01-16 05:04:47 -05:00
|
|
|
|
} else {
|
2021-04-20 07:30:34 -04:00
|
|
|
|
// HTTP 请求失败
|
2021-01-16 05:04:47 -05:00
|
|
|
|
result.Data = err.Error()
|
|
|
|
|
}
|
2021-01-18 20:59:04 -05:00
|
|
|
|
case model.TaskTypeICMPPing:
|
2021-01-16 05:04:47 -05:00
|
|
|
|
pinger, err := ping.NewPinger(task.GetData())
|
|
|
|
|
if err == nil {
|
2021-01-18 00:45:06 -05:00
|
|
|
|
pinger.SetPrivileged(true)
|
2021-05-21 22:56:30 -04:00
|
|
|
|
pinger.Count = 5
|
2021-01-16 05:04:47 -05:00
|
|
|
|
pinger.Timeout = time.Second * 20
|
|
|
|
|
err = pinger.Run() // Blocks until finished.
|
|
|
|
|
}
|
|
|
|
|
if err == nil {
|
|
|
|
|
result.Delay = float32(pinger.Statistics().AvgRtt.Microseconds()) / 1000.0
|
|
|
|
|
result.Successful = true
|
|
|
|
|
} else {
|
|
|
|
|
result.Data = err.Error()
|
|
|
|
|
}
|
2021-01-18 20:59:04 -05:00
|
|
|
|
case model.TaskTypeTCPPing:
|
2021-01-16 05:04:47 -05:00
|
|
|
|
start := time.Now()
|
|
|
|
|
conn, err := net.DialTimeout("tcp", task.GetData(), time.Second*10)
|
|
|
|
|
if err == nil {
|
|
|
|
|
conn.Write([]byte("ping\n"))
|
|
|
|
|
conn.Close()
|
2021-04-20 07:30:34 -04:00
|
|
|
|
result.Delay = float32(time.Since(start).Microseconds()) / 1000.0
|
2021-01-16 05:04:47 -05:00
|
|
|
|
result.Successful = true
|
|
|
|
|
} else {
|
|
|
|
|
result.Data = err.Error()
|
2019-12-09 03:02:49 -05:00
|
|
|
|
}
|
2021-01-18 20:59:04 -05:00
|
|
|
|
case model.TaskTypeCommand:
|
|
|
|
|
startedAt := time.Now()
|
2021-01-28 10:19:59 -05:00
|
|
|
|
var cmd *exec.Cmd
|
2021-01-28 21:40:57 -05:00
|
|
|
|
var endCh = make(chan struct{})
|
2021-01-28 22:59:35 -05:00
|
|
|
|
pg, err := utils.NewProcessExitGroup()
|
|
|
|
|
if err != nil {
|
|
|
|
|
// 进程组创建失败,直接退出
|
|
|
|
|
result.Data = err.Error()
|
|
|
|
|
client.ReportTask(ctx, &result)
|
|
|
|
|
return
|
|
|
|
|
}
|
2021-01-28 21:40:57 -05:00
|
|
|
|
timeout := time.NewTimer(time.Hour * 2)
|
2021-01-28 10:19:59 -05:00
|
|
|
|
if utils.IsWindows() {
|
|
|
|
|
cmd = exec.Command("cmd", "/c", task.GetData())
|
|
|
|
|
} else {
|
|
|
|
|
cmd = exec.Command("sh", "-c", task.GetData())
|
|
|
|
|
}
|
2021-01-28 22:59:35 -05:00
|
|
|
|
pg.AddProcess(cmd)
|
2021-01-28 21:40:57 -05:00
|
|
|
|
go func() {
|
|
|
|
|
select {
|
|
|
|
|
case <-timeout.C:
|
|
|
|
|
result.Data = "任务执行超时\n"
|
|
|
|
|
close(endCh)
|
2021-01-29 01:29:31 -05:00
|
|
|
|
pg.Dispose()
|
2021-01-28 21:40:57 -05:00
|
|
|
|
case <-endCh:
|
2021-01-29 01:29:31 -05:00
|
|
|
|
timeout.Stop()
|
2021-01-28 20:22:24 -05:00
|
|
|
|
}
|
2021-01-28 21:40:57 -05:00
|
|
|
|
}()
|
|
|
|
|
output, err := cmd.Output()
|
|
|
|
|
if err != nil {
|
|
|
|
|
result.Data += fmt.Sprintf("%s\n%s", string(output), err.Error())
|
|
|
|
|
} else {
|
|
|
|
|
close(endCh)
|
|
|
|
|
result.Data = string(output)
|
2021-01-18 20:59:04 -05:00
|
|
|
|
result.Successful = true
|
|
|
|
|
}
|
2021-04-20 07:30:34 -04:00
|
|
|
|
result.Delay = float32(time.Since(startedAt).Seconds())
|
2021-01-16 05:04:47 -05:00
|
|
|
|
default:
|
2021-04-25 11:46:15 -04:00
|
|
|
|
println("Unknown action: ", task)
|
2019-12-07 05:14:40 -05:00
|
|
|
|
}
|
2021-01-16 05:04:47 -05:00
|
|
|
|
client.ReportTask(ctx, &result)
|
2019-12-09 03:02:49 -05:00
|
|
|
|
}
|
2019-12-05 09:36:58 -05:00
|
|
|
|
|
2019-12-09 03:02:49 -05:00
|
|
|
|
func reportState() {
|
2021-01-13 09:30:28 -05:00
|
|
|
|
var lastReportHostInfo time.Time
|
2019-12-09 03:02:49 -05:00
|
|
|
|
var err error
|
2021-04-25 11:46:15 -04:00
|
|
|
|
defer println("reportState exit", time.Now(), "=>", err)
|
2019-12-09 03:02:49 -05:00
|
|
|
|
for {
|
2020-12-19 10:27:59 -05:00
|
|
|
|
if client != nil {
|
2019-12-09 10:45:23 -05:00
|
|
|
|
monitor.TrackNetworkSpeed()
|
2021-01-15 11:45:49 -05:00
|
|
|
|
_, err = client.ReportSystemState(ctx, monitor.GetState(dao.ReportDelay).PB())
|
2019-12-09 05:14:31 -05:00
|
|
|
|
if err != nil {
|
2021-04-25 11:46:15 -04:00
|
|
|
|
println("reportState error", err)
|
2019-12-09 05:14:31 -05:00
|
|
|
|
time.Sleep(delayWhenError)
|
|
|
|
|
}
|
2021-01-13 09:30:28 -05:00
|
|
|
|
if lastReportHostInfo.Before(time.Now().Add(-10 * time.Minute)) {
|
|
|
|
|
lastReportHostInfo = time.Now()
|
2021-01-15 11:45:49 -05:00
|
|
|
|
client.ReportSystemInfo(ctx, monitor.GetHost().PB())
|
2021-01-13 09:30:28 -05:00
|
|
|
|
}
|
2019-12-05 10:42:20 -05:00
|
|
|
|
}
|
|
|
|
|
}
|
2019-12-05 09:36:58 -05:00
|
|
|
|
}
|
2021-04-25 11:46:15 -04:00
|
|
|
|
|
2021-05-21 22:56:30 -04:00
|
|
|
|
func doSelfUpdate() {
|
|
|
|
|
defer func() {
|
|
|
|
|
time.Sleep(time.Minute * 20)
|
|
|
|
|
updateCh <- struct{}{}
|
|
|
|
|
}()
|
|
|
|
|
v := semver.MustParse(version)
|
|
|
|
|
println("Check update", v)
|
|
|
|
|
latest, err := selfupdate.UpdateSelf(v, "naiba/nezha")
|
|
|
|
|
if err != nil {
|
|
|
|
|
println("Binary update failed:", err)
|
|
|
|
|
return
|
|
|
|
|
}
|
|
|
|
|
if latest.Version.Equals(v) {
|
|
|
|
|
println("Current binary is up to date", version)
|
|
|
|
|
} else {
|
|
|
|
|
println("Upgrade successfully", latest.Version)
|
|
|
|
|
os.Exit(1)
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2021-04-25 11:46:15 -04:00
|
|
|
|
func println(v ...interface{}) {
|
|
|
|
|
if dao.Conf.Debug {
|
|
|
|
|
log.Println(v...)
|
|
|
|
|
}
|
|
|
|
|
}
|