ultisuite-backend/internal/search/handler.go
R3D347HR4Y 621b0099d6
Some checks are pending
CI / Go tests (push) Waiting to run
CI / Integration tests (push) Waiting to run
CI / DB migrations (push) Waiting to run
feat(deploy): enhance Nginx configuration and API integration for UltiAI
- Updated .env.example to include new configuration options for the UltiAI branding and API endpoints.
- Enhanced Nginx configuration to support new API routes for the MCP and WebSocket connections.
- Introduced sub-filters for branding adjustments in Nginx responses.
- Added new JavaScript patch for API endpoint adjustments.
- Implemented tests for new API functionalities and improved error handling in the AI gateway.
2026-06-15 00:22:23 +02:00

83 lines
2.4 KiB
Go

package search
import (
"log/slog"
"net/http"
"strings"
"github.com/jackc/pgx/v5/pgxpool"
"github.com/ultisuite/ulti-backend/internal/api/apiresponse"
"github.com/ultisuite/ulti-backend/internal/api/apivalidate"
"github.com/ultisuite/ulti-backend/internal/api/middleware"
"github.com/ultisuite/ulti-backend/internal/api/query"
"github.com/ultisuite/ulti-backend/internal/nextcloud"
)
type Handler struct {
svc *Service
logger *slog.Logger
}
type Options struct {
Nextcloud *nextcloud.Client
Engine string
MeilisearchURL string
MeilisearchKey string
MeilisearchIndex string
TypesenseURL string
TypesenseKey string
TypesenseCollection string
}
func NewHandler(db *pgxpool.Pool, opts Options) *Handler {
return &Handler{
svc: NewService(db, ServiceOptions{
Nextcloud: opts.Nextcloud,
Engine: opts.Engine,
MeilisearchURL: opts.MeilisearchURL,
MeilisearchKey: opts.MeilisearchKey,
MeilisearchIndex: opts.MeilisearchIndex,
TypesenseURL: opts.TypesenseURL,
TypesenseKey: opts.TypesenseKey,
TypesenseCollection: opts.TypesenseCollection,
}),
logger: slog.Default().With("component", "search"),
}
}
func (h *Handler) Search(w http.ResponseWriter, r *http.Request) {
claims := middleware.ClaimsFromContext(r.Context())
q := r.URL.Query().Get("q")
if verr := validateSearchQuery(q); verr != nil {
apiresponse.WriteError(w, r, http.StatusBadRequest, apiresponse.CodeInvalidQueryParam, "invalid query parameters", verr.Details)
return
}
types := r.URL.Query().Get("types")
if verr := validateSearchTypes(types); verr != nil {
apiresponse.WriteError(w, r, http.StatusBadRequest, apiresponse.CodeInvalidQueryParam, "invalid query parameters", verr.Details)
return
}
params, err := query.ParseListRequest(r)
if err != nil {
apivalidate.WriteQueryError(w, r, err)
return
}
filters := SearchFilters{
AccountID: strings.TrimSpace(r.URL.Query().Get("account_id")),
IncludeSpam: parseSearchOptionalBool(r.URL.Query().Get("include_spam")),
IncludeTrash: parseSearchOptionalBool(r.URL.Query().Get("include_trash")),
}
result, err := h.svc.Search(r.Context(), claims.Sub, q, types, params, filters)
if err != nil {
h.logger.Error("search", "error", err)
apivalidate.WriteInternal(w, r)
return
}
apiresponse.WriteJSON(w, http.StatusOK, result)
}