5662334799
Introduce service.ErrProviderUnavailable, wrapped only around the provider GetRecords call in ZoneRecords. handleZoneRecords and handleTemplateFromZone now use errors.Is against it to tell a real provider outage (502) apart from local resolution failures such as an unknown domain (404), instead of collapsing every ZoneRecords error into a blanket 502. Also fixes handleTemplateFromZone's GetDomain error branch to return 404 "domain not found" instead of 500, for consistency with handleSetDomainTemplate/handleDomainHistory. Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com> Claude-Session: https://claude.ai/code/session_01BwxdSt4reTm7Dj1oxRvpP3
148 lines
5.1 KiB
Go
148 lines
5.1 KiB
Go
package api
|
|
|
|
import (
|
|
"encoding/json"
|
|
"errors"
|
|
"io"
|
|
"log"
|
|
"net/http"
|
|
|
|
"github.com/go-chi/chi/v5"
|
|
"github.com/google/uuid"
|
|
|
|
"github.com/vasyakrg/dns-autoresolver/internal/model"
|
|
"github.com/vasyakrg/dns-autoresolver/internal/service"
|
|
"github.com/vasyakrg/dns-autoresolver/internal/store/dto"
|
|
)
|
|
|
|
func writeJSON(w http.ResponseWriter, status int, v any) {
|
|
w.Header().Set("Content-Type", "application/json")
|
|
w.WriteHeader(status)
|
|
_ = json.NewEncoder(w).Encode(v)
|
|
}
|
|
|
|
func writeErr(w http.ResponseWriter, status int, msg string) {
|
|
writeJSON(w, status, map[string]string{"error": msg})
|
|
}
|
|
|
|
func (a *API) handleCheck(w http.ResponseWriter, r *http.Request) {
|
|
// pid is guaranteed present and owned by the caller — RequireProjectAccess
|
|
// validated it before this handler ever runs.
|
|
pid, _ := projectIDFrom(r.Context())
|
|
did, err := uuid.Parse(chi.URLParam(r, "did"))
|
|
if err != nil {
|
|
writeErr(w, http.StatusBadRequest, "invalid domain id")
|
|
return
|
|
}
|
|
cs, err := a.Svc.Check(r.Context(), pid, did)
|
|
if err != nil {
|
|
log.Printf("api: check failed: %v", err)
|
|
writeErr(w, http.StatusInternalServerError, "internal error")
|
|
return
|
|
}
|
|
writeJSON(w, http.StatusOK, toChangesetResponse(cs))
|
|
}
|
|
|
|
func (a *API) handleApply(w http.ResponseWriter, r *http.Request) {
|
|
// pid is guaranteed present and owned by the caller — RequireProjectAccess
|
|
// validated it before this handler ever runs.
|
|
pid, _ := projectIDFrom(r.Context())
|
|
did, err := uuid.Parse(chi.URLParam(r, "did"))
|
|
if err != nil {
|
|
writeErr(w, http.StatusBadRequest, "invalid domain id")
|
|
return
|
|
}
|
|
var req applyRequest
|
|
if r.Body != nil {
|
|
r.Body = http.MaxBytesReader(w, r.Body, 1<<20) // 1 MiB
|
|
// пустое тело допустимо → значения по умолчанию (prune=false);
|
|
// любая другая ошибка decode (битый JSON, неверные типы) → 400
|
|
if err := json.NewDecoder(r.Body).Decode(&req); err != nil && !errors.Is(err, io.EOF) {
|
|
writeErr(w, http.StatusBadRequest, "invalid request body")
|
|
return
|
|
}
|
|
}
|
|
cs, err := a.Svc.Apply(r.Context(), pid, did, service.ApplyRequest{
|
|
ApplyUpdates: req.ApplyUpdates, ApplyPrunes: req.ApplyPrunes,
|
|
})
|
|
if err != nil {
|
|
log.Printf("api: apply failed: %v", err)
|
|
writeErr(w, http.StatusInternalServerError, "internal error")
|
|
return
|
|
}
|
|
writeJSON(w, http.StatusOK, toChangesetResponse(cs))
|
|
}
|
|
|
|
// handleZoneRecords reads a zone's current records straight from the
|
|
// provider — no template required, no diff computed. Backs read-only zone
|
|
// viewing for domains that don't have a template attached (yet).
|
|
func (a *API) handleZoneRecords(w http.ResponseWriter, r *http.Request) {
|
|
pid, _ := projectIDFrom(r.Context())
|
|
did, err := uuid.Parse(chi.URLParam(r, "did"))
|
|
if err != nil {
|
|
writeErr(w, http.StatusBadRequest, "invalid domain id")
|
|
return
|
|
}
|
|
recs, err := a.Svc.ZoneRecords(r.Context(), pid, did)
|
|
if err != nil {
|
|
log.Printf("api: zone records failed: %v", err)
|
|
if errors.Is(err, service.ErrProviderUnavailable) {
|
|
writeErr(w, http.StatusBadGateway, "не удалось получить записи зоны у провайдера")
|
|
} else {
|
|
writeErr(w, http.StatusNotFound, "домен не найден")
|
|
}
|
|
return
|
|
}
|
|
doc := dto.FromModel(recs)
|
|
writeJSON(w, http.StatusOK, doc.Records) // []dto.RecordDTO
|
|
}
|
|
|
|
// handleTemplateFromZone snapshots a zone's current managed records (NS/SOA
|
|
// excluded — they're read-only, never part of a template) into a brand new
|
|
// template, then auto-attaches it to the domain so check/apply become
|
|
// available immediately.
|
|
func (a *API) handleTemplateFromZone(w http.ResponseWriter, r *http.Request) {
|
|
pid, _ := projectIDFrom(r.Context())
|
|
did, err := uuid.Parse(chi.URLParam(r, "did"))
|
|
if err != nil {
|
|
writeErr(w, http.StatusBadRequest, "invalid domain id")
|
|
return
|
|
}
|
|
dom, err := a.Store.GetDomain(r.Context(), did, pid)
|
|
if err != nil {
|
|
log.Printf("api: template-from-zone: get domain failed: %v", err)
|
|
writeErr(w, http.StatusNotFound, "домен не найден")
|
|
return
|
|
}
|
|
recs, err := a.Svc.ZoneRecords(r.Context(), pid, did)
|
|
if err != nil {
|
|
log.Printf("api: template-from-zone: zone records failed: %v", err)
|
|
if errors.Is(err, service.ErrProviderUnavailable) {
|
|
writeErr(w, http.StatusBadGateway, "не удалось получить записи зоны у провайдера")
|
|
} else {
|
|
writeErr(w, http.StatusNotFound, "домен не найден")
|
|
}
|
|
return
|
|
}
|
|
// Snapshot only managed records — NS/SOA are read-only and never templated.
|
|
managed := make([]model.Record, 0, len(recs))
|
|
for _, rc := range recs {
|
|
if rc.Type.Managed() {
|
|
managed = append(managed, rc)
|
|
}
|
|
}
|
|
doc := dto.FromModel(managed)
|
|
tmpl, err := a.Store.CreateTemplate(r.Context(), pid, dom.ZoneName+" snapshot", doc)
|
|
if err != nil {
|
|
log.Printf("api: template-from-zone: create template failed: %v", err)
|
|
writeErr(w, http.StatusInternalServerError, "internal error")
|
|
return
|
|
}
|
|
if _, err := a.Store.SetDomainTemplate(r.Context(), did, pid, &tmpl.ID); err != nil {
|
|
log.Printf("api: template-from-zone: attach template failed: %v", err)
|
|
writeErr(w, http.StatusInternalServerError, "internal error")
|
|
return
|
|
}
|
|
writeJSON(w, http.StatusCreated, toTemplateResponse(tmpl))
|
|
}
|