mirror of
https://github.com/idanoo/autobrr
synced 2025-07-22 16:29:12 +00:00

* feat(lists): integrate Omegabrr * feat(lists): add missing lists index * feat(lists): add db repo * feat(lists): add db migrations * feat(lists): labels * feat(lists): url lists and more arrs * fix(lists): db migrations client_id wrong type * fix(lists): db fields * feat(lists): create list form wip * feat(lists): show in list and create * feat(lists): update and delete * feat(lists): trigger via webhook * feat(lists): add webhook handler * fix(arr): encode json to pointer * feat(lists): rename endpoint to lists * feat(lists): fetch tags from arr * feat(lists): process plaintext lists * feat(lists): add background refresh job * run every 6th hour with a random start delay between 1-35 seconds * feat(lists): refresh on save and improve logging * feat(lists): cast arr client to pointer * feat(lists): improve error handling * feat(lists): reset shows field with match release * feat(lists): filter opts all lists * feat(lists): trigger on update if enabled * feat(lists): update option for lists * feat(lists): show connected filters in list * feat(lists): missing listSvc dep * feat(lists): cleanup * feat(lists): typo arr list * feat(lists): radarr include original * feat(lists): rename ExcludeAlternateTitle to IncludeAlternateTitle * fix(lists): arr client type conversion to pointer * fix(actions): only log panic recover if err not nil * feat(lists): show spinner on save * feat(lists): show icon in filters list * feat(lists): change icon color in filters list * feat(lists): delete relations on filter delete
213 lines
6.8 KiB
Go
213 lines
6.8 KiB
Go
// Copyright (c) 2021 - 2024, Ludvig Lundgren and the autobrr contributors.
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
package http
|
|
|
|
import (
|
|
"fmt"
|
|
"net"
|
|
"net/http"
|
|
"time"
|
|
|
|
"github.com/autobrr/autobrr/internal/config"
|
|
"github.com/autobrr/autobrr/internal/database"
|
|
"github.com/autobrr/autobrr/internal/logger"
|
|
"github.com/autobrr/autobrr/web"
|
|
|
|
"github.com/go-chi/chi/v5"
|
|
"github.com/go-chi/chi/v5/middleware"
|
|
"github.com/gorilla/sessions"
|
|
"github.com/r3labs/sse/v2"
|
|
"github.com/rs/cors"
|
|
"github.com/rs/zerolog"
|
|
)
|
|
|
|
type Server struct {
|
|
log zerolog.Logger
|
|
sse *sse.Server
|
|
db *database.DB
|
|
|
|
config *config.AppConfig
|
|
cookieStore *sessions.CookieStore
|
|
|
|
version string
|
|
commit string
|
|
date string
|
|
|
|
actionService actionService
|
|
apiService apikeyService
|
|
authService authService
|
|
downloadClientService downloadClientService
|
|
filterService filterService
|
|
feedService feedService
|
|
indexerService indexerService
|
|
ircService ircService
|
|
listService listService
|
|
notificationService notificationService
|
|
proxyService proxyService
|
|
releaseService releaseService
|
|
updateService updateService
|
|
}
|
|
|
|
func NewServer(log logger.Logger, config *config.AppConfig, sse *sse.Server, db *database.DB, version string, commit string, date string, actionService actionService, apiService apikeyService, authService authService, downloadClientSvc downloadClientService, filterSvc filterService, feedSvc feedService, indexerSvc indexerService, ircSvc ircService, listSvc listService, notificationSvc notificationService, proxySvc proxyService, releaseSvc releaseService, updateSvc updateService) Server {
|
|
return Server{
|
|
log: log.With().Str("module", "http").Logger(),
|
|
config: config,
|
|
sse: sse,
|
|
db: db,
|
|
version: version,
|
|
commit: commit,
|
|
date: date,
|
|
|
|
cookieStore: sessions.NewCookieStore([]byte(config.Config.SessionSecret)),
|
|
|
|
actionService: actionService,
|
|
apiService: apiService,
|
|
authService: authService,
|
|
downloadClientService: downloadClientSvc,
|
|
filterService: filterSvc,
|
|
feedService: feedSvc,
|
|
indexerService: indexerSvc,
|
|
ircService: ircSvc,
|
|
listService: listSvc,
|
|
notificationService: notificationSvc,
|
|
proxyService: proxySvc,
|
|
releaseService: releaseSvc,
|
|
updateService: updateSvc,
|
|
}
|
|
}
|
|
|
|
func (s Server) Open() error {
|
|
addr := fmt.Sprintf("%v:%v", s.config.Config.Host, s.config.Config.Port)
|
|
|
|
var err error
|
|
for _, proto := range []string{"tcp", "tcp4", "tcp6"} {
|
|
if err = s.tryToServe(addr, proto); err == nil {
|
|
break
|
|
}
|
|
|
|
s.log.Error().Err(err).Msgf("Failed to start %s server. Attempted to listen on %s", proto, addr)
|
|
}
|
|
|
|
return err
|
|
}
|
|
|
|
func (s Server) tryToServe(addr, protocol string) error {
|
|
listener, err := net.Listen(protocol, addr)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
s.log.Info().Msgf("Starting API %s server. Listening on %s", protocol, listener.Addr().String())
|
|
|
|
server := http.Server{
|
|
Handler: s.Handler(),
|
|
ReadHeaderTimeout: time.Second * 15,
|
|
}
|
|
|
|
return server.Serve(listener)
|
|
}
|
|
|
|
func (s Server) Handler() http.Handler {
|
|
r := chi.NewRouter()
|
|
|
|
r.Use(middleware.RequestID)
|
|
r.Use(middleware.RealIP)
|
|
r.Use(middleware.Recoverer)
|
|
r.Use(LoggerMiddleware(&s.log))
|
|
|
|
c := cors.New(cors.Options{
|
|
AllowCredentials: true,
|
|
AllowedMethods: []string{"HEAD", "OPTIONS", "GET", "POST", "PUT", "PATCH", "DELETE"},
|
|
AllowOriginFunc: func(origin string) bool { return true },
|
|
OptionsPassthrough: true,
|
|
// Enable Debugging for testing, consider disabling in production
|
|
Debug: false,
|
|
})
|
|
|
|
r.Use(c.Handler)
|
|
|
|
encoder := newEncoder(s.log)
|
|
|
|
// Create a separate router for API
|
|
apiRouter := chi.NewRouter()
|
|
apiRouter.Route("/auth", newAuthHandler(encoder, s.log, s, s.config.Config, s.cookieStore, s.authService).Routes)
|
|
apiRouter.Route("/healthz", newHealthHandler(encoder, s.db).Routes)
|
|
apiRouter.Group(func(r chi.Router) {
|
|
r.Group(func(r chi.Router) {
|
|
r.Use(s.IsAuthenticated)
|
|
|
|
r.Route("/actions", newActionHandler(encoder, s.actionService).Routes)
|
|
r.Route("/config", newConfigHandler(encoder, s, s.config).Routes)
|
|
r.Route("/download_clients", newDownloadClientHandler(encoder, s.downloadClientService).Routes)
|
|
r.Route("/filters", newFilterHandler(encoder, s.filterService).Routes)
|
|
r.Route("/feeds", newFeedHandler(encoder, s.feedService).Routes)
|
|
r.Route("/irc", newIrcHandler(encoder, s.sse, s.ircService).Routes)
|
|
r.Route("/indexer", newIndexerHandler(encoder, s.indexerService, s.ircService).Routes)
|
|
r.Route("/lists", newListHandler(encoder, s.listService).Routes)
|
|
r.Route("/keys", newAPIKeyHandler(encoder, s.apiService).Routes)
|
|
r.Route("/logs", newLogsHandler(s.config).Routes)
|
|
r.Route("/notification", newNotificationHandler(encoder, s.notificationService).Routes)
|
|
r.Route("/proxy", newProxyHandler(encoder, s.proxyService).Routes)
|
|
r.Route("/release", newReleaseHandler(encoder, s.releaseService).Routes)
|
|
r.Route("/updates", newUpdateHandler(encoder, s.updateService).Routes)
|
|
r.Route("/webhook", newWebhookHandler(encoder, s.listService).Routes)
|
|
|
|
r.HandleFunc("/events", func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// inject CORS headers to bypass checks
|
|
s.sse.Headers = map[string]string{
|
|
"Content-Type": "text/event-stream",
|
|
"Cache-Control": "no-cache",
|
|
"Connection": "keep-alive",
|
|
"X-Accel-Buffering": "no",
|
|
}
|
|
|
|
s.sse.ServeHTTP(w, r)
|
|
})
|
|
})
|
|
})
|
|
|
|
routeBaseURL := "/"
|
|
|
|
webRouter := chi.NewRouter()
|
|
|
|
// handle backwards compatibility for base url routing
|
|
if s.config.Config.BaseURLModeLegacy {
|
|
// this is required to keep assets "url rewritable" via a reverse-proxy
|
|
routeAssetBaseURL := "./"
|
|
// serve the web
|
|
webHandlers := newWebLegacyHandler(s.log, web.DistDirFS, s.version, s.config.Config.BaseURL, routeAssetBaseURL)
|
|
webHandlers.RegisterRoutes(webRouter)
|
|
} else {
|
|
routeBaseURL = s.config.Config.BaseURL
|
|
|
|
// serve the web
|
|
webHandlers := newWebHandler(s.log, web.DistDirFS, s.version, routeBaseURL, routeBaseURL)
|
|
webHandlers.RegisterRoutes(webRouter)
|
|
|
|
// add fallback routes when base url is set to inform user to redirect and use /baseurl/
|
|
r.Get("/", func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusNotFound)
|
|
if err := webHandlers.RenderFallbackIndex(w); err != nil {
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
})
|
|
|
|
r.Get("/*", func(w http.ResponseWriter, r *http.Request) {
|
|
w.WriteHeader(http.StatusNotFound)
|
|
if err := webHandlers.RenderFallbackIndex(w); err != nil {
|
|
http.Error(w, err.Error(), http.StatusInternalServerError)
|
|
return
|
|
}
|
|
})
|
|
}
|
|
|
|
// Mount the web router under baseUrl + '/'
|
|
r.Mount(routeBaseURL, webRouter)
|
|
// Mount the API router under baseUrl + '/api'
|
|
r.Mount(routeBaseURL+"api", apiRouter)
|
|
|
|
return r
|
|
}
|