mirror of
https://github.com/idanoo/autobrr
synced 2025-07-23 08:49:13 +00:00

* fix(releases): add manual processing * feat(irc): add re-process button to channel msg * feat(irc): add missing client method * feat(web): change reprocess icon placement --------- Co-authored-by: martylukyy <35452459+martylukyy@users.noreply.github.com>
289 lines
6.4 KiB
Go
289 lines
6.4 KiB
Go
// Copyright (c) 2021 - 2024, 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/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) {
|
|
ctx := r.Context()
|
|
|
|
networks, err := h.service.GetNetworksWithHealth(ctx)
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, networks)
|
|
}
|
|
|
|
func (h ircHandler) getNetworkByID(w http.ResponseWriter, r *http.Request) {
|
|
var (
|
|
ctx = r.Context()
|
|
networkID = chi.URLParam(r, "networkID")
|
|
)
|
|
|
|
id, err := strconv.Atoi(networkID)
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
network, err := h.service.GetNetworkByID(ctx, int64(id))
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, network)
|
|
}
|
|
|
|
func (h ircHandler) restartNetwork(w http.ResponseWriter, r *http.Request) {
|
|
var (
|
|
ctx = r.Context()
|
|
networkID = chi.URLParam(r, "networkID")
|
|
)
|
|
|
|
id, err := strconv.Atoi(networkID)
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if err := h.service.RestartNetwork(ctx, int64(id)); err != nil {
|
|
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 (
|
|
ctx = r.Context()
|
|
data domain.IrcNetwork
|
|
)
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if err := h.service.UpdateNetwork(ctx, &data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|
|
|
|
func (h ircHandler) sendCmd(w http.ResponseWriter, r *http.Request) {
|
|
var (
|
|
ctx = r.Context()
|
|
networkID = chi.URLParam(r, "networkID")
|
|
data domain.SendIrcCmdRequest
|
|
)
|
|
|
|
id, err := strconv.Atoi(networkID)
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
data.NetworkId = int64(id)
|
|
|
|
if err := h.service.SendCmd(ctx, &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) {
|
|
var (
|
|
ctx = r.Context()
|
|
data domain.IRCManualProcessRequest
|
|
)
|
|
|
|
paramNetworkID := chi.URLParam(r, "networkID")
|
|
if paramNetworkID == "" {
|
|
h.encoder.StatusResponse(w, http.StatusBadRequest, map[string]interface{}{
|
|
"code": "BAD_REQUEST_PARAMS",
|
|
"message": "parameter networkID missing",
|
|
})
|
|
return
|
|
}
|
|
|
|
networkID, err := strconv.Atoi(paramNetworkID)
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
paramChannel := chi.URLParam(r, "channel")
|
|
if paramChannel == "" {
|
|
h.encoder.StatusResponse(w, http.StatusBadRequest, map[string]interface{}{
|
|
"code": "BAD_REQUEST_PARAMS",
|
|
"message": "parameter channel missing",
|
|
})
|
|
return
|
|
}
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
data.NetworkId = int64(networkID)
|
|
data.Channel = paramChannel
|
|
|
|
// 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(ctx, &data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|
|
|
|
func (h ircHandler) storeChannel(w http.ResponseWriter, r *http.Request) {
|
|
var (
|
|
ctx = r.Context()
|
|
networkID = chi.URLParam(r, "networkID")
|
|
data domain.IrcChannel
|
|
)
|
|
|
|
id, err := strconv.Atoi(networkID)
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if err := h.service.StoreChannel(ctx, int64(id), &data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|
|
|
|
func (h ircHandler) deleteNetwork(w http.ResponseWriter, r *http.Request) {
|
|
var (
|
|
ctx = r.Context()
|
|
networkID = chi.URLParam(r, "networkID")
|
|
)
|
|
|
|
id, err := strconv.Atoi(networkID)
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if err := h.service.DeleteNetwork(ctx, int64(id)); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|