Backend (Go): - Expanded integration_test.go: health, auth middleware (expired/invalid/wrong-secret JWT), tenant CRUD, case CRUD (create/list/get/update/delete + filters + validation), deadline CRUD (create/list/update/complete/delete), appointment CRUD, dashboard (verifies all sections), deadline calculator (valid/invalid/unknown type), proceeding types & rules, document endpoints, AI extraction (no-key path), and full critical path E2E (auth -> case -> deadline -> appointment -> dashboard -> complete) - New handler unit tests: case (10), appointment (11), dashboard (1), calculate (5), document (10), AI (4) — all testing validation, auth guards, and error paths without DB - Total: ~80 backend tests (unit + integration) Frontend (TypeScript/Vitest): - Installed vitest 2.x, @testing-library/react, @testing-library/jest-dom, jsdom 24, msw - vitest.config.ts with jsdom env, esbuild JSX automatic, path aliases - API client tests (13): URL construction, no double /api/, auth header, tenant header, POST/PUT/PATCH/DELETE methods, error handling, 204 responses - DeadlineTrafficLights tests (5): renders cards, correct counts, zero state, onFilter callback - CaseOverviewGrid tests (4): renders categories, counts, header, zero state - LoginPage tests (8): form rendering, mode toggle, password login, redirect, error display, magic link, registration link - Total: 30 frontend tests Makefile: test-frontend target now runs vitest instead of placeholder echo.
84 lines
2.0 KiB
Go
84 lines
2.0 KiB
Go
package handlers
|
|
|
|
import (
|
|
"bytes"
|
|
"encoding/json"
|
|
"net/http"
|
|
"net/http/httptest"
|
|
"testing"
|
|
)
|
|
|
|
func TestCalculate_MissingFields(t *testing.T) {
|
|
h := &CalculateHandlers{}
|
|
|
|
tests := []struct {
|
|
name string
|
|
body string
|
|
want string
|
|
}{
|
|
{
|
|
name: "empty body",
|
|
body: `{}`,
|
|
want: "proceeding_type and trigger_event_date are required",
|
|
},
|
|
{
|
|
name: "missing trigger_event_date",
|
|
body: `{"proceeding_type":"INF"}`,
|
|
want: "proceeding_type and trigger_event_date are required",
|
|
},
|
|
{
|
|
name: "missing proceeding_type",
|
|
body: `{"trigger_event_date":"2026-06-01"}`,
|
|
want: "proceeding_type and trigger_event_date are required",
|
|
},
|
|
}
|
|
|
|
for _, tt := range tests {
|
|
t.Run(tt.name, func(t *testing.T) {
|
|
r := httptest.NewRequest("POST", "/api/deadlines/calculate", bytes.NewBufferString(tt.body))
|
|
w := httptest.NewRecorder()
|
|
|
|
h.Calculate(w, r)
|
|
|
|
if w.Code != http.StatusBadRequest {
|
|
t.Errorf("expected 400, got %d", w.Code)
|
|
}
|
|
var resp map[string]string
|
|
json.NewDecoder(w.Body).Decode(&resp)
|
|
if resp["error"] != tt.want {
|
|
t.Errorf("expected error %q, got %q", tt.want, resp["error"])
|
|
}
|
|
})
|
|
}
|
|
}
|
|
|
|
func TestCalculate_InvalidDateFormat(t *testing.T) {
|
|
h := &CalculateHandlers{}
|
|
body := `{"proceeding_type":"INF","trigger_event_date":"01-06-2026"}`
|
|
r := httptest.NewRequest("POST", "/api/deadlines/calculate", bytes.NewBufferString(body))
|
|
w := httptest.NewRecorder()
|
|
|
|
h.Calculate(w, r)
|
|
|
|
if w.Code != http.StatusBadRequest {
|
|
t.Errorf("expected 400, got %d", w.Code)
|
|
}
|
|
var resp map[string]string
|
|
json.NewDecoder(w.Body).Decode(&resp)
|
|
if resp["error"] != "invalid trigger_event_date format, expected YYYY-MM-DD" {
|
|
t.Errorf("unexpected error: %s", resp["error"])
|
|
}
|
|
}
|
|
|
|
func TestCalculate_InvalidJSON(t *testing.T) {
|
|
h := &CalculateHandlers{}
|
|
r := httptest.NewRequest("POST", "/api/deadlines/calculate", bytes.NewBufferString(`not-json`))
|
|
w := httptest.NewRecorder()
|
|
|
|
h.Calculate(w, r)
|
|
|
|
if w.Code != http.StatusBadRequest {
|
|
t.Errorf("expected 400, got %d", w.Code)
|
|
}
|
|
}
|