mirror of
https://github.com/CJackHwang/ds2api.git
synced 2026-05-04 00:15:28 +08:00
103 lines
2.3 KiB
Go
103 lines
2.3 KiB
Go
package main
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net"
|
|
"net/http"
|
|
"os"
|
|
"os/signal"
|
|
"strings"
|
|
"syscall"
|
|
"time"
|
|
|
|
"ds2api/internal/auth"
|
|
"ds2api/internal/config"
|
|
"ds2api/internal/server"
|
|
"ds2api/internal/webui"
|
|
)
|
|
|
|
func main() {
|
|
webui.EnsureBuiltOnStartup()
|
|
_ = auth.AdminKey()
|
|
app := server.NewApp()
|
|
port := strings.TrimSpace(os.Getenv("PORT"))
|
|
if port == "" {
|
|
port = "5001"
|
|
}
|
|
|
|
srv := &http.Server{
|
|
Addr: "0.0.0.0:" + port,
|
|
Handler: app.Router,
|
|
}
|
|
localURL := fmt.Sprintf("http://127.0.0.1:%s", port)
|
|
lanIP := detectLANIPv4()
|
|
lanURL := ""
|
|
if lanIP != "" {
|
|
lanURL = fmt.Sprintf("http://%s:%s", lanIP, port)
|
|
}
|
|
|
|
// Start server in a goroutine so we can listen for shutdown signals.
|
|
go func() {
|
|
if lanURL != "" {
|
|
config.Logger.Info("starting ds2api", "bind", srv.Addr, "port", port, "local_url", localURL, "lan_url", lanURL, "lan_ip", lanIP)
|
|
} else {
|
|
config.Logger.Info("starting ds2api", "bind", srv.Addr, "port", port, "local_url", localURL)
|
|
config.Logger.Warn("lan ip not detected; check active network interfaces")
|
|
}
|
|
if err := srv.ListenAndServe(); err != nil && err != http.ErrServerClosed {
|
|
config.Logger.Error("server stopped unexpectedly", "error", err)
|
|
os.Exit(1)
|
|
}
|
|
}()
|
|
|
|
// Wait for interrupt signal (Ctrl+C / SIGTERM).
|
|
quit := make(chan os.Signal, 1)
|
|
signal.Notify(quit, os.Interrupt, syscall.SIGTERM)
|
|
sig := <-quit
|
|
config.Logger.Info("shutdown signal received", "signal", sig.String())
|
|
|
|
// Graceful shutdown: allow up to 10 seconds for in-flight requests to complete.
|
|
ctx, cancel := context.WithTimeout(context.Background(), 10*time.Second)
|
|
defer cancel()
|
|
|
|
if err := srv.Shutdown(ctx); err != nil {
|
|
config.Logger.Error("graceful shutdown failed, forcing exit", "error", err)
|
|
os.Exit(1)
|
|
}
|
|
config.Logger.Info("server gracefully stopped")
|
|
}
|
|
|
|
func detectLANIPv4() string {
|
|
ifaces, err := net.Interfaces()
|
|
if err != nil {
|
|
return ""
|
|
}
|
|
for _, iface := range ifaces {
|
|
if iface.Flags&net.FlagUp == 0 || iface.Flags&net.FlagLoopback != 0 {
|
|
continue
|
|
}
|
|
addrs, err := iface.Addrs()
|
|
if err != nil {
|
|
continue
|
|
}
|
|
for _, addr := range addrs {
|
|
var ip net.IP
|
|
switch v := addr.(type) {
|
|
case *net.IPNet:
|
|
ip = v.IP
|
|
case *net.IPAddr:
|
|
ip = v.IP
|
|
default:
|
|
continue
|
|
}
|
|
ip = ip.To4()
|
|
if ip == nil || !ip.IsPrivate() {
|
|
continue
|
|
}
|
|
return ip.String()
|
|
}
|
|
}
|
|
return ""
|
|
}
|