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
211 lines
4.8 KiB
Go
211 lines
4.8 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"
|
|
"net/http"
|
|
"strconv"
|
|
|
|
"github.com/autobrr/autobrr/internal/domain"
|
|
"github.com/autobrr/autobrr/pkg/errors"
|
|
|
|
"github.com/go-chi/chi/v5"
|
|
)
|
|
|
|
type indexerService interface {
|
|
Store(ctx context.Context, indexer domain.Indexer) (*domain.Indexer, error)
|
|
Update(ctx context.Context, indexer domain.Indexer) (*domain.Indexer, error)
|
|
List(ctx context.Context) ([]domain.Indexer, error)
|
|
FindByID(ctx context.Context, id int) (*domain.Indexer, error)
|
|
GetAll() ([]*domain.IndexerDefinition, error)
|
|
GetTemplates() ([]domain.IndexerDefinition, error)
|
|
Delete(ctx context.Context, id int) error
|
|
TestApi(ctx context.Context, req domain.IndexerTestApiRequest) error
|
|
ToggleEnabled(ctx context.Context, indexerID int, enabled bool) error
|
|
}
|
|
|
|
type indexerHandler struct {
|
|
encoder encoder
|
|
service indexerService
|
|
ircSvc ircService
|
|
}
|
|
|
|
func newIndexerHandler(encoder encoder, service indexerService, ircSvc ircService) *indexerHandler {
|
|
return &indexerHandler{
|
|
encoder: encoder,
|
|
service: service,
|
|
ircSvc: ircSvc,
|
|
}
|
|
}
|
|
|
|
func (h indexerHandler) Routes(r chi.Router) {
|
|
r.Get("/schema", h.getSchema)
|
|
r.Post("/", h.store)
|
|
r.Get("/", h.getAll)
|
|
r.Get("/options", h.list)
|
|
|
|
r.Route("/{indexerID}", func(r chi.Router) {
|
|
r.Get("/", h.findByID)
|
|
r.Put("/", h.update)
|
|
r.Delete("/", h.delete)
|
|
r.Post("/api/test", h.testApi)
|
|
|
|
r.Patch("/enabled", h.toggleEnabled)
|
|
})
|
|
}
|
|
|
|
func (h indexerHandler) getSchema(w http.ResponseWriter, r *http.Request) {
|
|
indexers, err := h.service.GetTemplates()
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, indexers)
|
|
}
|
|
|
|
func (h indexerHandler) store(w http.ResponseWriter, r *http.Request) {
|
|
var data domain.Indexer
|
|
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
indexer, err := h.service.Store(r.Context(), data)
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusCreated, indexer)
|
|
}
|
|
|
|
func (h indexerHandler) update(w http.ResponseWriter, r *http.Request) {
|
|
var data domain.Indexer
|
|
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
indexer, err := h.service.Update(r.Context(), data)
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, indexer)
|
|
}
|
|
|
|
func (h indexerHandler) delete(w http.ResponseWriter, r *http.Request) {
|
|
indexerID, err := strconv.Atoi(chi.URLParam(r, "indexerID"))
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if err := h.service.Delete(r.Context(), indexerID); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusNoContent, nil)
|
|
}
|
|
|
|
func (h indexerHandler) getAll(w http.ResponseWriter, r *http.Request) {
|
|
indexers, err := h.service.GetAll()
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, indexers)
|
|
}
|
|
|
|
func (h indexerHandler) list(w http.ResponseWriter, r *http.Request) {
|
|
indexers, err := h.service.List(r.Context())
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, indexers)
|
|
}
|
|
|
|
func (h indexerHandler) findByID(w http.ResponseWriter, r *http.Request) {
|
|
indexerID, err := strconv.Atoi(chi.URLParam(r, "indexerID"))
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
indexer, err := h.service.FindByID(r.Context(), indexerID)
|
|
if err != nil {
|
|
if errors.Is(err, domain.ErrRecordNotFound) {
|
|
h.encoder.NotFoundErr(w, errors.New("indexer with id %d not found", indexerID))
|
|
return
|
|
}
|
|
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, indexer)
|
|
}
|
|
|
|
func (h indexerHandler) testApi(w http.ResponseWriter, r *http.Request) {
|
|
indexerID, err := strconv.Atoi(chi.URLParam(r, "indexerID"))
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
var req domain.IndexerTestApiRequest
|
|
if err := json.NewDecoder(r.Body).Decode(&req); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if req.IndexerId == 0 {
|
|
req.IndexerId = indexerID
|
|
}
|
|
|
|
if err := h.service.TestApi(r.Context(), req); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
res := struct {
|
|
Message string `json:"message"`
|
|
}{
|
|
Message: "Indexer api test OK",
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, res)
|
|
}
|
|
|
|
func (h indexerHandler) toggleEnabled(w http.ResponseWriter, r *http.Request) {
|
|
indexerID, err := strconv.Atoi(chi.URLParam(r, "indexerID"))
|
|
if err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
var data struct {
|
|
Enabled bool `json:"enabled"`
|
|
}
|
|
|
|
if err := json.NewDecoder(r.Body).Decode(&data); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
if err := h.service.ToggleEnabled(r.Context(), indexerID, data.Enabled); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|