Files
lst/controller/main.go

188 lines
4.8 KiB
Go

package main
import (
"fmt"
"log"
"net/http"
"os"
"path/filepath"
"strings"
"time"
"github.com/gin-gonic/gin"
socketio "github.com/googollee/go-socket.io"
"github.com/joho/godotenv"
"lst.net/internal/bot"
"lst.net/pkg"
)
func main() {
exePath, _ := os.Executable()
exeDir := filepath.Dir(exePath)
if err := godotenv.Load(filepath.Join(exeDir, ".env")); err != nil {
// fallback dev path
_ = godotenv.Load("../.env")
}
// gin stuff
basePath := "/api/controller"
if os.Getenv("APP_MODE") != "production" {
basePath = "/lst/api/controller"
}
port := os.Getenv("PORT")
if port == "" {
port = "8080"
}
server := socketio.NewServer(nil)
r := Setup(basePath, server) // returns *gin.Engine
server.OnConnect("/", func(s socketio.Conn) error {
fmt.Println("✅ Client connected:", s.ID())
return nil
})
// ROOM SUBSCRIBE
server.OnEvent("/", "subscribe:logs", func(s socketio.Conn) {
s.Join("logs")
fmt.Println("📺", s.ID(), "joined logs")
s.Emit("info", "Subscribed to logs")
})
server.OnEvent("/", "unsubscribe:logs", func(s socketio.Conn) {
s.Leave("logs")
fmt.Println("👋", s.ID(), "left logs")
s.Emit("info", "Unsubscribed from logs")
})
server.OnEvent("/", "subscribe:errors", func(s socketio.Conn) {
s.Join("errors")
fmt.Println("📺", s.ID(), "joined errors")
s.Emit("info", "Subscribed to errors")
})
server.OnEvent("/", "unsubscribe:errors", func(s socketio.Conn) {
s.Leave("errors")
fmt.Println("👋", s.ID(), "left errors")
s.Emit("info", "Unsubscribed from errors")
})
server.OnDisconnect("/", func(s socketio.Conn, reason string) {
fmt.Println("❌ Client disconnected:", s.ID(), reason)
})
// build stuff
// Subscribe to build room
server.OnEvent("/", "subscribe:build", func(s socketio.Conn) {
s.Join("build")
fmt.Println("📺", s.ID(), "joined build room")
s.Emit("info", "Subscribed to build log room")
})
server.OnEvent("/", "unsubscribe:build", func(s socketio.Conn) {
s.Leave("build")
fmt.Println("👋", s.ID(), "left build room")
s.Emit("info", "Unsubscribed from build log room")
})
registerBuildChannel(server)
registerUpdateChannel(server)
// Broadcast logs to room
// go func() {
// for i := 0; ; i++ {
// time.Sleep(2 * time.Second)
// msg := fmt.Sprintf("Log line %d @ %s", i, time.Now().Format(time.RFC3339))
// server.BroadcastToRoom("/", "logs", "logs", msg)
// }
// }()
// go pkg.JobScheduler(server, "0 9 * * *", func() {
// fmt.Println("This is the time checker")
// server.BroadcastToRoom("/", "logs", "logs", "Just ran at 9:00 for example")
// })
// schedule the update checker
go pkg.JobScheduler(server, "0 8 * * 1-4", func() {
host, err := os.Hostname()
if err != nil {
server.BroadcastToRoom("/", "update", "updateLogs", "Could not retrieve hostname")
return
}
if strings.Contains(host, "VMS") || strings.Contains(host, "vms") {
fmt.Println("On a server ok to start the update job")
server.BroadcastToRoom("/", "update", "updateLogs", "On a server ok to start the update job")
updates := UpdateApp(server)
fmt.Println("📦 Starting update loop")
for msg := range updates { // this will block until the channel closes
fmt.Println(msg)
server.BroadcastToRoom("/", "logs", "logs", msg)
}
fmt.Println("✅ Update process finished")
}
})
// Broadcast errors to room
go func() {
for i := 0; ; i++ {
time.Sleep(5 * time.Second)
msg := fmt.Sprintf("Error #%d @ %s", i, time.Now().Format(time.RFC3339))
server.BroadcastToRoom("/", "errors", "errors", msg)
}
}()
// only activate the bot if its on VMS036 or my dev server
host, err := os.Hostname()
if err != nil {
server.BroadcastToRoom("/", "update", "updateLogs", "Could not retrieve hostname")
return
}
if host == os.Getenv("PRIMARY_SERVER") {
go bot.TheBot()
}
go server.Serve()
defer server.Close()
// mount socket.io on /socket.io/*
r.Any("/socket.io/*any", gin.WrapH(withCORS(server)))
// mount a static dir (like http.FileServer)
//r.Static("/", "./static")
fmt.Println("🚀 Server running on :" + port)
if err := r.Run(":" + port); err != nil {
log.Fatal("Server failed:", err)
}
}
// Reuse your proper CORS handler
func withCORS(h http.Handler) http.Handler {
return http.HandlerFunc(func(w http.ResponseWriter, r *http.Request) {
origin := r.Header.Get("Origin")
if origin != "" {
// 🔑 Echo the request Origin, not "*"
w.Header().Set("Access-Control-Allow-Origin", origin)
w.Header().Set("Vary", "Origin")
}
w.Header().Set("Access-Control-Allow-Methods", "GET,POST,OPTIONS")
w.Header().Set("Access-Control-Allow-Headers", "Content-Type, Authorization")
w.Header().Set("Access-Control-Allow-Credentials", "true")
if r.Method == http.MethodOptions {
w.WriteHeader(http.StatusNoContent)
return
}
h.ServeHTTP(w, r)
})
}