mirror of
https://github.com/TeaOSLab/EdgeAdmin.git
synced 2025-11-02 03:40:27 +08:00
405 lines
11 KiB
Go
405 lines
11 KiB
Go
package node
|
|
|
|
import (
|
|
"encoding/json"
|
|
"fmt"
|
|
"time"
|
|
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/utils/numberutils"
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/actionutils"
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/default/clusters/grants/grantutils"
|
|
"github.com/TeaOSLab/EdgeAdmin/internal/web/actions/default/nodes/ipAddresses/ipaddressutils"
|
|
"github.com/TeaOSLab/EdgeCommon/pkg/iputils"
|
|
"github.com/TeaOSLab/EdgeCommon/pkg/nodeconfigs"
|
|
"github.com/TeaOSLab/EdgeCommon/pkg/rpc/pb"
|
|
"github.com/TeaOSLab/EdgeCommon/pkg/serverconfigs"
|
|
"github.com/iwind/TeaGo/maps"
|
|
timeutil "github.com/iwind/TeaGo/utils/time"
|
|
)
|
|
|
|
type DetailAction struct {
|
|
actionutils.ParentAction
|
|
}
|
|
|
|
func (this *DetailAction) Init() {
|
|
this.Nav("", "node", "node")
|
|
this.SecondMenu("nodes")
|
|
}
|
|
|
|
func (this *DetailAction) RunGet(params struct {
|
|
NodeId int64
|
|
ClusterId int64
|
|
}) {
|
|
this.Data["nodeId"] = params.NodeId
|
|
|
|
nodeResp, err := this.RPC().NodeRPC().FindEnabledNode(this.AdminContext(), &pb.FindEnabledNodeRequest{NodeId: params.NodeId})
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
var node = nodeResp.Node
|
|
if node == nil {
|
|
this.WriteString("找不到要操作的节点")
|
|
return
|
|
}
|
|
|
|
// 主集群
|
|
var clusterMap maps.Map = nil
|
|
if node.NodeCluster != nil {
|
|
var clusterId = node.NodeCluster.Id
|
|
clusterResp, err := this.RPC().NodeClusterRPC().FindEnabledNodeCluster(this.AdminContext(), &pb.FindEnabledNodeClusterRequest{NodeClusterId: clusterId})
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
var cluster = clusterResp.NodeCluster
|
|
if cluster != nil {
|
|
clusterMap = maps.Map{
|
|
"id": cluster.Id,
|
|
"name": cluster.Name,
|
|
"installDir": cluster.InstallDir,
|
|
}
|
|
}
|
|
}
|
|
|
|
// 从集群
|
|
var secondaryClustersMaps = []maps.Map{}
|
|
for _, cluster := range node.SecondaryNodeClusters {
|
|
secondaryClustersMaps = append(secondaryClustersMaps, maps.Map{
|
|
"id": cluster.Id,
|
|
"name": cluster.Name,
|
|
"isOn": cluster.IsOn,
|
|
})
|
|
}
|
|
|
|
// 当前访问集群的DNS设置
|
|
clusterDNSInfo, err := this.RPC().NodeClusterRPC().FindEnabledNodeClusterDNS(this.AdminContext(), &pb.FindEnabledNodeClusterDNSRequest{NodeClusterId: params.ClusterId})
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
this.Data["dnsIsExcludingLnNode"] = clusterDNSInfo != nil && !clusterDNSInfo.IncludingLnNodes && node.Level > 1
|
|
|
|
// IP地址
|
|
ipAddressesResp, err := this.RPC().NodeIPAddressRPC().FindAllEnabledNodeIPAddressesWithNodeId(this.AdminContext(), &pb.FindAllEnabledNodeIPAddressesWithNodeIdRequest{
|
|
NodeId: params.NodeId,
|
|
Role: nodeconfigs.NodeRoleNode,
|
|
})
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
var ipAddresses = ipAddressesResp.NodeIPAddresses
|
|
var ipAddressMaps = []maps.Map{}
|
|
for _, addr := range ipAddressesResp.NodeIPAddresses {
|
|
thresholds, err := ipaddressutils.InitNodeIPAddressThresholds(this.Parent(), addr.Id)
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
|
|
// 是否有备用IP
|
|
var originIP = addr.Ip
|
|
if len(addr.BackupIP) > 0 {
|
|
addr.Ip = addr.BackupIP
|
|
}
|
|
|
|
// 专属集群
|
|
var addrClusterMaps = []maps.Map{}
|
|
for _, addrCluster := range addr.NodeClusters {
|
|
addrClusterMaps = append(addrClusterMaps, maps.Map{
|
|
"id": addrCluster.Id,
|
|
"name": addrCluster.Name,
|
|
})
|
|
}
|
|
|
|
ipAddressMaps = append(ipAddressMaps, maps.Map{
|
|
"id": addr.Id,
|
|
"name": addr.Name,
|
|
"originIP": originIP,
|
|
"ip": addr.Ip,
|
|
"canAccess": addr.CanAccess,
|
|
"isOn": addr.IsOn,
|
|
"isUp": addr.IsUp,
|
|
"clusters": addrClusterMaps,
|
|
"thresholds": thresholds,
|
|
})
|
|
}
|
|
|
|
// DNS相关
|
|
var clusters = []*pb.NodeCluster{node.NodeCluster}
|
|
clusters = append(clusters, node.SecondaryNodeClusters...)
|
|
var recordMaps = []maps.Map{}
|
|
var routeMaps = []maps.Map{}
|
|
for _, cluster := range clusters {
|
|
dnsInfoResp, err := this.RPC().NodeRPC().FindEnabledNodeDNS(this.AdminContext(), &pb.FindEnabledNodeDNSRequest{
|
|
NodeId: params.NodeId,
|
|
NodeClusterId: cluster.Id,
|
|
})
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
var dnsInfo = dnsInfoResp.Node
|
|
if len(dnsInfo.DnsDomainName) == 0 || len(dnsInfo.NodeClusterDNSName) == 0 {
|
|
continue
|
|
}
|
|
var domainName = dnsInfo.DnsDomainName
|
|
|
|
// 默认线路
|
|
if len(dnsInfo.Routes) == 0 {
|
|
dnsInfo.Routes = append(dnsInfo.Routes, &pb.DNSRoute{})
|
|
} else {
|
|
for _, route := range dnsInfo.Routes {
|
|
routeMaps = append(routeMaps, maps.Map{
|
|
"domainName": domainName,
|
|
"code": route.Code,
|
|
"name": route.Name,
|
|
})
|
|
}
|
|
}
|
|
|
|
for _, addr := range ipAddresses {
|
|
if !addr.CanAccess || !addr.IsUp || !addr.IsOn {
|
|
continue
|
|
}
|
|
|
|
// 过滤集群
|
|
if len(addr.NodeClusters) > 0 {
|
|
var inCluster = false
|
|
for _, addrCluster := range addr.NodeClusters {
|
|
if addrCluster.Id == cluster.Id {
|
|
inCluster = true
|
|
}
|
|
}
|
|
if !inCluster {
|
|
continue
|
|
}
|
|
}
|
|
|
|
for _, route := range dnsInfo.Routes {
|
|
var recordType = "A"
|
|
if iputils.IsIPv6(addr.Ip) {
|
|
recordType = "AAAA"
|
|
}
|
|
recordMaps = append(recordMaps, maps.Map{
|
|
"name": dnsInfo.NodeClusterDNSName + "." + domainName,
|
|
"type": recordType,
|
|
"route": route.Name,
|
|
"value": addr.Ip,
|
|
"clusterName": cluster.Name,
|
|
"isBackup": dnsInfo.IsBackupForCluster || dnsInfo.IsBackupForGroup,
|
|
"isOffline": dnsInfo.IsOffline,
|
|
})
|
|
}
|
|
}
|
|
}
|
|
|
|
// 登录信息
|
|
var loginMap maps.Map = nil
|
|
if node.NodeLogin != nil {
|
|
loginParams := maps.Map{}
|
|
if len(node.NodeLogin.Params) > 0 {
|
|
err = json.Unmarshal(node.NodeLogin.Params, &loginParams)
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
}
|
|
|
|
var grantMap = maps.Map{}
|
|
var grantId = loginParams.GetInt64("grantId")
|
|
if grantId > 0 {
|
|
grantResp, err := this.RPC().NodeGrantRPC().FindEnabledNodeGrant(this.AdminContext(), &pb.FindEnabledNodeGrantRequest{NodeGrantId: grantId})
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
if grantResp.NodeGrant != nil {
|
|
grantMap = maps.Map{
|
|
"id": grantResp.NodeGrant.Id,
|
|
"name": grantResp.NodeGrant.Name,
|
|
"method": grantResp.NodeGrant.Method,
|
|
"methodName": grantutils.FindGrantMethodName(grantResp.NodeGrant.Method, this.LangCode()),
|
|
"username": grantResp.NodeGrant.Username,
|
|
}
|
|
}
|
|
}
|
|
|
|
loginMap = maps.Map{
|
|
"id": node.NodeLogin.Id,
|
|
"name": node.NodeLogin.Name,
|
|
"type": node.NodeLogin.Type,
|
|
"params": loginParams,
|
|
"grant": grantMap,
|
|
}
|
|
}
|
|
|
|
// 运行状态
|
|
var status = &nodeconfigs.NodeStatus{}
|
|
this.Data["nodeDatetime"] = ""
|
|
this.Data["nodeTimeDiff"] = 0
|
|
if len(node.StatusJSON) > 0 {
|
|
err = json.Unmarshal(node.StatusJSON, &status)
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
status.IsActive = status.IsActive && time.Now().Unix()-status.UpdatedAt <= 60 // N秒之内认为活跃
|
|
|
|
if status.Timestamp > 0 {
|
|
this.Data["nodeDatetime"] = timeutil.FormatTime("Y-m-d H:i:s", status.Timestamp)
|
|
if status.UpdatedAt > 0 {
|
|
var diff = status.UpdatedAt - status.Timestamp
|
|
if diff < 0 {
|
|
diff = -diff
|
|
}
|
|
this.Data["nodeTimeDiff"] = diff
|
|
}
|
|
} else if status.UpdatedAt > 0 {
|
|
this.Data["nodeDatetime"] = timeutil.FormatTime("Y-m-d H:i:s", status.UpdatedAt)
|
|
}
|
|
}
|
|
|
|
// 检查是否有新版本
|
|
if len(status.OS) > 0 {
|
|
checkVersionResp, err := this.RPC().NodeRPC().CheckNodeLatestVersion(this.AdminContext(), &pb.CheckNodeLatestVersionRequest{
|
|
Os: status.OS,
|
|
Arch: status.Arch,
|
|
CurrentVersion: status.BuildVersion,
|
|
})
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
this.Data["shouldUpgrade"] = checkVersionResp.HasNewVersion
|
|
this.Data["newVersion"] = checkVersionResp.NewVersion
|
|
} else {
|
|
this.Data["shouldUpgrade"] = false
|
|
this.Data["newVersion"] = ""
|
|
}
|
|
|
|
// 分组
|
|
var groupMap maps.Map = nil
|
|
if node.NodeGroup != nil {
|
|
groupMap = maps.Map{
|
|
"id": node.NodeGroup.Id,
|
|
"name": node.NodeGroup.Name,
|
|
}
|
|
}
|
|
|
|
// 区域
|
|
var regionMap maps.Map = nil
|
|
if node.NodeRegion != nil {
|
|
regionMap = maps.Map{
|
|
"id": node.NodeRegion.Id,
|
|
"name": node.NodeRegion.Name,
|
|
}
|
|
}
|
|
|
|
// 缓存硬盘 & 内存容量
|
|
var maxCacheDiskCapacity maps.Map
|
|
if node.MaxCacheDiskCapacity != nil {
|
|
maxCacheDiskCapacity = maps.Map{
|
|
"count": node.MaxCacheDiskCapacity.Count,
|
|
"unit": node.MaxCacheDiskCapacity.Unit,
|
|
}
|
|
} else {
|
|
maxCacheDiskCapacity = maps.Map{
|
|
"count": 0,
|
|
"unit": "gb",
|
|
}
|
|
}
|
|
|
|
var maxCacheMemoryCapacity maps.Map
|
|
if node.MaxCacheMemoryCapacity != nil {
|
|
maxCacheMemoryCapacity = maps.Map{
|
|
"count": node.MaxCacheMemoryCapacity.Count,
|
|
"unit": node.MaxCacheMemoryCapacity.Unit,
|
|
}
|
|
} else {
|
|
maxCacheMemoryCapacity = maps.Map{
|
|
"count": 0,
|
|
"unit": "gb",
|
|
}
|
|
}
|
|
|
|
var lnAddrs = node.LnAddrs
|
|
if lnAddrs == nil {
|
|
lnAddrs = []string{}
|
|
}
|
|
|
|
// API节点地址
|
|
var apiNodeAddrStrings = []string{}
|
|
var apiNodeAddrs = []*serverconfigs.NetworkAddressConfig{}
|
|
if len(node.ApiNodeAddrsJSON) > 0 {
|
|
err = json.Unmarshal(node.ApiNodeAddrsJSON, &apiNodeAddrs)
|
|
if err != nil {
|
|
this.ErrorPage(err)
|
|
return
|
|
}
|
|
for _, addr := range apiNodeAddrs {
|
|
if addr.Init() == nil {
|
|
apiNodeAddrStrings = append(apiNodeAddrStrings, addr.FullAddresses()...)
|
|
}
|
|
}
|
|
}
|
|
|
|
this.Data["node"] = maps.Map{
|
|
"id": node.Id,
|
|
"name": node.Name,
|
|
"ipAddresses": ipAddressMaps,
|
|
"cluster": clusterMap,
|
|
"secondaryClusters": secondaryClustersMaps,
|
|
"login": loginMap,
|
|
"installDir": node.InstallDir,
|
|
"isInstalled": node.IsInstalled,
|
|
"uniqueId": node.UniqueId,
|
|
"secret": node.Secret,
|
|
"maxCPU": node.MaxCPU,
|
|
"isOn": node.IsOn,
|
|
"records": recordMaps,
|
|
"routes": routeMaps,
|
|
"level": node.Level,
|
|
"levelInfo": nodeconfigs.FindNodeLevel(int(node.Level)),
|
|
"lnAddrs": lnAddrs,
|
|
"enableIPLists": node.EnableIPLists,
|
|
"apiNodeAddrs": apiNodeAddrStrings,
|
|
"offlineDay": node.OfflineDay,
|
|
"isOffline": len(node.OfflineDay) > 0 && node.OfflineDay < timeutil.Format("Ymd"),
|
|
"isBackupForCluster": node.IsBackupForCluster,
|
|
"isBackupForGroup": node.IsBackupForGroup,
|
|
|
|
"status": maps.Map{
|
|
"isActive": status.IsActive,
|
|
"updatedAt": status.UpdatedAt,
|
|
"hostname": status.Hostname,
|
|
"cpuUsage": status.CPUUsage,
|
|
"cpuUsageText": fmt.Sprintf("%.2f%%", status.CPUUsage*100),
|
|
"memUsage": status.MemoryUsage,
|
|
"memUsageText": fmt.Sprintf("%.2f%%", status.MemoryUsage*100),
|
|
"connectionCount": status.ConnectionCount,
|
|
"buildVersion": status.BuildVersion,
|
|
"cpuPhysicalCount": status.CPUPhysicalCount,
|
|
"cpuLogicalCount": status.CPULogicalCount,
|
|
"load1m": numberutils.FormatFloat2(status.Load1m),
|
|
"load5m": numberutils.FormatFloat2(status.Load5m),
|
|
"load15m": numberutils.FormatFloat2(status.Load15m),
|
|
"cacheTotalDiskSize": numberutils.FormatBytes(status.CacheTotalDiskSize),
|
|
"cacheTotalMemorySize": numberutils.FormatBytes(status.CacheTotalMemorySize),
|
|
"exePath": status.ExePath,
|
|
"apiSuccessPercent": status.APISuccessPercent,
|
|
"apiAvgCostSeconds": status.APIAvgCostSeconds,
|
|
"diskWritingSpeedMB": status.DiskWritingSpeedMB,
|
|
},
|
|
|
|
"group": groupMap,
|
|
"region": regionMap,
|
|
|
|
"maxCacheDiskCapacity": maxCacheDiskCapacity,
|
|
"maxCacheMemoryCapacity": maxCacheMemoryCapacity,
|
|
}
|
|
|
|
this.Show()
|
|
}
|