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

* chore: update copyright year in license headers * Revert "chore: update copyright year in license headers" This reverts commit 3e58129c431b9a491089ce36b908f9bb6ba38ed3. * chore: update copyright year in license headers * fix: sort go imports * fix: add missing license headers
257 lines
6.3 KiB
Go
257 lines
6.3 KiB
Go
// Copyright (c) 2021 - 2025, Ludvig Lundgren and the autobrr contributors.
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
package http
|
|
|
|
import (
|
|
"context"
|
|
"encoding/json"
|
|
"fmt"
|
|
"net/http"
|
|
"strconv"
|
|
"strings"
|
|
|
|
"github.com/autobrr/autobrr/internal/domain"
|
|
"github.com/autobrr/autobrr/pkg/errors"
|
|
|
|
"github.com/go-chi/chi/v5"
|
|
"github.com/r3labs/sse/v2"
|
|
)
|
|
|
|
type ircService interface {
|
|
ListNetworks(ctx context.Context) ([]domain.IrcNetwork, error)
|
|
GetNetworksWithHealth(ctx context.Context) ([]domain.IrcNetworkWithHealth, error)
|
|
DeleteNetwork(ctx context.Context, id int64) error
|
|
GetNetworkByID(ctx context.Context, id int64) (*domain.IrcNetwork, error)
|
|
StoreNetwork(ctx context.Context, network *domain.IrcNetwork) error
|
|
UpdateNetwork(ctx context.Context, network *domain.IrcNetwork) error
|
|
StoreChannel(ctx context.Context, networkID int64, channel *domain.IrcChannel) error
|
|
RestartNetwork(ctx context.Context, id int64) error
|
|
SendCmd(ctx context.Context, req *domain.SendIrcCmdRequest) error
|
|
ManualProcessAnnounce(ctx context.Context, req *domain.IRCManualProcessRequest) error
|
|
}
|
|
|
|
type ircHandler struct {
|
|
encoder encoder
|
|
sse *sse.Server
|
|
|
|
service ircService
|
|
}
|
|
|
|
func newIrcHandler(encoder encoder, sse *sse.Server, service ircService) *ircHandler {
|
|
return &ircHandler{
|
|
encoder: encoder,
|
|
sse: sse,
|
|
service: service,
|
|
}
|
|
}
|
|
|
|
func (h ircHandler) Routes(r chi.Router) {
|
|
r.Get("/", h.listNetworks)
|
|
r.Post("/", h.storeNetwork)
|
|
|
|
r.Route("/network/{networkID}", func(r chi.Router) {
|
|
r.Put("/", h.updateNetwork)
|
|
r.Get("/", h.getNetworkByID)
|
|
r.Delete("/", h.deleteNetwork)
|
|
|
|
r.Post("/cmd", h.sendCmd)
|
|
r.Post("/channel", h.storeChannel)
|
|
r.Get("/restart", h.restartNetwork)
|
|
|
|
r.Post("/channel/{channel}/announce/process", h.announceProcess)
|
|
})
|
|
|
|
r.HandleFunc("/events", func(w http.ResponseWriter, r *http.Request) {
|
|
|
|
// inject CORS headers to bypass checks
|
|
h.sse.Headers = map[string]string{
|
|
"Content-Type": "text/event-stream",
|
|
"Cache-Control": "no-cache",
|
|
"Connection": "keep-alive",
|
|
"X-Accel-Buffering": "no",
|
|
}
|
|
|
|
h.sse.ServeHTTP(w, r)
|
|
})
|
|
}
|
|
|
|
func (h ircHandler) listNetworks(w http.ResponseWriter, r *http.Request) {
|
|
networks, err := h.service.GetNetworksWithHealth(r.Context())
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, networks)
|
|
}
|
|
|
|
func (h ircHandler) getNetworkByID(w http.ResponseWriter, r *http.Request) {
|
|
networkID, err := strconv.Atoi(chi.URLParam(r, "networkID"))
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
network, err := h.service.GetNetworkByID(r.Context(), int64(networkID))
|
|
if err != nil {
|
|
if errors.Is(err, domain.ErrRecordNotFound) {
|
|
h.encoder.NotFoundErr(w, errors.New("network with id %d not found", networkID))
|
|
return
|
|
}
|
|
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, network)
|
|
}
|
|
|
|
func (h ircHandler) restartNetwork(w http.ResponseWriter, r *http.Request) {
|
|
networkID, err := strconv.Atoi(chi.URLParam(r, "networkID"))
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if err := h.service.RestartNetwork(r.Context(), int64(networkID)); err != nil {
|
|
if errors.Is(err, domain.ErrRecordNotFound) {
|
|
h.encoder.NotFoundErr(w, errors.New("network with id %d not found", networkID))
|
|
return
|
|
}
|
|
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|
|
|
|
func (h ircHandler) storeNetwork(w http.ResponseWriter, r *http.Request) {
|
|
var data domain.IrcNetwork
|
|
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if err := h.service.StoreNetwork(r.Context(), &data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|
|
|
|
func (h ircHandler) updateNetwork(w http.ResponseWriter, r *http.Request) {
|
|
var data domain.IrcNetwork
|
|
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if err := h.service.UpdateNetwork(r.Context(), &data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|
|
|
|
func (h ircHandler) sendCmd(w http.ResponseWriter, r *http.Request) {
|
|
networkID, err := strconv.Atoi(chi.URLParam(r, "networkID"))
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
var data domain.SendIrcCmdRequest
|
|
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
data.NetworkId = int64(networkID)
|
|
|
|
if err := h.service.SendCmd(r.Context(), &data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|
|
|
|
// announceProcess manually trigger announce process
|
|
func (h ircHandler) announceProcess(w http.ResponseWriter, r *http.Request) {
|
|
networkID, err := strconv.Atoi(chi.URLParam(r, "networkID"))
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
var data domain.IRCManualProcessRequest
|
|
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
data.NetworkId = int64(networkID)
|
|
data.Channel = chi.URLParam(r, "channel")
|
|
|
|
// we cant pass # as an url parameter so the frontend has to strip it
|
|
if !strings.HasPrefix(data.Channel, "#") {
|
|
data.Channel = fmt.Sprintf("#%s", data.Channel)
|
|
}
|
|
|
|
if err := h.service.ManualProcessAnnounce(r.Context(), &data); err != nil {
|
|
if errors.Is(err, domain.ErrRecordNotFound) {
|
|
h.encoder.NotFoundErr(w, errors.New("network with id %d not found", data.NetworkId))
|
|
return
|
|
}
|
|
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|
|
|
|
func (h ircHandler) storeChannel(w http.ResponseWriter, r *http.Request) {
|
|
networkID, err := strconv.Atoi(chi.URLParam(r, "networkID"))
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
var data domain.IrcChannel
|
|
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if err := h.service.StoreChannel(r.Context(), int64(networkID), &data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|
|
|
|
func (h ircHandler) deleteNetwork(w http.ResponseWriter, r *http.Request) {
|
|
networkID, err := strconv.Atoi(chi.URLParam(r, "networkID"))
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if err := h.service.DeleteNetwork(r.Context(), int64(networkID)); err != nil {
|
|
if errors.Is(err, domain.ErrRecordNotFound) {
|
|
h.encoder.NotFoundErr(w, errors.New("network with id %d does not exist", networkID))
|
|
return
|
|
}
|
|
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|