mirror of
https://github.com/crocofied/CoreControl.git
synced 2025-12-17 15:36:50 +00:00
321 lines
7.4 KiB
Go
321 lines
7.4 KiB
Go
package main
|
||
|
||
import (
|
||
"context"
|
||
"database/sql"
|
||
"fmt"
|
||
"net/http"
|
||
"os"
|
||
"strings"
|
||
"sync"
|
||
"time"
|
||
|
||
_ "github.com/jackc/pgx/v4/stdlib"
|
||
"github.com/joho/godotenv"
|
||
"gopkg.in/gomail.v2"
|
||
)
|
||
|
||
type Application struct {
|
||
ID int
|
||
Name string
|
||
PublicURL string
|
||
Online bool
|
||
}
|
||
|
||
type Notification struct {
|
||
ID int
|
||
Enabled bool
|
||
Type string
|
||
SMTPHost sql.NullString
|
||
SMTPPort sql.NullInt64
|
||
SMTPFrom sql.NullString
|
||
SMTPUser sql.NullString
|
||
SMTPPass sql.NullString
|
||
SMTPSecure sql.NullBool
|
||
SMTPTo sql.NullString
|
||
TelegramChatID sql.NullString
|
||
TelegramToken sql.NullString
|
||
DiscordWebhook sql.NullString
|
||
}
|
||
|
||
var (
|
||
notifications []Notification
|
||
notifMutex sync.RWMutex
|
||
)
|
||
|
||
func main() {
|
||
if err := godotenv.Load(); err != nil {
|
||
fmt.Println("No env vars found")
|
||
}
|
||
|
||
dbURL := os.Getenv("DATABASE_URL")
|
||
if dbURL == "" {
|
||
panic("DATABASE_URL not set")
|
||
}
|
||
|
||
db, err := sql.Open("pgx", dbURL)
|
||
if err != nil {
|
||
panic(fmt.Sprintf("Database connection failed: %v\n", err))
|
||
}
|
||
defer db.Close()
|
||
|
||
// initial load
|
||
notifs, err := loadNotifications(db)
|
||
if err != nil {
|
||
panic(fmt.Sprintf("Failed to load notifications: %v", err))
|
||
}
|
||
notifMutex.Lock()
|
||
notifications = notifMutexCopy(notifs)
|
||
notifMutex.Unlock()
|
||
|
||
// reload notification configs every minute
|
||
go func() {
|
||
reloadTicker := time.NewTicker(time.Minute)
|
||
defer reloadTicker.Stop()
|
||
|
||
for range reloadTicker.C {
|
||
newNotifs, err := loadNotifications(db)
|
||
if err != nil {
|
||
fmt.Printf("Failed to reload notifications: %v\n", err)
|
||
continue
|
||
}
|
||
notifMutex.Lock()
|
||
notifications = notifMutexCopy(newNotifs)
|
||
notifMutex.Unlock()
|
||
fmt.Println("Reloaded notification configurations")
|
||
}
|
||
}()
|
||
|
||
// clean up old entries hourly
|
||
go func() {
|
||
deletionTicker := time.NewTicker(time.Hour)
|
||
defer deletionTicker.Stop()
|
||
|
||
for range deletionTicker.C {
|
||
if err := deleteOldEntries(db); err != nil {
|
||
fmt.Printf("Error deleting old entries: %v\n", err)
|
||
}
|
||
}
|
||
}()
|
||
|
||
ticker := time.NewTicker(time.Second)
|
||
defer ticker.Stop()
|
||
|
||
client := &http.Client{
|
||
Timeout: 4 * time.Second,
|
||
}
|
||
|
||
for now := range ticker.C {
|
||
if now.Second()%10 != 0 {
|
||
continue
|
||
}
|
||
|
||
apps := getApplications(db)
|
||
checkAndUpdateStatus(db, client, apps)
|
||
}
|
||
}
|
||
|
||
// helper to safely copy slice
|
||
func notifMutexCopy(src []Notification) []Notification {
|
||
copyDst := make([]Notification, len(src))
|
||
copy(copyDst, src)
|
||
return copyDst
|
||
}
|
||
|
||
func loadNotifications(db *sql.DB) ([]Notification, error) {
|
||
rows, err := db.Query(
|
||
`SELECT id, enabled, type, "smtpHost", "smtpPort", "smtpFrom", "smtpUser", "smtpPass", "smtpSecure", "smtpTo",
|
||
"telegramChatId", "telegramToken", "discordWebhook"
|
||
FROM notification
|
||
WHERE enabled = true`,
|
||
)
|
||
if err != nil {
|
||
return nil, err
|
||
}
|
||
defer rows.Close()
|
||
|
||
var configs []Notification
|
||
for rows.Next() {
|
||
var n Notification
|
||
if err := rows.Scan(
|
||
&n.ID, &n.Enabled, &n.Type,
|
||
&n.SMTPHost, &n.SMTPPort, &n.SMTPFrom, &n.SMTPUser, &n.SMTPPass, &n.SMTPSecure, &n.SMTPTo,
|
||
&n.TelegramChatID, &n.TelegramToken, &n.DiscordWebhook,
|
||
); err != nil {
|
||
fmt.Printf("Error scanning notification: %v\n", err)
|
||
continue
|
||
}
|
||
configs = append(configs, n)
|
||
}
|
||
return configs, nil
|
||
}
|
||
|
||
func deleteOldEntries(db *sql.DB) error {
|
||
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
||
defer cancel()
|
||
|
||
res, err := db.ExecContext(ctx,
|
||
`DELETE FROM uptime_history WHERE "createdAt" < now() - interval '30 days'`,
|
||
)
|
||
if err != nil {
|
||
return err
|
||
}
|
||
affected, _ := res.RowsAffected()
|
||
fmt.Printf("Deleted %d old entries from uptime_history\n", affected)
|
||
return nil
|
||
}
|
||
|
||
func getApplications(db *sql.DB) []Application {
|
||
rows, err := db.Query(
|
||
`SELECT id, name, "publicURL", online FROM application WHERE "publicURL" IS NOT NULL`,
|
||
)
|
||
if err != nil {
|
||
fmt.Printf("Error fetching applications: %v\n", err)
|
||
return nil
|
||
}
|
||
defer rows.Close()
|
||
|
||
var apps []Application
|
||
for rows.Next() {
|
||
var app Application
|
||
if err := rows.Scan(&app.ID, &app.Name, &app.PublicURL, &app.Online); err != nil {
|
||
fmt.Printf("Error scanning row: %v\n", err)
|
||
continue
|
||
}
|
||
apps = append(apps, app)
|
||
}
|
||
return apps
|
||
}
|
||
|
||
func checkAndUpdateStatus(db *sql.DB, client *http.Client, apps []Application) {
|
||
for _, app := range apps {
|
||
// — HTTP check with proper nil‑guard —
|
||
httpCtx, httpCancel := context.WithTimeout(context.Background(), 4*time.Second)
|
||
req, err := http.NewRequestWithContext(httpCtx, "GET", app.PublicURL, nil)
|
||
if err != nil {
|
||
httpCancel()
|
||
fmt.Printf("Error creating request: %v\n", err)
|
||
continue
|
||
}
|
||
|
||
resp, err := client.Do(req)
|
||
httpCancel()
|
||
|
||
var isOnline bool
|
||
if err == nil {
|
||
isOnline = resp.StatusCode >= 200 && resp.StatusCode < 300 ||
|
||
resp.StatusCode == 405
|
||
resp.Body.Close()
|
||
}
|
||
|
||
// — Notify on change —
|
||
if isOnline != app.Online {
|
||
status := "offline"
|
||
if isOnline {
|
||
status = "online"
|
||
}
|
||
sendNotifications(
|
||
fmt.Sprintf("The application '%s' (%s) went %s!", app.Name, app.PublicURL, status),
|
||
)
|
||
}
|
||
|
||
// — Update DB with its own context —
|
||
dbCtx, dbCancel := context.WithTimeout(context.Background(), 5*time.Second)
|
||
_, err = db.ExecContext(dbCtx,
|
||
`UPDATE application SET online = $1 WHERE id = $2`,
|
||
isOnline, app.ID,
|
||
)
|
||
dbCancel()
|
||
if err != nil {
|
||
fmt.Printf("Update failed for app %d: %v\n", app.ID, err)
|
||
}
|
||
|
||
dbCtx2, dbCancel2 := context.WithTimeout(context.Background(), 5*time.Second)
|
||
_, err = db.ExecContext(dbCtx2,
|
||
`INSERT INTO uptime_history("applicationId", online, "createdAt") VALUES ($1, $2, now())`,
|
||
app.ID, isOnline,
|
||
)
|
||
dbCancel2()
|
||
if err != nil {
|
||
fmt.Printf("Insert into uptime_history failed for app %d: %v\n", app.ID, err)
|
||
}
|
||
}
|
||
}
|
||
|
||
func sendNotifications(message string) {
|
||
notifMutex.RLock()
|
||
notifs := notifMutexCopy(notifications)
|
||
notifMutex.RUnlock()
|
||
|
||
for _, n := range notifs {
|
||
switch n.Type {
|
||
case "email":
|
||
if n.SMTPHost.Valid && n.SMTPTo.Valid {
|
||
sendEmail(n, message)
|
||
}
|
||
case "telegram":
|
||
if n.TelegramToken.Valid && n.TelegramChatID.Valid {
|
||
sendTelegram(n, message)
|
||
}
|
||
case "discord":
|
||
if n.DiscordWebhook.Valid {
|
||
sendDiscord(n, message)
|
||
}
|
||
}
|
||
}
|
||
}
|
||
|
||
func sendEmail(n Notification, body string) {
|
||
// Initialize SMTP dialer with host, port, user, pass
|
||
d := gomail.NewDialer(
|
||
n.SMTPHost.String,
|
||
int(n.SMTPPort.Int64),
|
||
n.SMTPUser.String,
|
||
n.SMTPPass.String,
|
||
)
|
||
if n.SMTPSecure.Valid && n.SMTPSecure.Bool {
|
||
d.SSL = true
|
||
}
|
||
|
||
m := gomail.NewMessage()
|
||
m.SetHeader("From", n.SMTPFrom.String)
|
||
m.SetHeader("To", n.SMTPTo.String)
|
||
m.SetHeader("Subject", "Uptime Notification")
|
||
m.SetBody("text/plain", body)
|
||
|
||
if err := d.DialAndSend(m); err != nil {
|
||
fmt.Printf("Email send failed: %v\n", err)
|
||
}
|
||
}
|
||
|
||
func sendTelegram(n Notification, message string) {
|
||
url := fmt.Sprintf("https://api.telegram.org/bot%s/sendMessage?chat_id=%s&text=%s",
|
||
n.TelegramToken.String,
|
||
n.TelegramChatID.String,
|
||
message,
|
||
)
|
||
resp, err := http.Get(url)
|
||
if err != nil {
|
||
fmt.Printf("Telegram send failed: %v\n", err)
|
||
return
|
||
}
|
||
resp.Body.Close()
|
||
}
|
||
|
||
func sendDiscord(n Notification, message string) {
|
||
payload := fmt.Sprintf(`{"content": "%s"}`, message)
|
||
req, err := http.NewRequest("POST", n.DiscordWebhook.String, strings.NewReader(payload))
|
||
if err != nil {
|
||
fmt.Printf("Discord request creation failed: %v\n", err)
|
||
return
|
||
}
|
||
req.Header.Set("Content-Type", "application/json")
|
||
client := &http.Client{Timeout: 5 * time.Second}
|
||
resp, err := client.Do(req)
|
||
if err != nil {
|
||
fmt.Printf("Discord send failed: %v\n", err)
|
||
return
|
||
}
|
||
resp.Body.Close()
|
||
}
|