mirror of
https://github.com/hay-kot/homebox.git
synced 2025-01-08 23:07:12 +00:00
23b5892aef
* introduce scaffold for new models * wip: shoutrrr wrapper (may remove) * update schema files * gen: ent code * gen: migrations * go mod tidy * add group_id to notifier * db migration * new mapper helpers * notifier repo * introduce experimental adapter pattern for hdlrs * refactor adapters to fit more common use cases * new routes for notifiers * update errors to fix validation panic * go tidy * reverse checkbox label display * wip: notifiers UI * use badges instead of text * improve documentation * add scaffold schema reference * remove notifier service * refactor schema folder * support group edges via scaffold * delete test file * include link to API docs * audit and update documentation + improve format * refactor schema edges * refactor * add custom validator * set validate + order fields by name * fix failing tests
301 lines
8.7 KiB
Go
301 lines
8.7 KiB
Go
package v1
|
|
|
|
import (
|
|
"database/sql"
|
|
"encoding/csv"
|
|
"errors"
|
|
"net/http"
|
|
"strings"
|
|
|
|
"github.com/hay-kot/homebox/backend/internal/core/services"
|
|
"github.com/hay-kot/homebox/backend/internal/data/repo"
|
|
"github.com/hay-kot/homebox/backend/internal/sys/validate"
|
|
"github.com/hay-kot/homebox/backend/pkgs/server"
|
|
"github.com/rs/zerolog/log"
|
|
)
|
|
|
|
// HandleItemsGetAll godoc
|
|
//
|
|
// @Summary Query All Items
|
|
// @Tags Items
|
|
// @Produce json
|
|
// @Param q query string false "search string"
|
|
// @Param page query int false "page number"
|
|
// @Param pageSize query int false "items per page"
|
|
// @Param labels query []string false "label Ids" collectionFormat(multi)
|
|
// @Param locations query []string false "location Ids" collectionFormat(multi)
|
|
// @Success 200 {object} repo.PaginationResult[repo.ItemSummary]{}
|
|
// @Router /v1/items [GET]
|
|
// @Security Bearer
|
|
func (ctrl *V1Controller) HandleItemsGetAll() server.HandlerFunc {
|
|
extractQuery := func(r *http.Request) repo.ItemQuery {
|
|
params := r.URL.Query()
|
|
|
|
filterFieldItems := func(raw []string) []repo.FieldQuery {
|
|
var items []repo.FieldQuery
|
|
|
|
for _, v := range raw {
|
|
parts := strings.SplitN(v, "=", 2)
|
|
if len(parts) == 2 {
|
|
items = append(items, repo.FieldQuery{
|
|
Name: parts[0],
|
|
Value: parts[1],
|
|
})
|
|
}
|
|
}
|
|
|
|
return items
|
|
}
|
|
|
|
v := repo.ItemQuery{
|
|
Page: queryIntOrNegativeOne(params.Get("page")),
|
|
PageSize: queryIntOrNegativeOne(params.Get("pageSize")),
|
|
Search: params.Get("q"),
|
|
LocationIDs: queryUUIDList(params, "locations"),
|
|
LabelIDs: queryUUIDList(params, "labels"),
|
|
IncludeArchived: queryBool(params.Get("includeArchived")),
|
|
Fields: filterFieldItems(params["fields"]),
|
|
}
|
|
|
|
if strings.HasPrefix(v.Search, "#") {
|
|
aidStr := strings.TrimPrefix(v.Search, "#")
|
|
|
|
aid, ok := repo.ParseAssetID(aidStr)
|
|
if ok {
|
|
v.Search = ""
|
|
v.AssetID = aid
|
|
}
|
|
}
|
|
|
|
return v
|
|
}
|
|
|
|
return func(w http.ResponseWriter, r *http.Request) error {
|
|
ctx := services.NewContext(r.Context())
|
|
|
|
items, err := ctrl.repo.Items.QueryByGroup(ctx, ctx.GID, extractQuery(r))
|
|
if err != nil {
|
|
if errors.Is(err, sql.ErrNoRows) {
|
|
return server.Respond(w, http.StatusOK, repo.PaginationResult[repo.ItemSummary]{
|
|
Items: []repo.ItemSummary{},
|
|
})
|
|
}
|
|
log.Err(err).Msg("failed to get items")
|
|
return validate.NewRequestError(err, http.StatusInternalServerError)
|
|
}
|
|
return server.Respond(w, http.StatusOK, items)
|
|
}
|
|
}
|
|
|
|
// HandleItemsCreate godoc
|
|
//
|
|
// @Summary Create Item
|
|
// @Tags Items
|
|
// @Produce json
|
|
// @Param payload body repo.ItemCreate true "Item Data"
|
|
// @Success 200 {object} repo.ItemSummary
|
|
// @Router /v1/items [POST]
|
|
// @Security Bearer
|
|
func (ctrl *V1Controller) HandleItemsCreate() server.HandlerFunc {
|
|
return func(w http.ResponseWriter, r *http.Request) error {
|
|
createData := repo.ItemCreate{}
|
|
if err := server.Decode(r, &createData); err != nil {
|
|
log.Err(err).Msg("failed to decode request body")
|
|
return validate.NewRequestError(err, http.StatusInternalServerError)
|
|
}
|
|
|
|
ctx := services.NewContext(r.Context())
|
|
item, err := ctrl.svc.Items.Create(ctx, createData)
|
|
if err != nil {
|
|
log.Err(err).Msg("failed to create item")
|
|
return validate.NewRequestError(err, http.StatusInternalServerError)
|
|
}
|
|
|
|
return server.Respond(w, http.StatusCreated, item)
|
|
}
|
|
}
|
|
|
|
// HandleItemGet godocs
|
|
//
|
|
// @Summary Get Item
|
|
// @Tags Items
|
|
// @Produce json
|
|
// @Param id path string true "Item ID"
|
|
// @Success 200 {object} repo.ItemOut
|
|
// @Router /v1/items/{id} [GET]
|
|
// @Security Bearer
|
|
func (ctrl *V1Controller) HandleItemGet() server.HandlerFunc {
|
|
return ctrl.handleItemsGeneral()
|
|
}
|
|
|
|
// HandleItemDelete godocs
|
|
//
|
|
// @Summary Delete Item
|
|
// @Tags Items
|
|
// @Produce json
|
|
// @Param id path string true "Item ID"
|
|
// @Success 204
|
|
// @Router /v1/items/{id} [DELETE]
|
|
// @Security Bearer
|
|
func (ctrl *V1Controller) HandleItemDelete() server.HandlerFunc {
|
|
return ctrl.handleItemsGeneral()
|
|
}
|
|
|
|
// HandleItemUpdate godocs
|
|
//
|
|
// @Summary Update Item
|
|
// @Tags Items
|
|
// @Produce json
|
|
// @Param id path string true "Item ID"
|
|
// @Param payload body repo.ItemUpdate true "Item Data"
|
|
// @Success 200 {object} repo.ItemOut
|
|
// @Router /v1/items/{id} [PUT]
|
|
// @Security Bearer
|
|
func (ctrl *V1Controller) HandleItemUpdate() server.HandlerFunc {
|
|
return ctrl.handleItemsGeneral()
|
|
}
|
|
|
|
func (ctrl *V1Controller) handleItemsGeneral() server.HandlerFunc {
|
|
return func(w http.ResponseWriter, r *http.Request) error {
|
|
ctx := services.NewContext(r.Context())
|
|
ID, err := ctrl.routeID(r)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
switch r.Method {
|
|
case http.MethodGet:
|
|
items, err := ctrl.repo.Items.GetOneByGroup(r.Context(), ctx.GID, ID)
|
|
if err != nil {
|
|
log.Err(err).Msg("failed to get item")
|
|
return validate.NewRequestError(err, http.StatusInternalServerError)
|
|
}
|
|
return server.Respond(w, http.StatusOK, items)
|
|
case http.MethodDelete:
|
|
err = ctrl.repo.Items.DeleteByGroup(r.Context(), ctx.GID, ID)
|
|
if err != nil {
|
|
log.Err(err).Msg("failed to delete item")
|
|
return validate.NewRequestError(err, http.StatusInternalServerError)
|
|
}
|
|
return server.Respond(w, http.StatusNoContent, nil)
|
|
case http.MethodPut:
|
|
body := repo.ItemUpdate{}
|
|
if err := server.Decode(r, &body); err != nil {
|
|
log.Err(err).Msg("failed to decode request body")
|
|
return validate.NewRequestError(err, http.StatusInternalServerError)
|
|
}
|
|
body.ID = ID
|
|
result, err := ctrl.repo.Items.UpdateByGroup(r.Context(), ctx.GID, body)
|
|
if err != nil {
|
|
log.Err(err).Msg("failed to update item")
|
|
return validate.NewRequestError(err, http.StatusInternalServerError)
|
|
}
|
|
return server.Respond(w, http.StatusOK, result)
|
|
}
|
|
|
|
return nil
|
|
}
|
|
}
|
|
|
|
// HandleGetAllCustomFieldNames godocs
|
|
//
|
|
// @Summary Get All Custom Field Names
|
|
// @Tags Items
|
|
// @Produce json
|
|
// @Success 200
|
|
// @Router /v1/items/fields [GET]
|
|
// @Success 200 {object} []string
|
|
// @Security Bearer
|
|
func (ctrl *V1Controller) HandleGetAllCustomFieldNames() server.HandlerFunc {
|
|
return func(w http.ResponseWriter, r *http.Request) error {
|
|
ctx := services.NewContext(r.Context())
|
|
|
|
v, err := ctrl.repo.Items.GetAllCustomFieldNames(r.Context(), ctx.GID)
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return server.Respond(w, http.StatusOK, v)
|
|
}
|
|
}
|
|
|
|
// HandleGetAllCustomFieldValues godocs
|
|
//
|
|
// @Summary Get All Custom Field Values
|
|
// @Tags Items
|
|
// @Produce json
|
|
// @Success 200
|
|
// @Router /v1/items/fields/values [GET]
|
|
// @Success 200 {object} []string
|
|
// @Security Bearer
|
|
func (ctrl *V1Controller) HandleGetAllCustomFieldValues() server.HandlerFunc {
|
|
return func(w http.ResponseWriter, r *http.Request) error {
|
|
ctx := services.NewContext(r.Context())
|
|
|
|
v, err := ctrl.repo.Items.GetAllCustomFieldValues(r.Context(), ctx.GID, r.URL.Query().Get("field"))
|
|
if err != nil {
|
|
return err
|
|
}
|
|
|
|
return server.Respond(w, http.StatusOK, v)
|
|
}
|
|
}
|
|
|
|
// HandleItemsImport godocs
|
|
//
|
|
// @Summary Import Items
|
|
// @Tags Items
|
|
// @Produce json
|
|
// @Success 204
|
|
// @Param csv formData file true "Image to upload"
|
|
// @Router /v1/items/import [Post]
|
|
// @Security Bearer
|
|
func (ctrl *V1Controller) HandleItemsImport() server.HandlerFunc {
|
|
return func(w http.ResponseWriter, r *http.Request) error {
|
|
err := r.ParseMultipartForm(ctrl.maxUploadSize << 20)
|
|
if err != nil {
|
|
log.Err(err).Msg("failed to parse multipart form")
|
|
return validate.NewRequestError(err, http.StatusInternalServerError)
|
|
}
|
|
|
|
file, _, err := r.FormFile("csv")
|
|
if err != nil {
|
|
log.Err(err).Msg("failed to get file from form")
|
|
return validate.NewRequestError(err, http.StatusInternalServerError)
|
|
}
|
|
|
|
user := services.UseUserCtx(r.Context())
|
|
|
|
_, err = ctrl.svc.Items.CsvImport(r.Context(), user.GroupID, file)
|
|
if err != nil {
|
|
log.Err(err).Msg("failed to import items")
|
|
return validate.NewRequestError(err, http.StatusInternalServerError)
|
|
}
|
|
|
|
return server.Respond(w, http.StatusNoContent, nil)
|
|
}
|
|
}
|
|
|
|
// HandleItemsExport godocs
|
|
//
|
|
// @Summary Export Items
|
|
// @Tags Items
|
|
// @Success 200 {string} string "text/csv"
|
|
// @Router /v1/items/export [GET]
|
|
// @Security Bearer
|
|
func (ctrl *V1Controller) HandleItemsExport() server.HandlerFunc {
|
|
return func(w http.ResponseWriter, r *http.Request) error {
|
|
ctx := services.NewContext(r.Context())
|
|
|
|
csvData, err := ctrl.svc.Items.ExportTSV(r.Context(), ctx.GID)
|
|
if err != nil {
|
|
log.Err(err).Msg("failed to export items")
|
|
return validate.NewRequestError(err, http.StatusInternalServerError)
|
|
}
|
|
|
|
w.Header().Set("Content-Type", "text/tsv")
|
|
w.Header().Set("Content-Disposition", "attachment;filename=homebox-items.tsv")
|
|
writer := csv.NewWriter(w)
|
|
return writer.WriteAll(csvData)
|
|
}
|
|
}
|