Files
EdgeAdmin/internal/tasks/task_sync_api_nodes.go

168 lines
3.5 KiB
Go
Raw Normal View History

2021-02-24 09:00:12 +08:00
package tasks
import (
"context"
"crypto/tls"
2024-07-27 15:42:58 +08:00
"net/url"
"sort"
"strings"
"sync"
"time"
2021-02-24 09:00:12 +08:00
"github.com/TeaOSLab/EdgeAdmin/internal/configs"
2021-07-20 17:15:17 +08:00
teaconst "github.com/TeaOSLab/EdgeAdmin/internal/const"
2021-02-24 09:00:12 +08:00
"github.com/TeaOSLab/EdgeAdmin/internal/events"
2021-12-21 15:18:11 +08:00
"github.com/TeaOSLab/EdgeAdmin/internal/goman"
2021-02-24 09:00:12 +08:00
"github.com/TeaOSLab/EdgeAdmin/internal/rpc"
"github.com/TeaOSLab/EdgeAdmin/internal/setup"
2021-02-24 09:00:12 +08:00
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
"github.com/iwind/TeaGo/Tea"
"github.com/iwind/TeaGo/logs"
"google.golang.org/grpc"
"google.golang.org/grpc/credentials"
2022-09-13 10:48:54 +08:00
"google.golang.org/grpc/credentials/insecure"
2021-02-24 09:00:12 +08:00
)
func init() {
events.On(events.EventStart, func() {
task := NewSyncAPINodesTask()
2021-12-21 15:18:11 +08:00
goman.New(func() {
task.Start()
})
2021-02-24 09:00:12 +08:00
})
}
// SyncAPINodesTask API节点同步任务
2021-02-24 09:00:12 +08:00
type SyncAPINodesTask struct {
}
func NewSyncAPINodesTask() *SyncAPINodesTask {
return &SyncAPINodesTask{}
}
func (this *SyncAPINodesTask) Start() {
ticker := time.NewTicker(5 * time.Minute)
if Tea.IsTesting() {
// 快速测试
ticker = time.NewTicker(1 * time.Minute)
}
for range ticker.C {
err := this.Loop()
if err != nil {
logs.Println("[TASK][SYNC_API_NODES]" + err.Error())
}
}
}
func (this *SyncAPINodesTask) Loop() error {
// 如果还没有安装直接返回
2021-07-20 17:15:17 +08:00
if !setup.IsConfigured() || teaconst.IsRecoverMode {
return nil
}
config, err := configs.LoadAPIConfig()
if err != nil {
return err
}
// 是否禁止自动升级
2023-08-12 17:58:00 +08:00
if config.RPCDisableUpdate {
return nil
}
2021-02-24 09:00:12 +08:00
// 获取所有可用的节点
rpcClient, err := rpc.SharedRPC()
if err != nil {
return err
}
resp, err := rpcClient.APINodeRPC().FindAllEnabledAPINodes(rpcClient.Context(0), &pb.FindAllEnabledAPINodesRequest{})
if err != nil {
return err
}
var newEndpoints = []string{}
2021-11-05 15:34:48 +08:00
for _, node := range resp.ApiNodes {
2021-02-24 09:00:12 +08:00
if !node.IsOn {
continue
}
newEndpoints = append(newEndpoints, node.AccessAddrs...)
}
// 和现有的对比
2023-08-12 17:58:00 +08:00
if this.isSame(newEndpoints, config.RPCEndpoints) {
2021-02-24 09:00:12 +08:00
return nil
}
// 测试是否有API节点可用
hasOk := this.testEndpoints(newEndpoints)
if !hasOk {
return nil
}
2021-02-24 09:00:12 +08:00
// 修改RPC对象配置
2023-08-12 17:58:00 +08:00
config.RPCEndpoints = newEndpoints
2021-02-24 09:00:12 +08:00
err = rpcClient.UpdateConfig(config)
if err != nil {
return err
}
// 保存到文件
2023-08-12 17:58:00 +08:00
err = config.WriteFile(Tea.ConfigFile(configs.ConfigFileName))
2021-02-24 09:00:12 +08:00
if err != nil {
return err
}
return nil
}
func (this *SyncAPINodesTask) isSame(endpoints1 []string, endpoints2 []string) bool {
sort.Strings(endpoints1)
sort.Strings(endpoints2)
return strings.Join(endpoints1, "&") == strings.Join(endpoints2, "&")
}
func (this *SyncAPINodesTask) testEndpoints(endpoints []string) bool {
if len(endpoints) == 0 {
return false
}
var wg = sync.WaitGroup{}
wg.Add(len(endpoints))
var ok = false
for _, endpoint := range endpoints {
go func(endpoint string) {
defer wg.Done()
u, err := url.Parse(endpoint)
if err != nil {
return
}
ctx, cancel := context.WithTimeout(context.Background(), 5*time.Second)
defer func() {
cancel()
}()
var conn *grpc.ClientConn
2022-09-13 10:48:54 +08:00
if u.Scheme == "http" {
2022-09-13 10:48:54 +08:00
conn, err = grpc.DialContext(ctx, u.Host, grpc.WithTransportCredentials(insecure.NewCredentials()), grpc.WithBlock())
} else if u.Scheme == "https" {
conn, err = grpc.DialContext(ctx, u.Host, grpc.WithTransportCredentials(credentials.NewTLS(&tls.Config{
InsecureSkipVerify: true,
})), grpc.WithBlock())
}
if err != nil {
return
}
_ = conn.Close()
ok = true
}(endpoint)
}
wg.Wait()
return ok
}