package api import ( "errors" "net/http" "github.com/alexei/tinyforge/internal/store" "github.com/go-chi/chi/v5" ) // listWorkloads handles GET /api/workloads. Optional ?kind=project|stack|site // filter narrows the result. The shape mirrors the projects/stacks/sites // listing endpoints — clients use this to render the global Workloads view. func (s *Server) listWorkloads(w http.ResponseWriter, r *http.Request) { kind := store.WorkloadKind(r.URL.Query().Get("kind")) out, err := s.store.ListWorkloads(kind) if err != nil { respondError(w, http.StatusInternalServerError, "list workloads") return } respondJSON(w, http.StatusOK, out) } // getWorkload handles GET /api/workloads/{id}. func (s *Server) getWorkload(w http.ResponseWriter, r *http.Request) { id := chi.URLParam(r, "id") wl, err := s.store.GetWorkloadByID(id) if err != nil { if errors.Is(err, store.ErrNotFound) { respondNotFound(w, "workload") return } respondError(w, http.StatusInternalServerError, "get workload") return } respondJSON(w, http.StatusOK, wl) } // listWorkloadContainers handles GET /api/workloads/{id}/containers. // Returns every Container row owned by this workload, newest first. The // frontend's component uses this on every kind-specific // detail page (project, stack, site) so the table shape is uniform. func (s *Server) listWorkloadContainers(w http.ResponseWriter, r *http.Request) { id := chi.URLParam(r, "id") out, err := s.store.ListContainersByWorkload(id) if err != nil { respondError(w, http.StatusInternalServerError, "list workload containers") return } respondJSON(w, http.StatusOK, out) } // updateWorkloadAppID handles PATCH /api/workloads/{id}/app. Body: {"app_id": "..."}. // Empty string clears the app assignment. Used by the (optional) Apps UI. func (s *Server) updateWorkloadAppID(w http.ResponseWriter, r *http.Request) { id := chi.URLParam(r, "id") var req struct { AppID string `json:"app_id"` } if !decodeJSON(w, r, &req) { return } wl, err := s.store.GetWorkloadByID(id) if err != nil { if errors.Is(err, store.ErrNotFound) { respondNotFound(w, "workload") return } respondError(w, http.StatusInternalServerError, "get workload") return } wl.AppID = req.AppID if err := s.store.UpdateWorkload(wl); err != nil { respondError(w, http.StatusInternalServerError, "update workload") return } respondJSON(w, http.StatusOK, wl) }