- New DB schema: users, channel_members, messages tables (migration 003) - Full C2S HTTP API: register, channels, messages, DMs, polling - Preact SPA embedded via embed.FS, served at GET / - IRC-style UI: tab bar, channel messages, user list, DM tabs, /commands - Dark theme, responsive, esbuild-bundled (~19KB) - Polling-based message delivery (1.5s interval) - Commands: /join, /part, /msg, /nick
95 lines
2.6 KiB
Go
95 lines
2.6 KiB
Go
package server
|
|
|
|
import (
|
|
"io/fs"
|
|
"net/http"
|
|
"time"
|
|
|
|
"git.eeqj.de/sneak/chat/web"
|
|
|
|
sentryhttp "github.com/getsentry/sentry-go/http"
|
|
"github.com/go-chi/chi"
|
|
"github.com/go-chi/chi/middleware"
|
|
"github.com/prometheus/client_golang/prometheus/promhttp"
|
|
"github.com/spf13/viper"
|
|
)
|
|
|
|
const routeTimeout = 60 * time.Second
|
|
|
|
// SetupRoutes configures the HTTP routes and middleware chain.
|
|
func (s *Server) SetupRoutes() {
|
|
s.router = chi.NewRouter()
|
|
|
|
s.router.Use(middleware.Recoverer)
|
|
s.router.Use(middleware.RequestID)
|
|
s.router.Use(s.mw.Logging())
|
|
|
|
if viper.GetString("METRICS_USERNAME") != "" {
|
|
s.router.Use(s.mw.Metrics())
|
|
}
|
|
|
|
s.router.Use(s.mw.CORS())
|
|
s.router.Use(middleware.Timeout(routeTimeout))
|
|
|
|
if s.sentryEnabled {
|
|
sentryHandler := sentryhttp.New(sentryhttp.Options{
|
|
Repanic: true,
|
|
})
|
|
s.router.Use(sentryHandler.Handle)
|
|
}
|
|
|
|
// Health check
|
|
s.router.Get("/.well-known/healthcheck.json", s.h.HandleHealthCheck())
|
|
|
|
// Protected metrics endpoint
|
|
if viper.GetString("METRICS_USERNAME") != "" {
|
|
s.router.Group(func(r chi.Router) {
|
|
r.Use(s.mw.MetricsAuth())
|
|
r.Get("/metrics", http.HandlerFunc(promhttp.Handler().ServeHTTP))
|
|
})
|
|
}
|
|
|
|
// API v1
|
|
s.router.Route("/api/v1", func(r chi.Router) {
|
|
r.Get("/server", s.h.HandleServerInfo())
|
|
r.Post("/register", s.h.HandleRegister())
|
|
r.Get("/me", s.h.HandleMe())
|
|
|
|
// Channels
|
|
r.Get("/channels", s.h.HandleListChannels())
|
|
r.Get("/channels/all", s.h.HandleListAllChannels())
|
|
r.Post("/channels/join", s.h.HandleJoinChannel())
|
|
r.Delete("/channels/{channel}/part", s.h.HandlePartChannel())
|
|
r.Get("/channels/{channel}/members", s.h.HandleChannelMembers())
|
|
r.Get("/channels/{channel}/messages", s.h.HandleGetMessages())
|
|
r.Post("/channels/{channel}/messages", s.h.HandleSendMessage())
|
|
|
|
// DMs
|
|
r.Get("/dm/{nick}/messages", s.h.HandleGetDMs())
|
|
r.Post("/dm/{nick}/messages", s.h.HandleSendDM())
|
|
|
|
// Polling
|
|
r.Get("/poll", s.h.HandlePoll())
|
|
})
|
|
|
|
// Serve embedded SPA
|
|
distFS, err := fs.Sub(web.Dist, "dist")
|
|
if err != nil {
|
|
s.log.Error("failed to get web dist filesystem", "error", err)
|
|
} else {
|
|
fileServer := http.FileServer(http.FS(distFS))
|
|
s.router.Get("/*", func(w http.ResponseWriter, r *http.Request) {
|
|
// Try to serve the file; if not found, serve index.html for SPA routing
|
|
f, err := distFS.(fs.ReadFileFS).ReadFile(r.URL.Path[1:])
|
|
if err != nil || len(f) == 0 {
|
|
indexHTML, _ := distFS.(fs.ReadFileFS).ReadFile("index.html")
|
|
w.Header().Set("Content-Type", "text/html; charset=utf-8")
|
|
w.WriteHeader(http.StatusOK)
|
|
_, _ = w.Write(indexHTML)
|
|
return
|
|
}
|
|
fileServer.ServeHTTP(w, r)
|
|
})
|
|
}
|
|
}
|