监控CPU磁盘网络数据库

This commit is contained in:
TsMask
2023-08-15 21:38:05 +08:00
parent 3672b73683
commit bfd3be9f03
3 changed files with 412 additions and 0 deletions

View File

@@ -0,0 +1,55 @@
package monitor
import "time"
type MonitorBase struct {
ID uint `xorm:"id" json:"id"`
CreatedAt time.Time `xorm:"created_at" json:"createdAt"`
UpdatedAt time.Time `xorm:"updated_at" json:"updatedAt"`
Cpu float64 `xorm:"cpu" json:"cpu"`
LoadUsage float64 `xorm:"load_usage" json:"loadUsage"`
CpuLoad1 float64 `xorm:"cpu_load1" json:"cpuLoad1"`
CpuLoad5 float64 `xorm:"cpu_load5" json:"cpuLoad5"`
CpuLoad15 float64 `xorm:"cpu_load15" json:"cpuLoad15"`
Memory float64 `xorm:"memory" json:"memory"`
DbSize uint `xorm:"db_size" json:"dbSize"`
}
type MonitorIO struct {
ID uint `xorm:"id" json:"id"`
CreatedAt time.Time `xorm:"created_at" json:"createdAt"`
UpdatedAt time.Time `xorm:"updated_at" json:"updatedAt"`
Name string `xorm:"name" json:"name"`
Read uint64 `xorm:"read" json:"read"`
Write uint64 `xorm:"write" json:"write"`
Count uint64 `xorm:"count" json:"count"`
Time uint64 `xorm:"time" json:"time"`
}
type MonitorNetwork struct {
ID uint `xorm:"id" json:"id"`
CreatedAt time.Time `xorm:"created_at" json:"createdAt"`
UpdatedAt time.Time `xorm:"updated_at" json:"updatedAt"`
Name string `xorm:"name" json:"name"`
Up float64 `xorm:"up" json:"up"`
Down float64 `xorm:"down" json:"down"`
}
type MonitorSearch struct {
Param string `json:"param" validate:"required,oneof=all cpu memory load io network"`
Info string `json:"info"`
StartTime time.Time `json:"startTime"`
EndTime time.Time `json:"endTime"`
}
type MonitorData struct {
Param string `json:"param" validate:"required,oneof=cpu memory load io network"`
Date []time.Time `json:"date"`
Value []interface{} `json:"value"`
}

View File

@@ -0,0 +1,124 @@
package monitor
import (
"net/http"
"sort"
"time"
"ems.agt/lib/dborm"
"ems.agt/lib/services"
"ems.agt/restagent/config"
"github.com/shirou/gopsutil/disk"
"github.com/shirou/gopsutil/net"
)
// 点击【主机 - 监控】菜单进入监控报表直观的了解服务器的运行状态包含【平均负载】、【CPU性能监控】、【内存使用监控】、【磁盘IO监控】、【网络IO监控】
// 可以查看昨天今天最近7天最近30天自定义时间的监控指标情况。
// 默认监控是开启的,可以在【面板设置 - 监控】页面中根据需求对监控进行开启和关闭。
// 监控数据默认保存30天可以自行修改也可手动清理该日志。
var (
// 可选网络
UriNetOpt = config.UriPrefix + "/monitor/{apiVersion}/monitor/netoptions"
// 可选磁盘
UriIoOpt = config.UriPrefix + "/monitor/{apiVersion}/monitor/iooptions"
// 加载
UriLoad = config.UriPrefix + "/monitor/{apiVersion}/monitor/load"
)
// Netoptions 可选网络
func Netoptions(w http.ResponseWriter, r *http.Request) {
netStat, _ := net.IOCounters(true)
var options []string
options = append(options, "all")
for _, net := range netStat {
options = append(options, net.Name)
}
sort.Strings(options)
services.ResponseWithJson(w, 200, options)
}
// Iooptions 可选磁盘
func Iooptions(w http.ResponseWriter, r *http.Request) {
diskStat, _ := disk.IOCounters()
var options []string
options = append(options, "all")
for _, net := range diskStat {
options = append(options, net.Name)
}
sort.Strings(options)
services.ResponseWithJson(w, 200, options)
}
// LoadMonitor 载入监控
func LoadMonitor(w http.ResponseWriter, r *http.Request) {
// json 請求參數獲取
var bodyArgs MonitorSearch
err := services.ShouldBindJSON(r, &bodyArgs)
if err != nil || dborm.DbClient.XEngine == nil {
services.ResponseErrorWithJson(w, 400, err.Error())
return
}
loc := time.Now().Location()
bodyArgs.StartTime = bodyArgs.StartTime.In(loc)
bodyArgs.EndTime = bodyArgs.EndTime.In(loc)
var backdatas []MonitorData
if bodyArgs.Param == "all" || bodyArgs.Param == "cpu" || bodyArgs.Param == "memory" || bodyArgs.Param == "load" {
var bases []MonitorBase
err := dborm.DbClient.XEngine.SQL("SELECT * FROM monitor_base").
Where("created_at > ? AND created_at < ?", bodyArgs.StartTime, bodyArgs.EndTime).
Find(&bases)
if err != nil {
services.ResponseErrorWithJson(w, 400, err.Error())
return
}
var itemData MonitorData
itemData.Param = "base"
for _, base := range bases {
itemData.Date = append(itemData.Date, base.CreatedAt)
itemData.Value = append(itemData.Value, base)
}
backdatas = append(backdatas, itemData)
}
if bodyArgs.Param == "all" || bodyArgs.Param == "io" {
var bases []MonitorIO
err := dborm.DbClient.XEngine.SQL("SELECT * FROM monitor_io").
Where("created_at > ? AND created_at < ?", bodyArgs.StartTime, bodyArgs.EndTime).
Find(&bases)
if err != nil {
services.ResponseErrorWithJson(w, 400, err.Error())
return
}
var itemData MonitorData
itemData.Param = "io"
for _, base := range bases {
itemData.Date = append(itemData.Date, base.CreatedAt)
itemData.Value = append(itemData.Value, base)
}
backdatas = append(backdatas, itemData)
}
if bodyArgs.Param == "all" || bodyArgs.Param == "network" {
var bases []MonitorNetwork
err := dborm.DbClient.XEngine.SQL("SELECT * FROM monitor_network").
Where("name = ? AND created_at > ? AND created_at < ?", bodyArgs.Info, bodyArgs.StartTime, bodyArgs.EndTime).
Find(&bases)
if err != nil {
services.ResponseErrorWithJson(w, 400, err.Error())
return
}
var itemData MonitorData
itemData.Param = "network"
for _, base := range bases {
itemData.Date = append(itemData.Date, base.CreatedAt)
itemData.Value = append(itemData.Value, base)
}
backdatas = append(backdatas, itemData)
}
services.ResponseWithJson(w, 200, backdatas)
}

View File

@@ -0,0 +1,233 @@
package monitor
import (
"encoding/json"
"fmt"
"strconv"
"time"
"ems.agt/lib/dborm"
"ems.agt/lib/log"
"ems.agt/restagent/config"
"github.com/robfig/cron/v3"
"github.com/shirou/gopsutil/v3/cpu"
"github.com/shirou/gopsutil/v3/disk"
"github.com/shirou/gopsutil/v3/load"
"github.com/shirou/gopsutil/v3/mem"
"github.com/shirou/gopsutil/v3/net"
)
type MonitorService struct{}
type IMonitorService interface {
Run()
}
func NewIMonitorService() IMonitorService {
return &MonitorService{}
}
func (m *MonitorService) Run() {
// monitorStatus, _ := dborm.XormGetConfig("SystemMonitor", "MonitorStatus")
// if monitorStatus["value"] == "disable" {
// return
// }
var itemModel MonitorBase
itemModel.CreatedAt = time.Now()
itemModel.UpdatedAt = time.Now()
totalPercent, _ := cpu.Percent(3*time.Second, false)
if len(totalPercent) == 1 {
itemModel.Cpu = totalPercent[0]
}
cpuCount, _ := cpu.Counts(false)
loadInfo, _ := load.Avg()
itemModel.CpuLoad1 = loadInfo.Load1
itemModel.CpuLoad5 = loadInfo.Load5
itemModel.CpuLoad15 = loadInfo.Load15
itemModel.LoadUsage = loadInfo.Load1 / (float64(cpuCount*2) * 0.75) * 100
memoryInfo, _ := mem.VirtualMemory()
itemModel.Memory = memoryInfo.UsedPercent
var dataSize int
conf := config.GetYamlConfig()
result, err := dborm.DbClient.XEngine.QueryString("SELECT SUM(data_length) AS data_size FROM information_schema.tables WHERE TABLE_SCHEMA = ?;", conf.Database.Name)
if err != nil {
dataSize = 0
} else {
v, _ := strconv.Atoi(result[0]["data_size"])
dataSize = v
}
itemModel.DbSize = uint(dataSize)
_, errx := dborm.DbClient.XEngine.Table("monitor_base").Insert(itemModel)
if errx != nil {
log.Errorf("Insert basic monitoring data failed, err: %v", err)
}
go loadDiskIO()
go loadNetIO()
// 删除保留的记录
// monitorStoreDays, _ := dborm.XormGetConfig("SystemMonitor", "MonitorStoreDays")
// if monitorStoreDays["value"] != "" {
// return
// }
// storeDays, err := strconv.Atoi(MonitorStoreDays.Value)
// if err != nil {
// timeForDelete := time.Now().AddDate(0, 0, -storeDays)
// DelMonitorBase(timeForDelete)
// DelMonitorIO(timeForDelete)
// DelMonitorNet(timeForDelete)
// }
}
func loadDiskIO() {
ioStat, _ := disk.IOCounters()
time.Sleep(60 * time.Second)
ioStat2, _ := disk.IOCounters()
var ioList []MonitorIO
for _, io2 := range ioStat2 {
for _, io1 := range ioStat {
if io2.Name == io1.Name {
var itemIO MonitorIO
itemIO.CreatedAt = time.Now()
itemIO.UpdatedAt = time.Now()
itemIO.Name = io1.Name
if io2.ReadBytes != 0 && io1.ReadBytes != 0 && io2.ReadBytes > io1.ReadBytes {
itemIO.Read = uint64(float64(io2.ReadBytes-io1.ReadBytes) / 60)
}
if io2.WriteBytes != 0 && io1.WriteBytes != 0 && io2.WriteBytes > io1.WriteBytes {
itemIO.Write = uint64(float64(io2.WriteBytes-io1.WriteBytes) / 60)
}
if io2.ReadCount != 0 && io1.ReadCount != 0 && io2.ReadCount > io1.ReadCount {
itemIO.Count = uint64(float64(io2.ReadCount-io1.ReadCount) / 60)
}
writeCount := uint64(0)
if io2.WriteCount != 0 && io1.WriteCount != 0 && io2.WriteCount > io1.WriteCount {
writeCount = uint64(float64(io2.WriteCount-io1.WriteCount) / 60)
}
if writeCount > itemIO.Count {
itemIO.Count = writeCount
}
if io2.ReadTime != 0 && io1.ReadTime != 0 && io2.ReadTime > io1.ReadTime {
itemIO.Time = uint64(float64(io2.ReadTime-io1.ReadTime) / 60)
}
writeTime := uint64(0)
if io2.WriteTime != 0 && io1.WriteTime != 0 && io2.WriteTime > io1.WriteTime {
writeTime = uint64(float64(io2.WriteTime-io1.WriteTime) / 60)
}
if writeTime > itemIO.Time {
itemIO.Time = writeTime
}
ioList = append(ioList, itemIO)
break
}
}
}
_, err := dborm.DbClient.XEngine.Table("monitor_io").Insert(ioList)
if err != nil {
log.Errorf("Insert io monitoring data failed, err: %v", err)
}
}
func loadNetIO() {
netStat, _ := net.IOCounters(true)
netStatAll, _ := net.IOCounters(false)
time.Sleep(60 * time.Second)
netStat2, _ := net.IOCounters(true)
var netList []MonitorNetwork
for _, net2 := range netStat2 {
for _, net1 := range netStat {
if net2.Name == net1.Name {
var itemNet MonitorNetwork
itemNet.CreatedAt = time.Now()
itemNet.UpdatedAt = time.Now()
itemNet.Name = net1.Name
if net2.BytesSent != 0 && net1.BytesSent != 0 && net2.BytesSent > net1.BytesSent {
itemNet.Up = float64(net2.BytesSent-net1.BytesSent) / 1024 / 60
}
if net2.BytesRecv != 0 && net1.BytesRecv != 0 && net2.BytesRecv > net1.BytesRecv {
itemNet.Down = float64(net2.BytesRecv-net1.BytesRecv) / 1024 / 60
}
netList = append(netList, itemNet)
break
}
}
}
netStatAll2, _ := net.IOCounters(false)
for _, net2 := range netStatAll2 {
for _, net1 := range netStatAll {
if net2.Name == net1.Name {
var itemNet MonitorNetwork
itemNet.Name = net1.Name
if net2.BytesSent != 0 && net1.BytesSent != 0 && net2.BytesSent > net1.BytesSent {
itemNet.Up = float64(net2.BytesSent-net1.BytesSent) / 1024 / 60
}
if net2.BytesRecv != 0 && net1.BytesRecv != 0 && net2.BytesRecv > net1.BytesRecv {
itemNet.Down = float64(net2.BytesRecv-net1.BytesRecv) / 1024 / 60
}
netList = append(netList, itemNet)
break
}
}
}
rows, err := dborm.DbClient.XEngine.Table("monitor_network").Insert(netList)
if err != nil {
log.Errorf("Insert network monitoring data failed, err: %v", err)
}
fmt.Println(rows, err)
}
var c *cron.Cron
var monitorCronID int
func init() {
c = cron.New()
monitorCronID = 0
}
// StartMonitor 开始监控任务 removeBefore删除上次任务间隔interval分钟
func StartMonitor(removeBefore bool, interval string) error {
if removeBefore {
c.Remove(cron.EntryID(monitorCronID))
}
// 读取配置
if interval == "" {
v, err := dborm.XormGetConfig("SystemMonitor", "sampleTime")
if err != nil {
return err
}
data := make(map[string]any)
err = json.Unmarshal([]byte(v["value_json"].(string)), &data)
if err != nil {
log.Error("json StartMonitor:%s", err.Error())
return err
}
interval = data["sampleTime"].(string)
}
imservice := NewIMonitorService()
monitorID, err := c.AddJob(fmt.Sprintf("@every %sm", interval), imservice)
if err != nil {
return err
}
imservice.Run()
monitorCronID = int(monitorID)
return nil
}