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

* fix(releases): search with postgres * fix: validate release push status from filters * fix: add missing import * fix: validate push status if not empty
259 lines
6.4 KiB
Go
259 lines
6.4 KiB
Go
// Copyright (c) 2021 - 2023, Ludvig Lundgren and the autobrr contributors.
|
|
// SPDX-License-Identifier: GPL-2.0-or-later
|
|
|
|
package http
|
|
|
|
import (
|
|
"context"
|
|
"fmt"
|
|
"net/http"
|
|
"net/url"
|
|
"strconv"
|
|
|
|
"github.com/autobrr/autobrr/internal/domain"
|
|
|
|
"github.com/go-chi/chi/v5"
|
|
)
|
|
|
|
type releaseService interface {
|
|
Find(ctx context.Context, query domain.ReleaseQueryParams) (res []*domain.Release, nextCursor int64, count int64, err error)
|
|
FindRecent(ctx context.Context) (res []*domain.Release, err error)
|
|
GetIndexerOptions(ctx context.Context) ([]string, error)
|
|
Stats(ctx context.Context) (*domain.ReleaseStats, error)
|
|
Delete(ctx context.Context, req *domain.DeleteReleaseRequest) error
|
|
Retry(ctx context.Context, req *domain.ReleaseActionRetryReq) error
|
|
}
|
|
|
|
type releaseHandler struct {
|
|
encoder encoder
|
|
service releaseService
|
|
}
|
|
|
|
func newReleaseHandler(encoder encoder, service releaseService) *releaseHandler {
|
|
return &releaseHandler{
|
|
encoder: encoder,
|
|
service: service,
|
|
}
|
|
}
|
|
|
|
func (h releaseHandler) Routes(r chi.Router) {
|
|
r.Get("/", h.findReleases)
|
|
r.Get("/recent", h.findRecentReleases)
|
|
r.Get("/stats", h.getStats)
|
|
r.Get("/indexers", h.getIndexerOptions)
|
|
r.Delete("/", h.deleteReleases)
|
|
|
|
r.Route("/{releaseId}", func(r chi.Router) {
|
|
r.Post("/actions/{actionStatusId}/retry", h.retryAction)
|
|
})
|
|
}
|
|
|
|
func (h releaseHandler) findReleases(w http.ResponseWriter, r *http.Request) {
|
|
|
|
limitP := r.URL.Query().Get("limit")
|
|
limit, err := strconv.Atoi(limitP)
|
|
if err != nil && limitP != "" {
|
|
h.encoder.StatusResponse(w, http.StatusBadRequest, map[string]interface{}{
|
|
"code": "BAD_REQUEST_PARAMS",
|
|
"message": "limit parameter is invalid",
|
|
})
|
|
return
|
|
}
|
|
if limit == 0 {
|
|
limit = 20
|
|
}
|
|
|
|
offsetP := r.URL.Query().Get("offset")
|
|
offset, err := strconv.Atoi(offsetP)
|
|
if err != nil && offsetP != "" {
|
|
h.encoder.StatusResponse(w, http.StatusBadRequest, map[string]interface{}{
|
|
"code": "BAD_REQUEST_PARAMS",
|
|
"message": "offset parameter is invalid",
|
|
})
|
|
return
|
|
}
|
|
|
|
cursorP := r.URL.Query().Get("cursor")
|
|
cursor := 0
|
|
if cursorP != "" {
|
|
cursor, err = strconv.Atoi(cursorP)
|
|
if err != nil && cursorP != "" {
|
|
h.encoder.StatusResponse(w, http.StatusBadRequest, map[string]interface{}{
|
|
"code": "BAD_REQUEST_PARAMS",
|
|
"message": "cursor parameter is invalid",
|
|
})
|
|
}
|
|
return
|
|
}
|
|
|
|
u, err := url.Parse(r.URL.String())
|
|
if err != nil {
|
|
h.encoder.StatusResponse(w, http.StatusBadRequest, map[string]interface{}{
|
|
"code": "BAD_REQUEST_PARAMS",
|
|
"message": "indexer parameter is invalid",
|
|
})
|
|
return
|
|
}
|
|
vals := u.Query()
|
|
indexer := vals["indexer"]
|
|
|
|
pushStatus := r.URL.Query().Get("push_status")
|
|
if pushStatus != "" {
|
|
if !domain.ValidReleasePushStatus(pushStatus) {
|
|
h.encoder.StatusResponse(w, http.StatusBadRequest, map[string]interface{}{
|
|
"code": "BAD_REQUEST_PARAMS",
|
|
"message": fmt.Sprintf("push_status parameter is of invalid type: %v", pushStatus),
|
|
})
|
|
return
|
|
}
|
|
}
|
|
|
|
search := r.URL.Query().Get("q")
|
|
|
|
query := domain.ReleaseQueryParams{
|
|
Limit: uint64(limit),
|
|
Offset: uint64(offset),
|
|
Cursor: uint64(cursor),
|
|
Sort: nil,
|
|
Filters: struct {
|
|
Indexers []string
|
|
PushStatus string
|
|
}{Indexers: indexer, PushStatus: pushStatus},
|
|
Search: search,
|
|
}
|
|
|
|
releases, nextCursor, count, err := h.service.Find(r.Context(), query)
|
|
if err != nil {
|
|
h.encoder.StatusResponse(w, http.StatusInternalServerError, map[string]interface{}{
|
|
"code": "INTERNAL_SERVER_ERROR",
|
|
"message": err.Error(),
|
|
})
|
|
return
|
|
}
|
|
|
|
ret := struct {
|
|
Data []*domain.Release `json:"data"`
|
|
NextCursor int64 `json:"next_cursor"`
|
|
Count int64 `json:"count"`
|
|
}{
|
|
Data: releases,
|
|
NextCursor: nextCursor,
|
|
Count: count,
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, ret)
|
|
}
|
|
|
|
func (h releaseHandler) findRecentReleases(w http.ResponseWriter, r *http.Request) {
|
|
|
|
releases, err := h.service.FindRecent(r.Context())
|
|
if err != nil {
|
|
h.encoder.StatusResponse(w, http.StatusInternalServerError, map[string]interface{}{
|
|
"code": "INTERNAL_SERVER_ERROR",
|
|
"message": err.Error(),
|
|
})
|
|
return
|
|
}
|
|
|
|
ret := struct {
|
|
Data []*domain.Release `json:"data"`
|
|
}{
|
|
Data: releases,
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, ret)
|
|
}
|
|
|
|
func (h releaseHandler) getIndexerOptions(w http.ResponseWriter, r *http.Request) {
|
|
stats, err := h.service.GetIndexerOptions(r.Context())
|
|
if err != nil {
|
|
h.encoder.StatusResponse(w, http.StatusInternalServerError, map[string]interface{}{
|
|
"code": "INTERNAL_SERVER_ERROR",
|
|
"message": err.Error(),
|
|
})
|
|
return
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, stats)
|
|
}
|
|
|
|
func (h releaseHandler) getStats(w http.ResponseWriter, r *http.Request) {
|
|
|
|
stats, err := h.service.Stats(r.Context())
|
|
if err != nil {
|
|
h.encoder.StatusResponse(w, http.StatusInternalServerError, map[string]interface{}{
|
|
"code": "INTERNAL_SERVER_ERROR",
|
|
"message": err.Error(),
|
|
})
|
|
return
|
|
}
|
|
|
|
h.encoder.StatusResponse(w, http.StatusOK, stats)
|
|
}
|
|
|
|
func (h releaseHandler) deleteReleases(w http.ResponseWriter, r *http.Request) {
|
|
req := domain.DeleteReleaseRequest{}
|
|
|
|
olderThanParam := r.URL.Query().Get("olderThan")
|
|
if olderThanParam != "" {
|
|
duration, err := strconv.Atoi(olderThanParam)
|
|
if err != nil {
|
|
h.encoder.StatusResponse(w, http.StatusBadRequest, map[string]interface{}{
|
|
"code": "BAD_REQUEST_PARAMS",
|
|
"message": "olderThan parameter is invalid",
|
|
})
|
|
return
|
|
}
|
|
req.OlderThan = duration
|
|
}
|
|
|
|
if err := h.service.Delete(r.Context(), &req); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|
|
|
|
func (h releaseHandler) retryAction(w http.ResponseWriter, r *http.Request) {
|
|
var (
|
|
req *domain.ReleaseActionRetryReq
|
|
err error
|
|
)
|
|
|
|
releaseIdParam := chi.URLParam(r, "releaseId")
|
|
if releaseIdParam == "" {
|
|
h.encoder.StatusError(w, http.StatusBadRequest, err)
|
|
return
|
|
}
|
|
|
|
releaseId, err := strconv.Atoi(releaseIdParam)
|
|
if err != nil {
|
|
h.encoder.StatusError(w, http.StatusBadRequest, err)
|
|
return
|
|
}
|
|
|
|
actionStatusIdParam := chi.URLParam(r, "actionStatusId")
|
|
if actionStatusIdParam == "" {
|
|
h.encoder.StatusError(w, http.StatusBadRequest, err)
|
|
return
|
|
}
|
|
|
|
actionStatusId, err := strconv.Atoi(actionStatusIdParam)
|
|
if err != nil {
|
|
h.encoder.StatusError(w, http.StatusBadRequest, err)
|
|
return
|
|
}
|
|
|
|
req = &domain.ReleaseActionRetryReq{
|
|
ReleaseId: releaseId,
|
|
ActionStatusId: actionStatusId,
|
|
}
|
|
|
|
if err := h.service.Retry(r.Context(), req); err != nil {
|
|
h.encoder.Error(w, err)
|
|
return
|
|
}
|
|
|
|
h.encoder.NoContent(w)
|
|
}
|