mirror of
https://github.com/fankes/beszel.git
synced 2025-10-19 09:49:28 +08:00
add alert collection and status alerts
This commit is contained in:
88
main.go
88
main.go
@@ -10,6 +10,7 @@ import (
|
|||||||
_ "monitor-site/migrations"
|
_ "monitor-site/migrations"
|
||||||
"net/http"
|
"net/http"
|
||||||
"net/http/httputil"
|
"net/http/httputil"
|
||||||
|
"net/mail"
|
||||||
"net/url"
|
"net/url"
|
||||||
"os"
|
"os"
|
||||||
"strings"
|
"strings"
|
||||||
@@ -23,6 +24,7 @@ import (
|
|||||||
"github.com/pocketbase/pocketbase/models"
|
"github.com/pocketbase/pocketbase/models"
|
||||||
"github.com/pocketbase/pocketbase/plugins/migratecmd"
|
"github.com/pocketbase/pocketbase/plugins/migratecmd"
|
||||||
"github.com/pocketbase/pocketbase/tools/cron"
|
"github.com/pocketbase/pocketbase/tools/cron"
|
||||||
|
"github.com/pocketbase/pocketbase/tools/mailer"
|
||||||
"golang.org/x/crypto/ssh"
|
"golang.org/x/crypto/ssh"
|
||||||
)
|
)
|
||||||
|
|
||||||
@@ -134,25 +136,30 @@ func main() {
|
|||||||
})
|
})
|
||||||
|
|
||||||
// immediately create connection for new servers
|
// immediately create connection for new servers
|
||||||
app.OnRecordAfterCreateRequest("systems").Add(func(e *core.RecordCreateEvent) error {
|
app.OnModelAfterCreate("systems").Add(func(e *core.ModelEvent) error {
|
||||||
go updateServer(e.Record)
|
go updateServer(e.Model.(*models.Record))
|
||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
|
|
||||||
// do things after a systems record is updated
|
// do things after a systems record is updated
|
||||||
app.OnRecordAfterUpdateRequest("systems").Add(func(e *core.RecordUpdateEvent) error {
|
app.OnModelAfterUpdate("systems").Add(func(e *core.ModelEvent) error {
|
||||||
status := e.Record.Get("status")
|
newRecord := e.Model.(*models.Record)
|
||||||
// if server connection exists, close it
|
oldRecord := newRecord.OriginalCopy()
|
||||||
if status == "down" || status == "paused" {
|
newStatus := newRecord.Get("status").(string)
|
||||||
deleteServerConnection(e.Record)
|
|
||||||
|
// if server is disconnected and connection exists, remove it
|
||||||
|
if newStatus == "down" || newStatus == "paused" {
|
||||||
|
deleteServerConnection(newRecord)
|
||||||
}
|
}
|
||||||
|
// alerts
|
||||||
|
handleStatusAlerts(newStatus, oldRecord)
|
||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
|
|
||||||
// do things after a systems record is deleted
|
// do things after a systems record is deleted
|
||||||
app.OnRecordAfterDeleteRequest("systems").Add(func(e *core.RecordDeleteEvent) error {
|
app.OnModelAfterDelete("systems").Add(func(e *core.ModelEvent) error {
|
||||||
// if server connection exists, close it
|
// if server connection exists, close it
|
||||||
deleteServerConnection(e.Record)
|
deleteServerConnection(e.Model.(*models.Record))
|
||||||
return nil
|
return nil
|
||||||
})
|
})
|
||||||
|
|
||||||
@@ -327,6 +334,69 @@ func requestJson(server *Server) (SystemData, error) {
|
|||||||
return systemData, nil
|
return systemData, nil
|
||||||
}
|
}
|
||||||
|
|
||||||
|
func sendAlert(data EmailData) {
|
||||||
|
message := &mailer.Message{
|
||||||
|
From: mail.Address{
|
||||||
|
Address: app.Settings().Meta.SenderAddress,
|
||||||
|
Name: app.Settings().Meta.SenderName,
|
||||||
|
},
|
||||||
|
To: []mail.Address{{Address: data.to}},
|
||||||
|
Subject: data.subj,
|
||||||
|
Text: data.body,
|
||||||
|
}
|
||||||
|
if err := app.NewMailClient().Send(message); err != nil {
|
||||||
|
app.Logger().Error("Failed to send alert: ", "err", err.Error())
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func handleStatusAlerts(newStatus string, oldRecord *models.Record) error {
|
||||||
|
var alertStatus string
|
||||||
|
switch newStatus {
|
||||||
|
case "up":
|
||||||
|
if oldRecord.Get("status") == "down" {
|
||||||
|
alertStatus = "up"
|
||||||
|
}
|
||||||
|
case "down":
|
||||||
|
if oldRecord.Get("status") == "up" {
|
||||||
|
alertStatus = "down"
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if alertStatus == "" {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
alerts, err := app.Dao().FindRecordsByFilter("alerts", "name = 'status' && system = {:system}", "-created", -1, 0, dbx.Params{
|
||||||
|
"system": oldRecord.Get("id")})
|
||||||
|
if err != nil {
|
||||||
|
fmt.Println("failed to get users", "err", err.Error())
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
if len(alerts) == 0 {
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
// expand the user relation
|
||||||
|
if errs := app.Dao().ExpandRecords(alerts, []string{"user"}, nil); len(errs) > 0 {
|
||||||
|
return fmt.Errorf("failed to expand: %v", errs)
|
||||||
|
}
|
||||||
|
systemName := oldRecord.Get("name").(string)
|
||||||
|
emoji := "\U0001F534"
|
||||||
|
if alertStatus == "up" {
|
||||||
|
emoji = "\u2705"
|
||||||
|
}
|
||||||
|
for _, alert := range alerts {
|
||||||
|
user := alert.ExpandedOne("user")
|
||||||
|
if user == nil {
|
||||||
|
continue
|
||||||
|
}
|
||||||
|
// send alert
|
||||||
|
sendAlert(EmailData{
|
||||||
|
to: user.Get("email").(string),
|
||||||
|
subj: fmt.Sprintf("Server %s is %s %v", systemName, alertStatus, emoji),
|
||||||
|
body: fmt.Sprintf("Server %s is %s %v", systemName, alertStatus, emoji),
|
||||||
|
})
|
||||||
|
}
|
||||||
|
return nil
|
||||||
|
}
|
||||||
|
|
||||||
func getSSHKey() ([]byte, error) {
|
func getSSHKey() ([]byte, error) {
|
||||||
// check if the key pair already exists
|
// check if the key pair already exists
|
||||||
existingKey, err := os.ReadFile("./pb_data/id_ed25519")
|
existingKey, err := os.ReadFile("./pb_data/id_ed25519")
|
||||||
|
@@ -251,6 +251,70 @@ func init() {
|
|||||||
"onlyVerified": false,
|
"onlyVerified": false,
|
||||||
"requireEmail": false
|
"requireEmail": false
|
||||||
}
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"id": "elngm8x1l60zi2v",
|
||||||
|
"created": "2024-07-15 01:16:04.044Z",
|
||||||
|
"updated": "2024-07-15 01:19:11.639Z",
|
||||||
|
"name": "alerts",
|
||||||
|
"type": "base",
|
||||||
|
"system": false,
|
||||||
|
"schema": [
|
||||||
|
{
|
||||||
|
"system": false,
|
||||||
|
"id": "hn5ly3vi",
|
||||||
|
"name": "user",
|
||||||
|
"type": "relation",
|
||||||
|
"required": true,
|
||||||
|
"presentable": false,
|
||||||
|
"unique": false,
|
||||||
|
"options": {
|
||||||
|
"collectionId": "_pb_users_auth_",
|
||||||
|
"cascadeDelete": false,
|
||||||
|
"minSelect": null,
|
||||||
|
"maxSelect": 1,
|
||||||
|
"displayFields": null
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"system": false,
|
||||||
|
"id": "g5sl3jdg",
|
||||||
|
"name": "system",
|
||||||
|
"type": "relation",
|
||||||
|
"required": true,
|
||||||
|
"presentable": false,
|
||||||
|
"unique": false,
|
||||||
|
"options": {
|
||||||
|
"collectionId": "2hz5ncl8tizk5nx",
|
||||||
|
"cascadeDelete": false,
|
||||||
|
"minSelect": null,
|
||||||
|
"maxSelect": 1,
|
||||||
|
"displayFields": null
|
||||||
|
}
|
||||||
|
},
|
||||||
|
{
|
||||||
|
"system": false,
|
||||||
|
"id": "zj3ingrv",
|
||||||
|
"name": "name",
|
||||||
|
"type": "select",
|
||||||
|
"required": true,
|
||||||
|
"presentable": false,
|
||||||
|
"unique": false,
|
||||||
|
"options": {
|
||||||
|
"maxSelect": 1,
|
||||||
|
"values": [
|
||||||
|
"status"
|
||||||
|
]
|
||||||
|
}
|
||||||
|
}
|
||||||
|
],
|
||||||
|
"indexes": [],
|
||||||
|
"listRule": "@request.auth.id != \"\" && user.id = @request.auth.id",
|
||||||
|
"viewRule": "",
|
||||||
|
"createRule": "@request.auth.id != \"\" && user.id = @request.auth.id",
|
||||||
|
"updateRule": null,
|
||||||
|
"deleteRule": "@request.auth.id != \"\" && user.id = @request.auth.id",
|
||||||
|
"options": {}
|
||||||
}
|
}
|
||||||
]`
|
]`
|
||||||
|
|
||||||
|
Reference in New Issue
Block a user