Add auto-generated API client and update authStore tests
- Integrated OpenAPI-generated TypeScript SDK (`sdk.gen.ts`, `types.gen.ts`, `client.gen.ts`) for API interactions. - Refactored `authStore` tests to include storage mock reset logic with default implementations.
This commit is contained in:
839
frontend/IMPLEMENTATION_PLAN.md
Normal file
839
frontend/IMPLEMENTATION_PLAN.md
Normal file
@@ -0,0 +1,839 @@
|
|||||||
|
# Frontend Implementation Plan: Next.js + FastAPI Template
|
||||||
|
|
||||||
|
**Last Updated:** October 31, 2025
|
||||||
|
**Current Phase:** Phase 1 COMPLETE ✅ | Ready for Phase 2
|
||||||
|
**Overall Progress:** 1 of 12 phases complete
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Summary
|
||||||
|
|
||||||
|
Build a production-ready Next.js 15 frontend with full authentication, admin dashboard, user/organization management, and session tracking. The frontend integrates with the existing FastAPI backend using OpenAPI-generated clients, TanStack Query for state, Zustand for auth, and shadcn/ui components.
|
||||||
|
|
||||||
|
**Target:** 90%+ test coverage, comprehensive documentation, and robust foundations for enterprise projects.
|
||||||
|
|
||||||
|
**Current State:** Phase 1 infrastructure complete with 81.6% test coverage, 66 passing tests, zero TypeScript errors
|
||||||
|
**Target State:** Complete template matching `frontend-requirements.md` with all 12 phases
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Implementation Directives (MUST FOLLOW)
|
||||||
|
|
||||||
|
### Documentation-First Approach
|
||||||
|
- Phase 0 created `/docs` folder with all architecture, standards, and guides ✅
|
||||||
|
- ALL subsequent phases MUST reference and follow patterns in `/docs`
|
||||||
|
- **If context is lost, `/docs` + this file + `frontend-requirements.md` are sufficient to resume**
|
||||||
|
|
||||||
|
### Quality Assurance Protocol
|
||||||
|
|
||||||
|
**1. After Each Task:** Launch self-review to check:
|
||||||
|
- Code quality issues
|
||||||
|
- Security vulnerabilities
|
||||||
|
- Performance problems
|
||||||
|
- Accessibility issues
|
||||||
|
- Standard violations (check against `/docs/CODING_STANDARDS.md`)
|
||||||
|
|
||||||
|
**2. After Each Phase:** Launch multi-agent deep review to:
|
||||||
|
- Verify phase objectives met
|
||||||
|
- Check integration with previous phases
|
||||||
|
- Identify critical issues requiring immediate fixes
|
||||||
|
- Recommend improvements before proceeding
|
||||||
|
- Update documentation if patterns evolved
|
||||||
|
|
||||||
|
**3. Testing Requirements:**
|
||||||
|
- Write tests alongside feature code (not after)
|
||||||
|
- Unit tests: All hooks, utilities, services
|
||||||
|
- Component tests: All reusable components
|
||||||
|
- Integration tests: All pages and flows
|
||||||
|
- E2E tests: Critical user journeys (auth, admin CRUD)
|
||||||
|
- Target: 90%+ coverage for template robustness
|
||||||
|
- Use Jest + React Testing Library + Playwright
|
||||||
|
|
||||||
|
**4. Context Preservation:**
|
||||||
|
- Update `/docs` with implementation decisions
|
||||||
|
- Document deviations from requirements in `ARCHITECTURE.md`
|
||||||
|
- Keep `frontend-requirements.md` updated if backend changes
|
||||||
|
- Update THIS FILE after each phase with actual progress
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Current System State (Phase 1 Complete)
|
||||||
|
|
||||||
|
### ✅ What's Implemented
|
||||||
|
|
||||||
|
**Project Infrastructure:**
|
||||||
|
- Next.js 15 with App Router
|
||||||
|
- TypeScript strict mode enabled
|
||||||
|
- Tailwind CSS 4 configured
|
||||||
|
- shadcn/ui components installed (15+ components)
|
||||||
|
- Path aliases configured (@/)
|
||||||
|
|
||||||
|
**Authentication System:**
|
||||||
|
- `src/lib/auth/crypto.ts` - AES-GCM encryption (82% coverage)
|
||||||
|
- `src/lib/auth/storage.ts` - Secure token storage (72.85% coverage)
|
||||||
|
- `src/stores/authStore.ts` - Zustand auth store (92.59% coverage)
|
||||||
|
- `src/config/app.config.ts` - Centralized configuration (81% coverage)
|
||||||
|
- SSR-safe implementations throughout
|
||||||
|
|
||||||
|
**API Integration:**
|
||||||
|
- `src/lib/api/client.ts` - Axios wrapper with interceptors (to be replaced)
|
||||||
|
- `src/lib/api/errors.ts` - Error parsing utilities (to be replaced)
|
||||||
|
- `scripts/generate-api-client.sh` - OpenAPI generation script
|
||||||
|
- **NOTE:** Manual client files marked for replacement with generated client
|
||||||
|
|
||||||
|
**Testing Infrastructure:**
|
||||||
|
- Jest configured with Next.js integration
|
||||||
|
- 66 tests passing (100%)
|
||||||
|
- 81.6% code coverage (exceeds 70% target)
|
||||||
|
- Real crypto testing (@peculiar/webcrypto)
|
||||||
|
- No mocks for security-critical code
|
||||||
|
|
||||||
|
**Documentation:**
|
||||||
|
- `/docs/ARCHITECTURE.md` - System design ✅
|
||||||
|
- `/docs/CODING_STANDARDS.md` - Code standards ✅
|
||||||
|
- `/docs/COMPONENT_GUIDE.md` - Component patterns ✅
|
||||||
|
- `/docs/FEATURE_EXAMPLES.md` - Implementation examples ✅
|
||||||
|
- `/docs/API_INTEGRATION.md` - API integration guide ✅
|
||||||
|
|
||||||
|
### 📊 Test Coverage Details
|
||||||
|
|
||||||
|
```
|
||||||
|
File | Statements | Branches | Functions | Lines
|
||||||
|
----------------|------------|----------|-----------|-------
|
||||||
|
All files | 81.60% | 84.09% | 93.10% | 82.08%
|
||||||
|
config | 81.08% | 81.25% | 80.00% | 84.37%
|
||||||
|
lib/auth | 76.85% | 73.07% | 92.30% | 76.66%
|
||||||
|
stores | 92.59% | 97.91% | 100.00% | 93.87%
|
||||||
|
```
|
||||||
|
|
||||||
|
**Coverage Exclusions (Properly Configured):**
|
||||||
|
- Auto-generated API client (`src/lib/api/generated/**`)
|
||||||
|
- Manual API client (to be replaced)
|
||||||
|
- Third-party UI components (`src/components/ui/**`)
|
||||||
|
- Next.js app directory (`src/app/**` - test with E2E)
|
||||||
|
- Re-export index files
|
||||||
|
- Old implementation files (`.old.ts`)
|
||||||
|
|
||||||
|
### 🎯 Quality Metrics
|
||||||
|
|
||||||
|
- ✅ TypeScript: 0 compilation errors
|
||||||
|
- ✅ ESLint: 0 warnings
|
||||||
|
- ✅ Tests: 66/66 passing
|
||||||
|
- ✅ Coverage: 81.6% (target: 70%)
|
||||||
|
- ✅ Security: No vulnerabilities
|
||||||
|
- ✅ SSR: All browser APIs properly guarded
|
||||||
|
|
||||||
|
### 📁 Current Folder Structure
|
||||||
|
|
||||||
|
```
|
||||||
|
frontend/
|
||||||
|
├── docs/ ✅ Phase 0 complete
|
||||||
|
│ ├── ARCHITECTURE.md
|
||||||
|
│ ├── CODING_STANDARDS.md
|
||||||
|
│ ├── COMPONENT_GUIDE.md
|
||||||
|
│ ├── FEATURE_EXAMPLES.md
|
||||||
|
│ └── API_INTEGRATION.md
|
||||||
|
├── src/
|
||||||
|
│ ├── app/ # Next.js app directory
|
||||||
|
│ ├── components/
|
||||||
|
│ │ └── ui/ # shadcn/ui components ✅
|
||||||
|
│ ├── lib/
|
||||||
|
│ │ ├── api/
|
||||||
|
│ │ │ ├── generated/ # OpenAPI client (empty, needs generation)
|
||||||
|
│ │ │ ├── client.ts # ✅ Axios wrapper (to replace)
|
||||||
|
│ │ │ └── errors.ts # ✅ Error parsing (to replace)
|
||||||
|
│ │ ├── auth/
|
||||||
|
│ │ │ ├── crypto.ts # ✅ 82% coverage
|
||||||
|
│ │ │ └── storage.ts # ✅ 72.85% coverage
|
||||||
|
│ │ └── utils/
|
||||||
|
│ ├── stores/
|
||||||
|
│ │ └── authStore.ts # ✅ 92.59% coverage
|
||||||
|
│ └── config/
|
||||||
|
│ └── app.config.ts # ✅ 81% coverage
|
||||||
|
├── tests/ # ✅ 66 tests
|
||||||
|
│ ├── lib/auth/ # Crypto & storage tests
|
||||||
|
│ ├── stores/ # Auth store tests
|
||||||
|
│ └── config/ # Config tests
|
||||||
|
├── scripts/
|
||||||
|
│ └── generate-api-client.sh # ✅ OpenAPI generation
|
||||||
|
├── jest.config.js # ✅ Configured
|
||||||
|
├── jest.setup.js # ✅ Global mocks
|
||||||
|
├── frontend-requirements.md # ✅ Updated
|
||||||
|
└── IMPLEMENTATION_PLAN.md # ✅ This file
|
||||||
|
|
||||||
|
```
|
||||||
|
|
||||||
|
### ⚠️ Known Technical Debt
|
||||||
|
|
||||||
|
1. **Manual API Client Files** - Need replacement when backend ready:
|
||||||
|
- Delete: `src/lib/api/client.ts`
|
||||||
|
- Delete: `src/lib/api/errors.ts`
|
||||||
|
- Run: `npm run generate:api`
|
||||||
|
- Create: Thin interceptor wrapper if needed
|
||||||
|
|
||||||
|
2. **Old Implementation Files** - Need cleanup:
|
||||||
|
- Delete: `src/stores/authStore.old.ts`
|
||||||
|
|
||||||
|
3. **API Client Generation** - Needs backend running:
|
||||||
|
- Backend must be at `http://localhost:8000`
|
||||||
|
- OpenAPI spec at `/api/v1/openapi.json`
|
||||||
|
- Run `npm run generate:api` to create client
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Phase 0: Foundation Documents & Requirements Alignment ✅
|
||||||
|
|
||||||
|
**Status:** COMPLETE
|
||||||
|
**Duration:** 1 day
|
||||||
|
**Completed:** October 31, 2025
|
||||||
|
|
||||||
|
### Task 0.1: Update Requirements Document ✅
|
||||||
|
- ✅ Updated `frontend-requirements.md` with API corrections
|
||||||
|
- ✅ Added Section 4.5 (Session Management UI)
|
||||||
|
- ✅ Added Section 15 (API Endpoint Reference)
|
||||||
|
- ✅ Updated auth flow with token rotation details
|
||||||
|
- ✅ Added missing User/Organization model fields
|
||||||
|
|
||||||
|
### Task 0.2: Create Architecture Documentation ✅
|
||||||
|
- ✅ Created `docs/ARCHITECTURE.md`
|
||||||
|
- ✅ System overview (Next.js App Router, TanStack Query, Zustand)
|
||||||
|
- ✅ Technology stack rationale
|
||||||
|
- ✅ Data flow diagrams
|
||||||
|
- ✅ Folder structure explanation
|
||||||
|
- ✅ Design patterns documented
|
||||||
|
|
||||||
|
### Task 0.3: Create Coding Standards Documentation ✅
|
||||||
|
- ✅ Created `docs/CODING_STANDARDS.md`
|
||||||
|
- ✅ TypeScript standards (strict mode, no any)
|
||||||
|
- ✅ React component patterns
|
||||||
|
- ✅ Naming conventions
|
||||||
|
- ✅ State management rules
|
||||||
|
- ✅ Form patterns
|
||||||
|
- ✅ Error handling patterns
|
||||||
|
- ✅ Testing standards
|
||||||
|
|
||||||
|
### Task 0.4: Create Component & Feature Guides ✅
|
||||||
|
- ✅ Created `docs/COMPONENT_GUIDE.md`
|
||||||
|
- ✅ Created `docs/FEATURE_EXAMPLES.md`
|
||||||
|
- ✅ Created `docs/API_INTEGRATION.md`
|
||||||
|
- ✅ Complete walkthroughs for common patterns
|
||||||
|
|
||||||
|
**Phase 0 Review:** ✅ All docs complete, clear, and accurate
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Phase 1: Project Setup & Infrastructure ✅
|
||||||
|
|
||||||
|
**Status:** COMPLETE
|
||||||
|
**Duration:** 3 days
|
||||||
|
**Completed:** October 31, 2025
|
||||||
|
|
||||||
|
### Task 1.1: Dependency Installation & Configuration ✅
|
||||||
|
**Status:** COMPLETE
|
||||||
|
**Blockers:** None
|
||||||
|
|
||||||
|
**Installed Dependencies:**
|
||||||
|
```bash
|
||||||
|
# Core
|
||||||
|
@tanstack/react-query@5, zustand@4, axios@1
|
||||||
|
@hey-api/openapi-ts (dev)
|
||||||
|
react-hook-form@7, zod@3, @hookform/resolvers
|
||||||
|
date-fns, clsx, tailwind-merge, lucide-react
|
||||||
|
recharts@2
|
||||||
|
|
||||||
|
# shadcn/ui
|
||||||
|
npx shadcn@latest init
|
||||||
|
npx shadcn@latest add button card input label form select table dialog
|
||||||
|
toast tabs dropdown-menu popover sheet avatar badge separator skeleton alert
|
||||||
|
|
||||||
|
# Testing
|
||||||
|
jest, @testing-library/react, @testing-library/jest-dom
|
||||||
|
@testing-library/user-event, @playwright/test, @types/jest
|
||||||
|
@peculiar/webcrypto (for real crypto in tests)
|
||||||
|
```
|
||||||
|
|
||||||
|
**Configuration:**
|
||||||
|
- ✅ `components.json` for shadcn/ui
|
||||||
|
- ✅ `tsconfig.json` with path aliases
|
||||||
|
- ✅ Tailwind configured for dark mode
|
||||||
|
- ✅ `.env.example` and `.env.local` created
|
||||||
|
- ✅ `jest.config.js` with Next.js integration
|
||||||
|
- ✅ `jest.setup.js` with global mocks
|
||||||
|
|
||||||
|
### Task 1.2: OpenAPI Client Generation Setup ✅
|
||||||
|
**Status:** COMPLETE
|
||||||
|
**Can run parallel with:** 1.3, 1.4
|
||||||
|
|
||||||
|
**Completed:**
|
||||||
|
- ✅ Created `scripts/generate-api-client.sh` using `@hey-api/openapi-ts`
|
||||||
|
- ✅ Configured output to `src/lib/api/generated/`
|
||||||
|
- ✅ Added npm script: `"generate:api": "./scripts/generate-api-client.sh"`
|
||||||
|
- ✅ Fixed deprecated options (removed `--name`, `--useOptions`, `--exportSchemas`)
|
||||||
|
- ✅ Used modern syntax: `--client @hey-api/client-axios`
|
||||||
|
- ✅ Successfully generated TypeScript client from backend API
|
||||||
|
- ✅ TypeScript compilation passes with generated types
|
||||||
|
|
||||||
|
**Generated Files:**
|
||||||
|
- `src/lib/api/generated/index.ts` - Main exports
|
||||||
|
- `src/lib/api/generated/types.gen.ts` - TypeScript types (35KB)
|
||||||
|
- `src/lib/api/generated/sdk.gen.ts` - API functions (29KB)
|
||||||
|
- `src/lib/api/generated/client.gen.ts` - Axios client
|
||||||
|
- `src/lib/api/generated/client/` - Client utilities
|
||||||
|
- `src/lib/api/generated/core/` - Core utilities
|
||||||
|
|
||||||
|
**To Regenerate (When Backend Changes):**
|
||||||
|
```bash
|
||||||
|
npm run generate:api
|
||||||
|
```
|
||||||
|
|
||||||
|
### Task 1.3: Axios Client & Interceptors ✅
|
||||||
|
**Status:** COMPLETE (needs replacement in Phase 2)
|
||||||
|
**Can run parallel with:** 1.2, 1.4
|
||||||
|
|
||||||
|
**Completed:**
|
||||||
|
- ✅ Created `src/lib/api/client.ts` - Axios wrapper
|
||||||
|
- Request interceptor: Add Authorization header
|
||||||
|
- Response interceptor: Handle 401, 403, 429, 500
|
||||||
|
- Error response parser
|
||||||
|
- Timeout configuration (30s default)
|
||||||
|
- Development logging
|
||||||
|
- ✅ Created `src/lib/api/errors.ts` - Error types and parsing
|
||||||
|
- ✅ Tests written for error parsing
|
||||||
|
|
||||||
|
**⚠️ Note:** This is a manual implementation. Will be replaced with generated client + thin interceptor wrapper once backend API is generated.
|
||||||
|
|
||||||
|
### Task 1.4: Folder Structure Creation ✅
|
||||||
|
**Status:** COMPLETE
|
||||||
|
**Can run parallel with:** 1.2, 1.3
|
||||||
|
|
||||||
|
**Completed:**
|
||||||
|
- ✅ All directories created per requirements
|
||||||
|
- ✅ Placeholder index.ts files for exports
|
||||||
|
- ✅ Structure matches `docs/ARCHITECTURE.md`
|
||||||
|
|
||||||
|
### Task 1.5: Authentication Core Implementation ✅
|
||||||
|
**Status:** COMPLETE (additional work beyond original plan)
|
||||||
|
|
||||||
|
**Completed:**
|
||||||
|
- ✅ `src/lib/auth/crypto.ts` - AES-GCM encryption with random IVs
|
||||||
|
- ✅ `src/lib/auth/storage.ts` - Encrypted token storage with localStorage
|
||||||
|
- ✅ `src/stores/authStore.ts` - Complete Zustand auth store
|
||||||
|
- ✅ `src/config/app.config.ts` - Centralized configuration with validation
|
||||||
|
- ✅ All SSR-safe with proper browser API guards
|
||||||
|
- ✅ 66 comprehensive tests written (81.6% coverage)
|
||||||
|
- ✅ Security audit completed
|
||||||
|
- ✅ Real crypto testing (no mocks)
|
||||||
|
|
||||||
|
**Security Features:**
|
||||||
|
- AES-GCM encryption with 256-bit keys
|
||||||
|
- Random IV per encryption
|
||||||
|
- Key stored in sessionStorage (per-session)
|
||||||
|
- Token validation (JWT format checking)
|
||||||
|
- Type-safe throughout
|
||||||
|
- No token leaks in logs
|
||||||
|
|
||||||
|
**Phase 1 Review:** ✅ Multi-agent audit completed. Infrastructure solid. All tests passing. Ready for Phase 2.
|
||||||
|
|
||||||
|
### Audit Results (October 31, 2025)
|
||||||
|
|
||||||
|
**Comprehensive audit conducted with the following results:**
|
||||||
|
|
||||||
|
**Critical Issues Found:** 5
|
||||||
|
**Critical Issues Fixed:** 5 ✅
|
||||||
|
|
||||||
|
**Issues Resolved:**
|
||||||
|
1. ✅ TypeScript compilation error (unused @ts-expect-error)
|
||||||
|
2. ✅ Duplicate configuration files
|
||||||
|
3. ✅ Test mocks didn't match real implementation
|
||||||
|
4. ✅ Test coverage properly configured
|
||||||
|
5. ✅ API client exclusions documented
|
||||||
|
|
||||||
|
**Final Metrics:**
|
||||||
|
- Tests: 66/66 passing (100%)
|
||||||
|
- Coverage: 81.6% (exceeds 70% target)
|
||||||
|
- TypeScript: 0 errors
|
||||||
|
- Security: No vulnerabilities
|
||||||
|
|
||||||
|
**Audit Documents:**
|
||||||
|
- `/tmp/AUDIT_SUMMARY.txt` - Executive summary
|
||||||
|
- `/tmp/AUDIT_COMPLETE.md` - Full report
|
||||||
|
- `/tmp/COVERAGE_CONFIG.md` - Coverage configuration
|
||||||
|
- `/tmp/detailed_findings.md` - Issue details
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Phase 2: Authentication System
|
||||||
|
|
||||||
|
**Status:** READY TO START 📋
|
||||||
|
**Duration:** 3-4 days
|
||||||
|
**Prerequisites:** Phase 1 complete ✅
|
||||||
|
|
||||||
|
**Context for Phase 2:**
|
||||||
|
Phase 1 already implemented core authentication infrastructure (crypto, storage, auth store). Phase 2 will build the UI layer on top of this foundation.
|
||||||
|
|
||||||
|
### Task 2.1: Token Storage & Auth Store ✅ (Done in Phase 1)
|
||||||
|
**Status:** COMPLETE (already done)
|
||||||
|
|
||||||
|
This was completed as part of Phase 1 infrastructure:
|
||||||
|
- ✅ `src/lib/auth/crypto.ts` - AES-GCM encryption
|
||||||
|
- ✅ `src/lib/auth/storage.ts` - Token storage utilities
|
||||||
|
- ✅ `src/stores/authStore.ts` - Complete Zustand store
|
||||||
|
- ✅ 92.59% test coverage on auth store
|
||||||
|
- ✅ Security audit passed
|
||||||
|
|
||||||
|
**Skip this task - move to 2.2**
|
||||||
|
|
||||||
|
### Task 2.2: Auth Interceptor Integration 🔗
|
||||||
|
**Status:** PARTIALLY COMPLETE (needs update)
|
||||||
|
**Depends on:** 2.1 ✅ (already complete)
|
||||||
|
|
||||||
|
**Current State:**
|
||||||
|
- `src/lib/api/client.ts` exists with basic interceptor logic
|
||||||
|
- Integrates with auth store
|
||||||
|
- Has token refresh flow
|
||||||
|
- Has retry mechanism
|
||||||
|
|
||||||
|
**Actions Needed:**
|
||||||
|
- [ ] Test with generated API client (once backend ready)
|
||||||
|
- [ ] Verify token rotation works
|
||||||
|
- [ ] Add race condition testing
|
||||||
|
- [ ] Verify no infinite refresh loops
|
||||||
|
|
||||||
|
**Reference:** `docs/API_INTEGRATION.md`, Requirements Section 5.2
|
||||||
|
|
||||||
|
### Task 2.3: Auth Hooks & Components 🔐
|
||||||
|
**Status:** TODO 📋
|
||||||
|
**Can run parallel with:** 2.4, 2.5 after 2.2 complete
|
||||||
|
|
||||||
|
**Actions Needed:**
|
||||||
|
|
||||||
|
Create React Query hooks in `src/lib/api/hooks/useAuth.ts`:
|
||||||
|
- [ ] `useLogin` - Login mutation
|
||||||
|
- [ ] `useRegister` - Register mutation
|
||||||
|
- [ ] `useLogout` - Logout mutation
|
||||||
|
- [ ] `useLogoutAll` - Logout all devices
|
||||||
|
- [ ] `useRefreshToken` - Token refresh
|
||||||
|
- [ ] `useMe` - Get current user
|
||||||
|
|
||||||
|
Create convenience hooks in `src/hooks/useAuth.ts`:
|
||||||
|
- [ ] Wrapper around auth store for easy component access
|
||||||
|
|
||||||
|
Create auth protection components:
|
||||||
|
- [ ] `src/components/auth/AuthGuard.tsx` - HOC for route protection
|
||||||
|
- [ ] `src/components/auth/ProtectedRoute.tsx` - Client component wrapper
|
||||||
|
|
||||||
|
**Testing:**
|
||||||
|
- [ ] Unit tests for each hook
|
||||||
|
- [ ] Test loading states
|
||||||
|
- [ ] Test error handling
|
||||||
|
- [ ] Test redirect logic
|
||||||
|
|
||||||
|
**Reference:** `docs/FEATURE_EXAMPLES.md` (auth patterns), Requirements Section 4.3
|
||||||
|
|
||||||
|
### Task 2.4: Login & Registration Pages 📄
|
||||||
|
**Status:** TODO 📋
|
||||||
|
**Can run parallel with:** 2.3, 2.5 after 2.2 complete
|
||||||
|
|
||||||
|
**Actions Needed:**
|
||||||
|
|
||||||
|
Create forms with validation:
|
||||||
|
- [ ] `src/components/auth/LoginForm.tsx`
|
||||||
|
- Email + password fields
|
||||||
|
- react-hook-form + zod validation
|
||||||
|
- Loading states
|
||||||
|
- Error display
|
||||||
|
- Remember me checkbox (optional)
|
||||||
|
- [ ] `src/components/auth/RegisterForm.tsx`
|
||||||
|
- Email, password, first_name, last_name
|
||||||
|
- Password confirmation field
|
||||||
|
- Password strength indicator
|
||||||
|
- Validation matching backend rules:
|
||||||
|
- Min 8 chars
|
||||||
|
- 1 digit
|
||||||
|
- 1 uppercase letter
|
||||||
|
|
||||||
|
Create pages:
|
||||||
|
- [ ] `src/app/(auth)/layout.tsx` - Centered form layout
|
||||||
|
- [ ] `src/app/(auth)/login/page.tsx` - Login page
|
||||||
|
- [ ] `src/app/(auth)/register/page.tsx` - Registration page
|
||||||
|
|
||||||
|
**API Endpoints:**
|
||||||
|
- POST `/api/v1/auth/register` - Register new user
|
||||||
|
- POST `/api/v1/auth/login` - Authenticate user
|
||||||
|
|
||||||
|
**Testing:**
|
||||||
|
- [ ] Form validation tests
|
||||||
|
- [ ] Submission success/error
|
||||||
|
- [ ] E2E login flow
|
||||||
|
- [ ] E2E registration flow
|
||||||
|
- [ ] Accessibility (keyboard nav, screen reader)
|
||||||
|
|
||||||
|
**Reference:** `docs/COMPONENT_GUIDE.md` (form patterns), Requirements Section 8.1
|
||||||
|
|
||||||
|
### Task 2.5: Password Reset Flow 🔑
|
||||||
|
**Status:** TODO 📋
|
||||||
|
**Can run parallel with:** 2.3, 2.4 after 2.2 complete
|
||||||
|
|
||||||
|
**Actions Needed:**
|
||||||
|
|
||||||
|
Create password reset pages:
|
||||||
|
- [ ] `src/app/(auth)/password-reset/page.tsx` - Request reset
|
||||||
|
- [ ] `src/app/(auth)/password-reset/confirm/page.tsx` - Confirm reset with token
|
||||||
|
|
||||||
|
Create forms:
|
||||||
|
- [ ] `src/components/auth/PasswordResetForm.tsx` - Email input form
|
||||||
|
- [ ] `src/components/auth/PasswordResetConfirmForm.tsx` - New password form
|
||||||
|
|
||||||
|
**Flow:**
|
||||||
|
1. User enters email → POST `/api/v1/auth/password-reset/request`
|
||||||
|
2. User receives email with token link
|
||||||
|
3. User clicks link → Opens confirm page with token in URL
|
||||||
|
4. User enters new password → POST `/api/v1/auth/password-reset/confirm`
|
||||||
|
|
||||||
|
**API Endpoints:**
|
||||||
|
- POST `/api/v1/auth/password-reset/request` - Request reset email
|
||||||
|
- POST `/api/v1/auth/password-reset/confirm` - Reset with token
|
||||||
|
|
||||||
|
**Testing:**
|
||||||
|
- [ ] Request form validation
|
||||||
|
- [ ] Email sent confirmation message
|
||||||
|
- [ ] Token validation
|
||||||
|
- [ ] Password update success
|
||||||
|
- [ ] Expired token handling
|
||||||
|
- [ ] E2E password reset flow
|
||||||
|
|
||||||
|
**Security Considerations:**
|
||||||
|
- [ ] Email enumeration protection (always show success)
|
||||||
|
- [ ] Token expiry handling
|
||||||
|
- [ ] Single-use tokens
|
||||||
|
|
||||||
|
**Reference:** Requirements Section 4.3, `docs/FEATURE_EXAMPLES.md`
|
||||||
|
|
||||||
|
### Phase 2 Review Checklist
|
||||||
|
|
||||||
|
When Phase 2 is complete, verify:
|
||||||
|
- [ ] All auth pages functional
|
||||||
|
- [ ] Forms have proper validation
|
||||||
|
- [ ] Error messages are user-friendly
|
||||||
|
- [ ] Loading states on all async operations
|
||||||
|
- [ ] E2E tests for full auth flows pass
|
||||||
|
- [ ] Security audit completed
|
||||||
|
- [ ] Accessibility audit completed
|
||||||
|
- [ ] No console errors
|
||||||
|
- [ ] Works in mobile viewport
|
||||||
|
- [ ] Dark mode works on all pages
|
||||||
|
|
||||||
|
**Before proceeding to Phase 3:**
|
||||||
|
- [ ] Run multi-agent review
|
||||||
|
- [ ] Security audit of auth implementation
|
||||||
|
- [ ] E2E test full auth flows
|
||||||
|
- [ ] Update this plan with actual progress
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Phase 3: User Profile & Settings
|
||||||
|
|
||||||
|
**Status:** TODO 📋
|
||||||
|
**Duration:** 3-4 days
|
||||||
|
**Prerequisites:** Phase 2 complete
|
||||||
|
|
||||||
|
**Detailed tasks will be added here after Phase 2 is complete.**
|
||||||
|
|
||||||
|
**High-level Overview:**
|
||||||
|
- Authenticated layout with navigation
|
||||||
|
- User profile management
|
||||||
|
- Password change
|
||||||
|
- Session management UI
|
||||||
|
- User preferences (optional)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Phase 4-12: Future Phases
|
||||||
|
|
||||||
|
**Status:** TODO 📋
|
||||||
|
|
||||||
|
**Remaining Phases:**
|
||||||
|
- **Phase 4:** Base Component Library & Layout
|
||||||
|
- **Phase 5:** Admin Dashboard Foundation
|
||||||
|
- **Phase 6:** User Management (Admin)
|
||||||
|
- **Phase 7:** Organization Management (Admin)
|
||||||
|
- **Phase 8:** Charts & Analytics
|
||||||
|
- **Phase 9:** Testing & Quality Assurance
|
||||||
|
- **Phase 10:** Documentation & Dev Tools
|
||||||
|
- **Phase 11:** Production Readiness & Optimization
|
||||||
|
- **Phase 12:** Final Integration & Handoff
|
||||||
|
|
||||||
|
**Note:** These phases will be detailed in this document as we progress through each phase. Context from completed phases will inform the implementation of future phases.
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Progress Tracking
|
||||||
|
|
||||||
|
### Overall Progress Dashboard
|
||||||
|
|
||||||
|
| Phase | Status | Started | Completed | Duration | Key Deliverables |
|
||||||
|
|-------|--------|---------|-----------|----------|------------------|
|
||||||
|
| 0: Foundation Docs | ✅ Complete | Oct 29 | Oct 29 | 1 day | 5 documentation files |
|
||||||
|
| 1: Infrastructure | ✅ Complete | Oct 29 | Oct 31 | 3 days | Setup + auth core + tests |
|
||||||
|
| 2: Auth System | 📋 TODO | - | - | 3-4 days | Login, register, reset flows |
|
||||||
|
| 3: User Settings | 📋 TODO | - | - | 3-4 days | Profile, password, sessions |
|
||||||
|
| 4: Component Library | 📋 TODO | - | - | 2-3 days | Common components |
|
||||||
|
| 5: Admin Foundation | 📋 TODO | - | - | 2-3 days | Admin layout, navigation |
|
||||||
|
| 6: User Management | 📋 TODO | - | - | 4-5 days | Admin user CRUD |
|
||||||
|
| 7: Org Management | 📋 TODO | - | - | 4-5 days | Admin org CRUD |
|
||||||
|
| 8: Charts | 📋 TODO | - | - | 2-3 days | Dashboard analytics |
|
||||||
|
| 9: Testing | 📋 TODO | - | - | 3-4 days | Comprehensive test suite |
|
||||||
|
| 10: Documentation | 📋 TODO | - | - | 2-3 days | Final docs |
|
||||||
|
| 11: Production Prep | 📋 TODO | - | - | 2-3 days | Performance, security |
|
||||||
|
| 12: Handoff | 📋 TODO | - | - | 1-2 days | Final validation |
|
||||||
|
|
||||||
|
**Current:** Phase 1 Complete, Ready for Phase 2
|
||||||
|
**Next:** Start Phase 2 - Authentication System UI
|
||||||
|
|
||||||
|
### Task Status Legend
|
||||||
|
- ✅ **Complete** - Finished and reviewed
|
||||||
|
- ⚙ **In Progress** - Currently being worked on
|
||||||
|
- 📋 **TODO** - Not started
|
||||||
|
- ❌ **Blocked** - Cannot proceed due to dependencies
|
||||||
|
- 🔗 **Depends on** - Waiting for specific task
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Critical Path & Dependencies
|
||||||
|
|
||||||
|
### Sequential Dependencies (Must Complete in Order)
|
||||||
|
|
||||||
|
1. **Phase 0** → Phase 1 (Foundation docs must exist before setup)
|
||||||
|
2. **Phase 1** → Phase 2 (Infrastructure needed for auth UI)
|
||||||
|
3. **Phase 2** → Phase 3 (Auth system needed for user features)
|
||||||
|
4. **Phase 1-4** → Phase 5 (Base components needed for admin)
|
||||||
|
5. **Phase 5** → Phase 6, 7 (Admin layout needed for CRUD)
|
||||||
|
|
||||||
|
### Parallelization Opportunities
|
||||||
|
|
||||||
|
**Within Phase 2 (After Task 2.2):**
|
||||||
|
- Tasks 2.3, 2.4, 2.5 can run in parallel (3 agents)
|
||||||
|
|
||||||
|
**Within Phase 3 (After Task 3.1):**
|
||||||
|
- Tasks 3.2, 3.3, 3.4, 3.5 can run in parallel (4 agents)
|
||||||
|
|
||||||
|
**Within Phase 4:**
|
||||||
|
- All tasks 4.1, 4.2, 4.3 can run in parallel (3 agents)
|
||||||
|
|
||||||
|
**Within Phase 5 (After Task 5.1):**
|
||||||
|
- Tasks 5.2, 5.3, 5.4 can run in parallel (3 agents)
|
||||||
|
|
||||||
|
**Phase 9 (Testing):**
|
||||||
|
- All testing tasks can run in parallel (4 agents)
|
||||||
|
|
||||||
|
**Estimated Timeline:**
|
||||||
|
- **With 4 parallel agents:** 8-10 weeks
|
||||||
|
- **With 2 parallel agents:** 12-14 weeks
|
||||||
|
- **With 1 agent (sequential):** 18-20 weeks
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Success Criteria
|
||||||
|
|
||||||
|
### Template is Production-Ready When:
|
||||||
|
|
||||||
|
1. ✅ All 12 phases complete
|
||||||
|
2. ✅ Test coverage ≥90% (unit + component + integration)
|
||||||
|
3. ✅ All E2E tests passing
|
||||||
|
4. ✅ Lighthouse scores:
|
||||||
|
- Performance >90
|
||||||
|
- Accessibility 100
|
||||||
|
- Best Practices >90
|
||||||
|
5. ✅ WCAG 2.1 Level AA compliance verified
|
||||||
|
6. ✅ No high/critical security vulnerabilities
|
||||||
|
7. ✅ All documentation complete and accurate
|
||||||
|
8. ✅ Production deployment successful
|
||||||
|
9. ✅ Frontend-backend integration verified
|
||||||
|
10. ✅ Template can be extended by new developer using docs alone
|
||||||
|
|
||||||
|
### Per-Phase Success Criteria
|
||||||
|
|
||||||
|
**Each phase must meet these before proceeding:**
|
||||||
|
- [ ] All tasks complete
|
||||||
|
- [ ] Tests written and passing
|
||||||
|
- [ ] Code reviewed (self + multi-agent)
|
||||||
|
- [ ] Documentation updated
|
||||||
|
- [ ] No regressions in previous functionality
|
||||||
|
- [ ] This plan updated with actual progress
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Critical Context for Resuming Work
|
||||||
|
|
||||||
|
### If Conversation is Interrupted
|
||||||
|
|
||||||
|
**To Resume Work, Read These Files in Order:**
|
||||||
|
|
||||||
|
1. **THIS FILE** - `IMPLEMENTATION_PLAN.md`
|
||||||
|
- Current phase and progress
|
||||||
|
- What's been completed
|
||||||
|
- What's next
|
||||||
|
|
||||||
|
2. **`frontend-requirements.md`**
|
||||||
|
- Complete feature requirements
|
||||||
|
- API endpoint reference
|
||||||
|
- User model details
|
||||||
|
|
||||||
|
3. **`docs/ARCHITECTURE.md`**
|
||||||
|
- System design
|
||||||
|
- Technology stack
|
||||||
|
- Data flow patterns
|
||||||
|
|
||||||
|
4. **`docs/CODING_STANDARDS.md`**
|
||||||
|
- Code style rules
|
||||||
|
- Testing standards
|
||||||
|
- Best practices
|
||||||
|
|
||||||
|
5. **`docs/FEATURE_EXAMPLES.md`**
|
||||||
|
- Implementation patterns
|
||||||
|
- Code examples
|
||||||
|
- Common pitfalls
|
||||||
|
|
||||||
|
### Key Commands Reference
|
||||||
|
|
||||||
|
```bash
|
||||||
|
# Development
|
||||||
|
npm run dev # Start dev server (http://localhost:3000)
|
||||||
|
npm run build # Production build
|
||||||
|
npm run start # Start production server
|
||||||
|
|
||||||
|
# Testing
|
||||||
|
npm test # Run tests
|
||||||
|
npm test -- --coverage # Run tests with coverage report
|
||||||
|
npm run type-check # TypeScript compilation check
|
||||||
|
npm run lint # ESLint check
|
||||||
|
|
||||||
|
# API Client Generation (needs backend running)
|
||||||
|
npm run generate:api # Generate TypeScript client from OpenAPI spec
|
||||||
|
|
||||||
|
# Package Management
|
||||||
|
npm install # Install dependencies
|
||||||
|
npm audit # Check for vulnerabilities
|
||||||
|
```
|
||||||
|
|
||||||
|
### Environment Variables
|
||||||
|
|
||||||
|
**Required:**
|
||||||
|
```env
|
||||||
|
NEXT_PUBLIC_API_URL=http://localhost:8000
|
||||||
|
NEXT_PUBLIC_APP_NAME=Template Project
|
||||||
|
```
|
||||||
|
|
||||||
|
**Optional:**
|
||||||
|
```env
|
||||||
|
NEXT_PUBLIC_API_TIMEOUT=30000
|
||||||
|
NEXT_PUBLIC_TOKEN_REFRESH_THRESHOLD=300000
|
||||||
|
NEXT_PUBLIC_DEBUG_API=false
|
||||||
|
```
|
||||||
|
|
||||||
|
See `.env.example` for complete list.
|
||||||
|
|
||||||
|
### Current Technical State
|
||||||
|
|
||||||
|
**What Works:**
|
||||||
|
- ✅ Authentication core (crypto, storage, store)
|
||||||
|
- ✅ Configuration management
|
||||||
|
- ✅ Test infrastructure
|
||||||
|
- ✅ TypeScript compilation
|
||||||
|
- ✅ Development environment
|
||||||
|
|
||||||
|
**What's Needed Next:**
|
||||||
|
- [ ] Generate API client from backend
|
||||||
|
- [ ] Build auth UI (login, register, password reset)
|
||||||
|
- [ ] Implement auth pages
|
||||||
|
- [ ] Add E2E tests for auth flows
|
||||||
|
|
||||||
|
**Technical Debt:**
|
||||||
|
- Manual API client files (will be replaced)
|
||||||
|
- Old implementation files (need cleanup)
|
||||||
|
- No API generation yet (needs backend)
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## References
|
||||||
|
|
||||||
|
### Always Reference During Implementation
|
||||||
|
|
||||||
|
**Primary Documents:**
|
||||||
|
- `IMPLEMENTATION_PLAN.md` (this file) - Implementation roadmap
|
||||||
|
- `frontend-requirements.md` - Detailed requirements
|
||||||
|
- `docs/ARCHITECTURE.md` - System design and patterns
|
||||||
|
- `docs/CODING_STANDARDS.md` - Code style and standards
|
||||||
|
- `docs/COMPONENT_GUIDE.md` - Component usage
|
||||||
|
- `docs/FEATURE_EXAMPLES.md` - Implementation examples
|
||||||
|
- `docs/API_INTEGRATION.md` - Backend API integration
|
||||||
|
|
||||||
|
**Backend References:**
|
||||||
|
- `../backend/docs/ARCHITECTURE.md` - Backend patterns to mirror
|
||||||
|
- `../backend/docs/CODING_STANDARDS.md` - Backend conventions
|
||||||
|
- Backend OpenAPI spec: `http://localhost:8000/api/v1/openapi.json`
|
||||||
|
|
||||||
|
**Testing References:**
|
||||||
|
- `jest.config.js` - Test configuration
|
||||||
|
- `jest.setup.js` - Global test setup
|
||||||
|
- `tests/` directory - Existing test patterns
|
||||||
|
|
||||||
|
### Audit & Quality Reports
|
||||||
|
|
||||||
|
**Available in `/tmp/`:**
|
||||||
|
- `AUDIT_SUMMARY.txt` - Quick reference
|
||||||
|
- `AUDIT_COMPLETE.md` - Full audit results
|
||||||
|
- `COVERAGE_CONFIG.md` - Coverage explanation
|
||||||
|
- `detailed_findings.md` - Issue analysis
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Version History
|
||||||
|
|
||||||
|
| Version | Date | Changes | Author |
|
||||||
|
|---------|------|---------|--------|
|
||||||
|
| 1.0 | Oct 29, 2025 | Initial plan created | Claude |
|
||||||
|
| 1.1 | Oct 31, 2025 | Phase 0 complete, updated structure | Claude |
|
||||||
|
| 1.2 | Oct 31, 2025 | Phase 1 complete, comprehensive audit | Claude |
|
||||||
|
| 1.3 | Oct 31, 2025 | **Major Update:** Reformatted as self-contained document | Claude |
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
## Notes for Future Development
|
||||||
|
|
||||||
|
### When Starting Phase 2
|
||||||
|
|
||||||
|
1. Generate API client first:
|
||||||
|
```bash
|
||||||
|
# Ensure backend is running
|
||||||
|
cd ../backend && uvicorn app.main:app --reload
|
||||||
|
|
||||||
|
# In separate terminal
|
||||||
|
cd frontend
|
||||||
|
npm run generate:api
|
||||||
|
```
|
||||||
|
|
||||||
|
2. Review generated types in `src/lib/api/generated/`
|
||||||
|
|
||||||
|
3. Replace manual client files:
|
||||||
|
- Archive or delete `src/lib/api/client.ts`
|
||||||
|
- Archive or delete `src/lib/api/errors.ts`
|
||||||
|
- Create thin wrapper if interceptor logic needed
|
||||||
|
|
||||||
|
4. Follow patterns in `docs/FEATURE_EXAMPLES.md`
|
||||||
|
|
||||||
|
5. Write tests alongside code (not after)
|
||||||
|
|
||||||
|
### Remember
|
||||||
|
|
||||||
|
- **Documentation First:** Check docs before implementing
|
||||||
|
- **Test As You Go:** Don't batch testing at end
|
||||||
|
- **Review Often:** Self-review after each task
|
||||||
|
- **Update This Plan:** Keep it current with actual progress
|
||||||
|
- **Context Matters:** This file + docs = full context
|
||||||
|
|
||||||
|
---
|
||||||
|
|
||||||
|
**Last Updated:** October 31, 2025
|
||||||
|
**Next Review:** After Phase 2 completion
|
||||||
|
**Contact:** Update this section with team contact info
|
||||||
@@ -48,10 +48,7 @@ echo -e "${YELLOW}⚙️ Generating TypeScript API client...${NC}"
|
|||||||
if npx @hey-api/openapi-ts \
|
if npx @hey-api/openapi-ts \
|
||||||
--input /tmp/openapi.json \
|
--input /tmp/openapi.json \
|
||||||
--output "$OUTPUT_DIR" \
|
--output "$OUTPUT_DIR" \
|
||||||
--client axios \
|
--client @hey-api/client-axios; then
|
||||||
--name ApiClient \
|
|
||||||
--useOptions true \
|
|
||||||
--exportSchemas true; then
|
|
||||||
echo -e "${GREEN}✓ API client generated successfully${NC}"
|
echo -e "${GREEN}✓ API client generated successfully${NC}"
|
||||||
else
|
else
|
||||||
echo -e "${RED}✗ Failed to generate API client${NC}"
|
echo -e "${RED}✗ Failed to generate API client${NC}"
|
||||||
|
|||||||
16
frontend/src/lib/api/generated/client.gen.ts
Normal file
16
frontend/src/lib/api/generated/client.gen.ts
Normal file
@@ -0,0 +1,16 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
import { type ClientOptions, type Config, createClient, createConfig } from './client';
|
||||||
|
import type { ClientOptions as ClientOptions2 } from './types.gen';
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The `createClientConfig()` function will be called on client initialization
|
||||||
|
* and the returned object will become the client's initial configuration.
|
||||||
|
*
|
||||||
|
* You may want to initialize your client this way instead of calling
|
||||||
|
* `setConfig()`. This is useful for example if you're using Next.js
|
||||||
|
* to ensure your client always has the correct values.
|
||||||
|
*/
|
||||||
|
export type CreateClientConfig<T extends ClientOptions = ClientOptions2> = (override?: Config<ClientOptions & T>) => Config<Required<ClientOptions> & T>;
|
||||||
|
|
||||||
|
export const client = createClient(createConfig<ClientOptions2>());
|
||||||
163
frontend/src/lib/api/generated/client/client.gen.ts
Normal file
163
frontend/src/lib/api/generated/client/client.gen.ts
Normal file
@@ -0,0 +1,163 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
import type { AxiosError, AxiosInstance, RawAxiosRequestHeaders } from 'axios';
|
||||||
|
import axios from 'axios';
|
||||||
|
|
||||||
|
import { createSseClient } from '../core/serverSentEvents.gen';
|
||||||
|
import type { HttpMethod } from '../core/types.gen';
|
||||||
|
import { getValidRequestBody } from '../core/utils.gen';
|
||||||
|
import type { Client, Config, RequestOptions } from './types.gen';
|
||||||
|
import {
|
||||||
|
buildUrl,
|
||||||
|
createConfig,
|
||||||
|
mergeConfigs,
|
||||||
|
mergeHeaders,
|
||||||
|
setAuthParams,
|
||||||
|
} from './utils.gen';
|
||||||
|
|
||||||
|
export const createClient = (config: Config = {}): Client => {
|
||||||
|
let _config = mergeConfigs(createConfig(), config);
|
||||||
|
|
||||||
|
let instance: AxiosInstance;
|
||||||
|
|
||||||
|
if (_config.axios && !('Axios' in _config.axios)) {
|
||||||
|
instance = _config.axios;
|
||||||
|
} else {
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
||||||
|
const { auth, ...configWithoutAuth } = _config;
|
||||||
|
instance = axios.create(configWithoutAuth);
|
||||||
|
}
|
||||||
|
|
||||||
|
const getConfig = (): Config => ({ ..._config });
|
||||||
|
|
||||||
|
const setConfig = (config: Config): Config => {
|
||||||
|
_config = mergeConfigs(_config, config);
|
||||||
|
instance.defaults = {
|
||||||
|
...instance.defaults,
|
||||||
|
..._config,
|
||||||
|
// @ts-expect-error
|
||||||
|
headers: mergeHeaders(instance.defaults.headers, _config.headers),
|
||||||
|
};
|
||||||
|
return getConfig();
|
||||||
|
};
|
||||||
|
|
||||||
|
const beforeRequest = async (options: RequestOptions) => {
|
||||||
|
const opts = {
|
||||||
|
..._config,
|
||||||
|
...options,
|
||||||
|
axios: options.axios ?? _config.axios ?? instance,
|
||||||
|
headers: mergeHeaders(_config.headers, options.headers),
|
||||||
|
};
|
||||||
|
|
||||||
|
if (opts.security) {
|
||||||
|
await setAuthParams({
|
||||||
|
...opts,
|
||||||
|
security: opts.security,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
if (opts.requestValidator) {
|
||||||
|
await opts.requestValidator(opts);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (opts.body !== undefined && opts.bodySerializer) {
|
||||||
|
opts.body = opts.bodySerializer(opts.body);
|
||||||
|
}
|
||||||
|
|
||||||
|
const url = buildUrl(opts);
|
||||||
|
|
||||||
|
return { opts, url };
|
||||||
|
};
|
||||||
|
|
||||||
|
// @ts-expect-error
|
||||||
|
const request: Client['request'] = async (options) => {
|
||||||
|
// @ts-expect-error
|
||||||
|
const { opts, url } = await beforeRequest(options);
|
||||||
|
try {
|
||||||
|
// assign Axios here for consistency with fetch
|
||||||
|
const _axios = opts.axios!;
|
||||||
|
// eslint-disable-next-line @typescript-eslint/no-unused-vars
|
||||||
|
const { auth, ...optsWithoutAuth } = opts;
|
||||||
|
const response = await _axios({
|
||||||
|
...optsWithoutAuth,
|
||||||
|
baseURL: '', // the baseURL is already included in `url`
|
||||||
|
data: getValidRequestBody(opts),
|
||||||
|
headers: opts.headers as RawAxiosRequestHeaders,
|
||||||
|
// let `paramsSerializer()` handle query params if it exists
|
||||||
|
params: opts.paramsSerializer ? opts.query : undefined,
|
||||||
|
url,
|
||||||
|
});
|
||||||
|
|
||||||
|
let { data } = response;
|
||||||
|
|
||||||
|
if (opts.responseType === 'json') {
|
||||||
|
if (opts.responseValidator) {
|
||||||
|
await opts.responseValidator(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (opts.responseTransformer) {
|
||||||
|
data = await opts.responseTransformer(data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
...response,
|
||||||
|
data: data ?? {},
|
||||||
|
};
|
||||||
|
} catch (error) {
|
||||||
|
const e = error as AxiosError;
|
||||||
|
if (opts.throwOnError) {
|
||||||
|
throw e;
|
||||||
|
}
|
||||||
|
// @ts-expect-error
|
||||||
|
e.error = e.response?.data ?? {};
|
||||||
|
return e;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const makeMethodFn =
|
||||||
|
(method: Uppercase<HttpMethod>) => (options: RequestOptions) =>
|
||||||
|
request({ ...options, method });
|
||||||
|
|
||||||
|
const makeSseFn =
|
||||||
|
(method: Uppercase<HttpMethod>) => async (options: RequestOptions) => {
|
||||||
|
const { opts, url } = await beforeRequest(options);
|
||||||
|
return createSseClient({
|
||||||
|
...opts,
|
||||||
|
body: opts.body as BodyInit | null | undefined,
|
||||||
|
headers: opts.headers as Record<string, string>,
|
||||||
|
method,
|
||||||
|
// @ts-expect-error
|
||||||
|
signal: opts.signal,
|
||||||
|
url,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
return {
|
||||||
|
buildUrl,
|
||||||
|
connect: makeMethodFn('CONNECT'),
|
||||||
|
delete: makeMethodFn('DELETE'),
|
||||||
|
get: makeMethodFn('GET'),
|
||||||
|
getConfig,
|
||||||
|
head: makeMethodFn('HEAD'),
|
||||||
|
instance,
|
||||||
|
options: makeMethodFn('OPTIONS'),
|
||||||
|
patch: makeMethodFn('PATCH'),
|
||||||
|
post: makeMethodFn('POST'),
|
||||||
|
put: makeMethodFn('PUT'),
|
||||||
|
request,
|
||||||
|
setConfig,
|
||||||
|
sse: {
|
||||||
|
connect: makeSseFn('CONNECT'),
|
||||||
|
delete: makeSseFn('DELETE'),
|
||||||
|
get: makeSseFn('GET'),
|
||||||
|
head: makeSseFn('HEAD'),
|
||||||
|
options: makeSseFn('OPTIONS'),
|
||||||
|
patch: makeSseFn('PATCH'),
|
||||||
|
post: makeSseFn('POST'),
|
||||||
|
put: makeSseFn('PUT'),
|
||||||
|
trace: makeSseFn('TRACE'),
|
||||||
|
},
|
||||||
|
trace: makeMethodFn('TRACE'),
|
||||||
|
} as Client;
|
||||||
|
};
|
||||||
24
frontend/src/lib/api/generated/client/index.ts
Normal file
24
frontend/src/lib/api/generated/client/index.ts
Normal file
@@ -0,0 +1,24 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
export type { Auth } from '../core/auth.gen';
|
||||||
|
export type { QuerySerializerOptions } from '../core/bodySerializer.gen';
|
||||||
|
export {
|
||||||
|
formDataBodySerializer,
|
||||||
|
jsonBodySerializer,
|
||||||
|
urlSearchParamsBodySerializer,
|
||||||
|
} from '../core/bodySerializer.gen';
|
||||||
|
export { buildClientParams } from '../core/params.gen';
|
||||||
|
export { serializeQueryKeyValue } from '../core/queryKeySerializer.gen';
|
||||||
|
export { createClient } from './client.gen';
|
||||||
|
export type {
|
||||||
|
Client,
|
||||||
|
ClientOptions,
|
||||||
|
Config,
|
||||||
|
CreateClientConfig,
|
||||||
|
Options,
|
||||||
|
OptionsLegacyParser,
|
||||||
|
RequestOptions,
|
||||||
|
RequestResult,
|
||||||
|
TDataShape,
|
||||||
|
} from './types.gen';
|
||||||
|
export { createConfig } from './utils.gen';
|
||||||
216
frontend/src/lib/api/generated/client/types.gen.ts
Normal file
216
frontend/src/lib/api/generated/client/types.gen.ts
Normal file
@@ -0,0 +1,216 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
import type {
|
||||||
|
AxiosError,
|
||||||
|
AxiosInstance,
|
||||||
|
AxiosRequestHeaders,
|
||||||
|
AxiosResponse,
|
||||||
|
AxiosStatic,
|
||||||
|
CreateAxiosDefaults,
|
||||||
|
} from 'axios';
|
||||||
|
|
||||||
|
import type { Auth } from '../core/auth.gen';
|
||||||
|
import type {
|
||||||
|
ServerSentEventsOptions,
|
||||||
|
ServerSentEventsResult,
|
||||||
|
} from '../core/serverSentEvents.gen';
|
||||||
|
import type {
|
||||||
|
Client as CoreClient,
|
||||||
|
Config as CoreConfig,
|
||||||
|
} from '../core/types.gen';
|
||||||
|
|
||||||
|
export interface Config<T extends ClientOptions = ClientOptions>
|
||||||
|
extends Omit<CreateAxiosDefaults, 'auth' | 'baseURL' | 'headers' | 'method'>,
|
||||||
|
CoreConfig {
|
||||||
|
/**
|
||||||
|
* Axios implementation. You can use this option to provide either an
|
||||||
|
* `AxiosStatic` or an `AxiosInstance`.
|
||||||
|
*
|
||||||
|
* @default axios
|
||||||
|
*/
|
||||||
|
axios?: AxiosStatic | AxiosInstance;
|
||||||
|
/**
|
||||||
|
* Base URL for all requests made by this client.
|
||||||
|
*/
|
||||||
|
baseURL?: T['baseURL'];
|
||||||
|
/**
|
||||||
|
* An object containing any HTTP headers that you want to pre-populate your
|
||||||
|
* `Headers` object with.
|
||||||
|
*
|
||||||
|
* {@link https://developer.mozilla.org/docs/Web/API/Headers/Headers#init See more}
|
||||||
|
*/
|
||||||
|
headers?:
|
||||||
|
| AxiosRequestHeaders
|
||||||
|
| Record<
|
||||||
|
string,
|
||||||
|
| string
|
||||||
|
| number
|
||||||
|
| boolean
|
||||||
|
| (string | number | boolean)[]
|
||||||
|
| null
|
||||||
|
| undefined
|
||||||
|
| unknown
|
||||||
|
>;
|
||||||
|
/**
|
||||||
|
* Throw an error instead of returning it in the response?
|
||||||
|
*
|
||||||
|
* @default false
|
||||||
|
*/
|
||||||
|
throwOnError?: T['throwOnError'];
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface RequestOptions<
|
||||||
|
TData = unknown,
|
||||||
|
ThrowOnError extends boolean = boolean,
|
||||||
|
Url extends string = string,
|
||||||
|
> extends Config<{
|
||||||
|
throwOnError: ThrowOnError;
|
||||||
|
}>,
|
||||||
|
Pick<
|
||||||
|
ServerSentEventsOptions<TData>,
|
||||||
|
| 'onSseError'
|
||||||
|
| 'onSseEvent'
|
||||||
|
| 'sseDefaultRetryDelay'
|
||||||
|
| 'sseMaxRetryAttempts'
|
||||||
|
| 'sseMaxRetryDelay'
|
||||||
|
> {
|
||||||
|
/**
|
||||||
|
* Any body that you want to add to your request.
|
||||||
|
*
|
||||||
|
* {@link https://developer.mozilla.org/docs/Web/API/fetch#body}
|
||||||
|
*/
|
||||||
|
body?: unknown;
|
||||||
|
path?: Record<string, unknown>;
|
||||||
|
query?: Record<string, unknown>;
|
||||||
|
/**
|
||||||
|
* Security mechanism(s) to use for the request.
|
||||||
|
*/
|
||||||
|
security?: ReadonlyArray<Auth>;
|
||||||
|
url: Url;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface ClientOptions {
|
||||||
|
baseURL?: string;
|
||||||
|
throwOnError?: boolean;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type RequestResult<
|
||||||
|
TData = unknown,
|
||||||
|
TError = unknown,
|
||||||
|
ThrowOnError extends boolean = boolean,
|
||||||
|
> = ThrowOnError extends true
|
||||||
|
? Promise<
|
||||||
|
AxiosResponse<
|
||||||
|
TData extends Record<string, unknown> ? TData[keyof TData] : TData
|
||||||
|
>
|
||||||
|
>
|
||||||
|
: Promise<
|
||||||
|
| (AxiosResponse<
|
||||||
|
TData extends Record<string, unknown> ? TData[keyof TData] : TData
|
||||||
|
> & { error: undefined })
|
||||||
|
| (AxiosError<
|
||||||
|
TError extends Record<string, unknown> ? TError[keyof TError] : TError
|
||||||
|
> & {
|
||||||
|
data: undefined;
|
||||||
|
error: TError extends Record<string, unknown>
|
||||||
|
? TError[keyof TError]
|
||||||
|
: TError;
|
||||||
|
})
|
||||||
|
>;
|
||||||
|
|
||||||
|
type MethodFn = <
|
||||||
|
TData = unknown,
|
||||||
|
TError = unknown,
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
>(
|
||||||
|
options: Omit<RequestOptions<TData, ThrowOnError>, 'method'>,
|
||||||
|
) => RequestResult<TData, TError, ThrowOnError>;
|
||||||
|
|
||||||
|
type SseFn = <
|
||||||
|
TData = unknown,
|
||||||
|
TError = unknown,
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
>(
|
||||||
|
options: Omit<RequestOptions<TData, ThrowOnError>, 'method'>,
|
||||||
|
) => Promise<ServerSentEventsResult<TData, TError>>;
|
||||||
|
|
||||||
|
type RequestFn = <
|
||||||
|
TData = unknown,
|
||||||
|
TError = unknown,
|
||||||
|
ThrowOnError extends boolean = false,
|
||||||
|
>(
|
||||||
|
options: Omit<RequestOptions<TData, ThrowOnError>, 'method'> &
|
||||||
|
Pick<Required<RequestOptions<TData, ThrowOnError>>, 'method'>,
|
||||||
|
) => RequestResult<TData, TError, ThrowOnError>;
|
||||||
|
|
||||||
|
type BuildUrlFn = <
|
||||||
|
TData extends {
|
||||||
|
body?: unknown;
|
||||||
|
path?: Record<string, unknown>;
|
||||||
|
query?: Record<string, unknown>;
|
||||||
|
url: string;
|
||||||
|
},
|
||||||
|
>(
|
||||||
|
options: Pick<TData, 'url'> & Options<TData>,
|
||||||
|
) => string;
|
||||||
|
|
||||||
|
export type Client = CoreClient<
|
||||||
|
RequestFn,
|
||||||
|
Config,
|
||||||
|
MethodFn,
|
||||||
|
BuildUrlFn,
|
||||||
|
SseFn
|
||||||
|
> & {
|
||||||
|
instance: AxiosInstance;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* The `createClientConfig()` function will be called on client initialization
|
||||||
|
* and the returned object will become the client's initial configuration.
|
||||||
|
*
|
||||||
|
* You may want to initialize your client this way instead of calling
|
||||||
|
* `setConfig()`. This is useful for example if you're using Next.js
|
||||||
|
* to ensure your client always has the correct values.
|
||||||
|
*/
|
||||||
|
export type CreateClientConfig<T extends ClientOptions = ClientOptions> = (
|
||||||
|
override?: Config<ClientOptions & T>,
|
||||||
|
) => Config<Required<ClientOptions> & T>;
|
||||||
|
|
||||||
|
export interface TDataShape {
|
||||||
|
body?: unknown;
|
||||||
|
headers?: unknown;
|
||||||
|
path?: unknown;
|
||||||
|
query?: unknown;
|
||||||
|
url: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
type OmitKeys<T, K> = Pick<T, Exclude<keyof T, K>>;
|
||||||
|
|
||||||
|
export type Options<
|
||||||
|
TData extends TDataShape = TDataShape,
|
||||||
|
ThrowOnError extends boolean = boolean,
|
||||||
|
TResponse = unknown,
|
||||||
|
> = OmitKeys<
|
||||||
|
RequestOptions<TResponse, ThrowOnError>,
|
||||||
|
'body' | 'path' | 'query' | 'url'
|
||||||
|
> &
|
||||||
|
Omit<TData, 'url'>;
|
||||||
|
|
||||||
|
export type OptionsLegacyParser<
|
||||||
|
TData = unknown,
|
||||||
|
ThrowOnError extends boolean = boolean,
|
||||||
|
> = TData extends { body?: any }
|
||||||
|
? TData extends { headers?: any }
|
||||||
|
? OmitKeys<
|
||||||
|
RequestOptions<unknown, ThrowOnError>,
|
||||||
|
'body' | 'headers' | 'url'
|
||||||
|
> &
|
||||||
|
TData
|
||||||
|
: OmitKeys<RequestOptions<unknown, ThrowOnError>, 'body' | 'url'> &
|
||||||
|
TData &
|
||||||
|
Pick<RequestOptions<unknown, ThrowOnError>, 'headers'>
|
||||||
|
: TData extends { headers?: any }
|
||||||
|
? OmitKeys<RequestOptions<unknown, ThrowOnError>, 'headers' | 'url'> &
|
||||||
|
TData &
|
||||||
|
Pick<RequestOptions<unknown, ThrowOnError>, 'body'>
|
||||||
|
: OmitKeys<RequestOptions<unknown, ThrowOnError>, 'url'> & TData;
|
||||||
213
frontend/src/lib/api/generated/client/utils.gen.ts
Normal file
213
frontend/src/lib/api/generated/client/utils.gen.ts
Normal file
@@ -0,0 +1,213 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
import { getAuthToken } from '../core/auth.gen';
|
||||||
|
import type { QuerySerializerOptions } from '../core/bodySerializer.gen';
|
||||||
|
import {
|
||||||
|
serializeArrayParam,
|
||||||
|
serializeObjectParam,
|
||||||
|
serializePrimitiveParam,
|
||||||
|
} from '../core/pathSerializer.gen';
|
||||||
|
import { getUrl } from '../core/utils.gen';
|
||||||
|
import type { Client, ClientOptions, Config, RequestOptions } from './types.gen';
|
||||||
|
|
||||||
|
export const createQuerySerializer = <T = unknown>({
|
||||||
|
parameters = {},
|
||||||
|
...args
|
||||||
|
}: QuerySerializerOptions = {}) => {
|
||||||
|
const querySerializer = (queryParams: T) => {
|
||||||
|
const search: string[] = [];
|
||||||
|
if (queryParams && typeof queryParams === 'object') {
|
||||||
|
for (const name in queryParams) {
|
||||||
|
const value = queryParams[name];
|
||||||
|
|
||||||
|
if (value === undefined || value === null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const options = parameters[name] || args;
|
||||||
|
|
||||||
|
if (Array.isArray(value)) {
|
||||||
|
const serializedArray = serializeArrayParam({
|
||||||
|
allowReserved: options.allowReserved,
|
||||||
|
explode: true,
|
||||||
|
name,
|
||||||
|
style: 'form',
|
||||||
|
value,
|
||||||
|
...options.array,
|
||||||
|
});
|
||||||
|
if (serializedArray) search.push(serializedArray);
|
||||||
|
} else if (typeof value === 'object') {
|
||||||
|
const serializedObject = serializeObjectParam({
|
||||||
|
allowReserved: options.allowReserved,
|
||||||
|
explode: true,
|
||||||
|
name,
|
||||||
|
style: 'deepObject',
|
||||||
|
value: value as Record<string, unknown>,
|
||||||
|
...options.object,
|
||||||
|
});
|
||||||
|
if (serializedObject) search.push(serializedObject);
|
||||||
|
} else {
|
||||||
|
const serializedPrimitive = serializePrimitiveParam({
|
||||||
|
allowReserved: options.allowReserved,
|
||||||
|
name,
|
||||||
|
value: value as string,
|
||||||
|
});
|
||||||
|
if (serializedPrimitive) search.push(serializedPrimitive);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return search.join('&');
|
||||||
|
};
|
||||||
|
return querySerializer;
|
||||||
|
};
|
||||||
|
|
||||||
|
const checkForExistence = (
|
||||||
|
options: Pick<RequestOptions, 'auth' | 'query'> & {
|
||||||
|
headers: Record<any, unknown>;
|
||||||
|
},
|
||||||
|
name?: string,
|
||||||
|
): boolean => {
|
||||||
|
if (!name) {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
if (name in options.headers || options.query?.[name]) {
|
||||||
|
return true;
|
||||||
|
}
|
||||||
|
if (
|
||||||
|
'Cookie' in options.headers &&
|
||||||
|
options.headers['Cookie'] &&
|
||||||
|
typeof options.headers['Cookie'] === 'string'
|
||||||
|
) {
|
||||||
|
return options.headers['Cookie'].includes(`${name}=`);
|
||||||
|
}
|
||||||
|
return false;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const setAuthParams = async ({
|
||||||
|
security,
|
||||||
|
...options
|
||||||
|
}: Pick<Required<RequestOptions>, 'security'> &
|
||||||
|
Pick<RequestOptions, 'auth' | 'query'> & {
|
||||||
|
headers: Record<any, unknown>;
|
||||||
|
}) => {
|
||||||
|
for (const auth of security) {
|
||||||
|
if (checkForExistence(options, auth.name)) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
const token = await getAuthToken(auth, options.auth);
|
||||||
|
|
||||||
|
if (!token) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const name = auth.name ?? 'Authorization';
|
||||||
|
|
||||||
|
switch (auth.in) {
|
||||||
|
case 'query':
|
||||||
|
if (!options.query) {
|
||||||
|
options.query = {};
|
||||||
|
}
|
||||||
|
options.query[name] = token;
|
||||||
|
break;
|
||||||
|
case 'cookie': {
|
||||||
|
const value = `${name}=${token}`;
|
||||||
|
if ('Cookie' in options.headers && options.headers['Cookie']) {
|
||||||
|
options.headers['Cookie'] = `${options.headers['Cookie']}; ${value}`;
|
||||||
|
} else {
|
||||||
|
options.headers['Cookie'] = value;
|
||||||
|
}
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
case 'header':
|
||||||
|
default:
|
||||||
|
options.headers[name] = token;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const buildUrl: Client['buildUrl'] = (options) => {
|
||||||
|
const instanceBaseUrl = options.axios?.defaults?.baseURL;
|
||||||
|
|
||||||
|
const baseUrl =
|
||||||
|
!!options.baseURL && typeof options.baseURL === 'string'
|
||||||
|
? options.baseURL
|
||||||
|
: instanceBaseUrl;
|
||||||
|
|
||||||
|
return getUrl({
|
||||||
|
baseUrl: baseUrl as string,
|
||||||
|
path: options.path,
|
||||||
|
// let `paramsSerializer()` handle query params if it exists
|
||||||
|
query: !options.paramsSerializer ? options.query : undefined,
|
||||||
|
querySerializer:
|
||||||
|
typeof options.querySerializer === 'function'
|
||||||
|
? options.querySerializer
|
||||||
|
: createQuerySerializer(options.querySerializer),
|
||||||
|
url: options.url,
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
export const mergeConfigs = (a: Config, b: Config): Config => {
|
||||||
|
const config = { ...a, ...b };
|
||||||
|
config.headers = mergeHeaders(a.headers, b.headers);
|
||||||
|
return config;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Special Axios headers keywords allowing to set headers by request method.
|
||||||
|
*/
|
||||||
|
export const axiosHeadersKeywords = [
|
||||||
|
'common',
|
||||||
|
'delete',
|
||||||
|
'get',
|
||||||
|
'head',
|
||||||
|
'patch',
|
||||||
|
'post',
|
||||||
|
'put',
|
||||||
|
] as const;
|
||||||
|
|
||||||
|
export const mergeHeaders = (
|
||||||
|
...headers: Array<Required<Config>['headers'] | undefined>
|
||||||
|
): Record<any, unknown> => {
|
||||||
|
const mergedHeaders: Record<any, unknown> = {};
|
||||||
|
for (const header of headers) {
|
||||||
|
if (!header || typeof header !== 'object') {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const iterator = Object.entries(header);
|
||||||
|
|
||||||
|
for (const [key, value] of iterator) {
|
||||||
|
if (
|
||||||
|
axiosHeadersKeywords.includes(
|
||||||
|
key as (typeof axiosHeadersKeywords)[number],
|
||||||
|
) &&
|
||||||
|
typeof value === 'object'
|
||||||
|
) {
|
||||||
|
mergedHeaders[key] = {
|
||||||
|
...(mergedHeaders[key] as Record<any, unknown>),
|
||||||
|
...value,
|
||||||
|
};
|
||||||
|
} else if (value === null) {
|
||||||
|
delete mergedHeaders[key];
|
||||||
|
} else if (Array.isArray(value)) {
|
||||||
|
for (const v of value) {
|
||||||
|
// @ts-expect-error
|
||||||
|
mergedHeaders[key] = [...(mergedHeaders[key] ?? []), v as string];
|
||||||
|
}
|
||||||
|
} else if (value !== undefined) {
|
||||||
|
// assume object headers are meant to be JSON stringified, i.e. their
|
||||||
|
// content value in OpenAPI specification is 'application/json'
|
||||||
|
mergedHeaders[key] =
|
||||||
|
typeof value === 'object' ? JSON.stringify(value) : (value as string);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return mergedHeaders;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const createConfig = <T extends ClientOptions = ClientOptions>(
|
||||||
|
override: Config<Omit<ClientOptions, keyof T> & T> = {},
|
||||||
|
): Config<Omit<ClientOptions, keyof T> & T> => ({
|
||||||
|
...override,
|
||||||
|
});
|
||||||
42
frontend/src/lib/api/generated/core/auth.gen.ts
Normal file
42
frontend/src/lib/api/generated/core/auth.gen.ts
Normal file
@@ -0,0 +1,42 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
export type AuthToken = string | undefined;
|
||||||
|
|
||||||
|
export interface Auth {
|
||||||
|
/**
|
||||||
|
* Which part of the request do we use to send the auth?
|
||||||
|
*
|
||||||
|
* @default 'header'
|
||||||
|
*/
|
||||||
|
in?: 'header' | 'query' | 'cookie';
|
||||||
|
/**
|
||||||
|
* Header or query parameter name.
|
||||||
|
*
|
||||||
|
* @default 'Authorization'
|
||||||
|
*/
|
||||||
|
name?: string;
|
||||||
|
scheme?: 'basic' | 'bearer';
|
||||||
|
type: 'apiKey' | 'http';
|
||||||
|
}
|
||||||
|
|
||||||
|
export const getAuthToken = async (
|
||||||
|
auth: Auth,
|
||||||
|
callback: ((auth: Auth) => Promise<AuthToken> | AuthToken) | AuthToken,
|
||||||
|
): Promise<string | undefined> => {
|
||||||
|
const token =
|
||||||
|
typeof callback === 'function' ? await callback(auth) : callback;
|
||||||
|
|
||||||
|
if (!token) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auth.scheme === 'bearer') {
|
||||||
|
return `Bearer ${token}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (auth.scheme === 'basic') {
|
||||||
|
return `Basic ${btoa(token)}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
return token;
|
||||||
|
};
|
||||||
100
frontend/src/lib/api/generated/core/bodySerializer.gen.ts
Normal file
100
frontend/src/lib/api/generated/core/bodySerializer.gen.ts
Normal file
@@ -0,0 +1,100 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
import type {
|
||||||
|
ArrayStyle,
|
||||||
|
ObjectStyle,
|
||||||
|
SerializerOptions,
|
||||||
|
} from './pathSerializer.gen';
|
||||||
|
|
||||||
|
export type QuerySerializer = (query: Record<string, unknown>) => string;
|
||||||
|
|
||||||
|
export type BodySerializer = (body: any) => any;
|
||||||
|
|
||||||
|
type QuerySerializerOptionsObject = {
|
||||||
|
allowReserved?: boolean;
|
||||||
|
array?: Partial<SerializerOptions<ArrayStyle>>;
|
||||||
|
object?: Partial<SerializerOptions<ObjectStyle>>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export type QuerySerializerOptions = QuerySerializerOptionsObject & {
|
||||||
|
/**
|
||||||
|
* Per-parameter serialization overrides. When provided, these settings
|
||||||
|
* override the global array/object settings for specific parameter names.
|
||||||
|
*/
|
||||||
|
parameters?: Record<string, QuerySerializerOptionsObject>;
|
||||||
|
};
|
||||||
|
|
||||||
|
const serializeFormDataPair = (
|
||||||
|
data: FormData,
|
||||||
|
key: string,
|
||||||
|
value: unknown,
|
||||||
|
): void => {
|
||||||
|
if (typeof value === 'string' || value instanceof Blob) {
|
||||||
|
data.append(key, value);
|
||||||
|
} else if (value instanceof Date) {
|
||||||
|
data.append(key, value.toISOString());
|
||||||
|
} else {
|
||||||
|
data.append(key, JSON.stringify(value));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const serializeUrlSearchParamsPair = (
|
||||||
|
data: URLSearchParams,
|
||||||
|
key: string,
|
||||||
|
value: unknown,
|
||||||
|
): void => {
|
||||||
|
if (typeof value === 'string') {
|
||||||
|
data.append(key, value);
|
||||||
|
} else {
|
||||||
|
data.append(key, JSON.stringify(value));
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const formDataBodySerializer = {
|
||||||
|
bodySerializer: <T extends Record<string, any> | Array<Record<string, any>>>(
|
||||||
|
body: T,
|
||||||
|
): FormData => {
|
||||||
|
const data = new FormData();
|
||||||
|
|
||||||
|
Object.entries(body).forEach(([key, value]) => {
|
||||||
|
if (value === undefined || value === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (Array.isArray(value)) {
|
||||||
|
value.forEach((v) => serializeFormDataPair(data, key, v));
|
||||||
|
} else {
|
||||||
|
serializeFormDataPair(data, key, value);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return data;
|
||||||
|
},
|
||||||
|
};
|
||||||
|
|
||||||
|
export const jsonBodySerializer = {
|
||||||
|
bodySerializer: <T>(body: T): string =>
|
||||||
|
JSON.stringify(body, (_key, value) =>
|
||||||
|
typeof value === 'bigint' ? value.toString() : value,
|
||||||
|
),
|
||||||
|
};
|
||||||
|
|
||||||
|
export const urlSearchParamsBodySerializer = {
|
||||||
|
bodySerializer: <T extends Record<string, any> | Array<Record<string, any>>>(
|
||||||
|
body: T,
|
||||||
|
): string => {
|
||||||
|
const data = new URLSearchParams();
|
||||||
|
|
||||||
|
Object.entries(body).forEach(([key, value]) => {
|
||||||
|
if (value === undefined || value === null) {
|
||||||
|
return;
|
||||||
|
}
|
||||||
|
if (Array.isArray(value)) {
|
||||||
|
value.forEach((v) => serializeUrlSearchParamsPair(data, key, v));
|
||||||
|
} else {
|
||||||
|
serializeUrlSearchParamsPair(data, key, value);
|
||||||
|
}
|
||||||
|
});
|
||||||
|
|
||||||
|
return data.toString();
|
||||||
|
},
|
||||||
|
};
|
||||||
153
frontend/src/lib/api/generated/core/params.gen.ts
Normal file
153
frontend/src/lib/api/generated/core/params.gen.ts
Normal file
@@ -0,0 +1,153 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
type Slot = 'body' | 'headers' | 'path' | 'query';
|
||||||
|
|
||||||
|
export type Field =
|
||||||
|
| {
|
||||||
|
in: Exclude<Slot, 'body'>;
|
||||||
|
/**
|
||||||
|
* Field name. This is the name we want the user to see and use.
|
||||||
|
*/
|
||||||
|
key: string;
|
||||||
|
/**
|
||||||
|
* Field mapped name. This is the name we want to use in the request.
|
||||||
|
* If omitted, we use the same value as `key`.
|
||||||
|
*/
|
||||||
|
map?: string;
|
||||||
|
}
|
||||||
|
| {
|
||||||
|
in: Extract<Slot, 'body'>;
|
||||||
|
/**
|
||||||
|
* Key isn't required for bodies.
|
||||||
|
*/
|
||||||
|
key?: string;
|
||||||
|
map?: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export interface Fields {
|
||||||
|
allowExtra?: Partial<Record<Slot, boolean>>;
|
||||||
|
args?: ReadonlyArray<Field>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type FieldsConfig = ReadonlyArray<Field | Fields>;
|
||||||
|
|
||||||
|
const extraPrefixesMap: Record<string, Slot> = {
|
||||||
|
$body_: 'body',
|
||||||
|
$headers_: 'headers',
|
||||||
|
$path_: 'path',
|
||||||
|
$query_: 'query',
|
||||||
|
};
|
||||||
|
const extraPrefixes = Object.entries(extraPrefixesMap);
|
||||||
|
|
||||||
|
type KeyMap = Map<
|
||||||
|
string,
|
||||||
|
{
|
||||||
|
in: Slot;
|
||||||
|
map?: string;
|
||||||
|
}
|
||||||
|
>;
|
||||||
|
|
||||||
|
const buildKeyMap = (fields: FieldsConfig, map?: KeyMap): KeyMap => {
|
||||||
|
if (!map) {
|
||||||
|
map = new Map();
|
||||||
|
}
|
||||||
|
|
||||||
|
for (const config of fields) {
|
||||||
|
if ('in' in config) {
|
||||||
|
if (config.key) {
|
||||||
|
map.set(config.key, {
|
||||||
|
in: config.in,
|
||||||
|
map: config.map,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
} else if (config.args) {
|
||||||
|
buildKeyMap(config.args, map);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return map;
|
||||||
|
};
|
||||||
|
|
||||||
|
interface Params {
|
||||||
|
body: unknown;
|
||||||
|
headers: Record<string, unknown>;
|
||||||
|
path: Record<string, unknown>;
|
||||||
|
query: Record<string, unknown>;
|
||||||
|
}
|
||||||
|
|
||||||
|
const stripEmptySlots = (params: Params) => {
|
||||||
|
for (const [slot, value] of Object.entries(params)) {
|
||||||
|
if (value && typeof value === 'object' && !Object.keys(value).length) {
|
||||||
|
delete params[slot as Slot];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const buildClientParams = (
|
||||||
|
args: ReadonlyArray<unknown>,
|
||||||
|
fields: FieldsConfig,
|
||||||
|
) => {
|
||||||
|
const params: Params = {
|
||||||
|
body: {},
|
||||||
|
headers: {},
|
||||||
|
path: {},
|
||||||
|
query: {},
|
||||||
|
};
|
||||||
|
|
||||||
|
const map = buildKeyMap(fields);
|
||||||
|
|
||||||
|
let config: FieldsConfig[number] | undefined;
|
||||||
|
|
||||||
|
for (const [index, arg] of args.entries()) {
|
||||||
|
if (fields[index]) {
|
||||||
|
config = fields[index];
|
||||||
|
}
|
||||||
|
|
||||||
|
if (!config) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if ('in' in config) {
|
||||||
|
if (config.key) {
|
||||||
|
const field = map.get(config.key)!;
|
||||||
|
const name = field.map || config.key;
|
||||||
|
(params[field.in] as Record<string, unknown>)[name] = arg;
|
||||||
|
} else {
|
||||||
|
params.body = arg;
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
for (const [key, value] of Object.entries(arg ?? {})) {
|
||||||
|
const field = map.get(key);
|
||||||
|
|
||||||
|
if (field) {
|
||||||
|
const name = field.map || key;
|
||||||
|
(params[field.in] as Record<string, unknown>)[name] = value;
|
||||||
|
} else {
|
||||||
|
const extra = extraPrefixes.find(([prefix]) =>
|
||||||
|
key.startsWith(prefix),
|
||||||
|
);
|
||||||
|
|
||||||
|
if (extra) {
|
||||||
|
const [prefix, slot] = extra;
|
||||||
|
(params[slot] as Record<string, unknown>)[
|
||||||
|
key.slice(prefix.length)
|
||||||
|
] = value;
|
||||||
|
} else {
|
||||||
|
for (const [slot, allowed] of Object.entries(
|
||||||
|
config.allowExtra ?? {},
|
||||||
|
)) {
|
||||||
|
if (allowed) {
|
||||||
|
(params[slot as Slot] as Record<string, unknown>)[key] = value;
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
stripEmptySlots(params);
|
||||||
|
|
||||||
|
return params;
|
||||||
|
};
|
||||||
181
frontend/src/lib/api/generated/core/pathSerializer.gen.ts
Normal file
181
frontend/src/lib/api/generated/core/pathSerializer.gen.ts
Normal file
@@ -0,0 +1,181 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
interface SerializeOptions<T>
|
||||||
|
extends SerializePrimitiveOptions,
|
||||||
|
SerializerOptions<T> {}
|
||||||
|
|
||||||
|
interface SerializePrimitiveOptions {
|
||||||
|
allowReserved?: boolean;
|
||||||
|
name: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export interface SerializerOptions<T> {
|
||||||
|
/**
|
||||||
|
* @default true
|
||||||
|
*/
|
||||||
|
explode: boolean;
|
||||||
|
style: T;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type ArrayStyle = 'form' | 'spaceDelimited' | 'pipeDelimited';
|
||||||
|
export type ArraySeparatorStyle = ArrayStyle | MatrixStyle;
|
||||||
|
type MatrixStyle = 'label' | 'matrix' | 'simple';
|
||||||
|
export type ObjectStyle = 'form' | 'deepObject';
|
||||||
|
type ObjectSeparatorStyle = ObjectStyle | MatrixStyle;
|
||||||
|
|
||||||
|
interface SerializePrimitiveParam extends SerializePrimitiveOptions {
|
||||||
|
value: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const separatorArrayExplode = (style: ArraySeparatorStyle) => {
|
||||||
|
switch (style) {
|
||||||
|
case 'label':
|
||||||
|
return '.';
|
||||||
|
case 'matrix':
|
||||||
|
return ';';
|
||||||
|
case 'simple':
|
||||||
|
return ',';
|
||||||
|
default:
|
||||||
|
return '&';
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const separatorArrayNoExplode = (style: ArraySeparatorStyle) => {
|
||||||
|
switch (style) {
|
||||||
|
case 'form':
|
||||||
|
return ',';
|
||||||
|
case 'pipeDelimited':
|
||||||
|
return '|';
|
||||||
|
case 'spaceDelimited':
|
||||||
|
return '%20';
|
||||||
|
default:
|
||||||
|
return ',';
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const separatorObjectExplode = (style: ObjectSeparatorStyle) => {
|
||||||
|
switch (style) {
|
||||||
|
case 'label':
|
||||||
|
return '.';
|
||||||
|
case 'matrix':
|
||||||
|
return ';';
|
||||||
|
case 'simple':
|
||||||
|
return ',';
|
||||||
|
default:
|
||||||
|
return '&';
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
export const serializeArrayParam = ({
|
||||||
|
allowReserved,
|
||||||
|
explode,
|
||||||
|
name,
|
||||||
|
style,
|
||||||
|
value,
|
||||||
|
}: SerializeOptions<ArraySeparatorStyle> & {
|
||||||
|
value: unknown[];
|
||||||
|
}) => {
|
||||||
|
if (!explode) {
|
||||||
|
const joinedValues = (
|
||||||
|
allowReserved ? value : value.map((v) => encodeURIComponent(v as string))
|
||||||
|
).join(separatorArrayNoExplode(style));
|
||||||
|
switch (style) {
|
||||||
|
case 'label':
|
||||||
|
return `.${joinedValues}`;
|
||||||
|
case 'matrix':
|
||||||
|
return `;${name}=${joinedValues}`;
|
||||||
|
case 'simple':
|
||||||
|
return joinedValues;
|
||||||
|
default:
|
||||||
|
return `${name}=${joinedValues}`;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const separator = separatorArrayExplode(style);
|
||||||
|
const joinedValues = value
|
||||||
|
.map((v) => {
|
||||||
|
if (style === 'label' || style === 'simple') {
|
||||||
|
return allowReserved ? v : encodeURIComponent(v as string);
|
||||||
|
}
|
||||||
|
|
||||||
|
return serializePrimitiveParam({
|
||||||
|
allowReserved,
|
||||||
|
name,
|
||||||
|
value: v as string,
|
||||||
|
});
|
||||||
|
})
|
||||||
|
.join(separator);
|
||||||
|
return style === 'label' || style === 'matrix'
|
||||||
|
? separator + joinedValues
|
||||||
|
: joinedValues;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const serializePrimitiveParam = ({
|
||||||
|
allowReserved,
|
||||||
|
name,
|
||||||
|
value,
|
||||||
|
}: SerializePrimitiveParam) => {
|
||||||
|
if (value === undefined || value === null) {
|
||||||
|
return '';
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof value === 'object') {
|
||||||
|
throw new Error(
|
||||||
|
'Deeply-nested arrays/objects aren’t supported. Provide your own `querySerializer()` to handle these.',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return `${name}=${allowReserved ? value : encodeURIComponent(value)}`;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const serializeObjectParam = ({
|
||||||
|
allowReserved,
|
||||||
|
explode,
|
||||||
|
name,
|
||||||
|
style,
|
||||||
|
value,
|
||||||
|
valueOnly,
|
||||||
|
}: SerializeOptions<ObjectSeparatorStyle> & {
|
||||||
|
value: Record<string, unknown> | Date;
|
||||||
|
valueOnly?: boolean;
|
||||||
|
}) => {
|
||||||
|
if (value instanceof Date) {
|
||||||
|
return valueOnly ? value.toISOString() : `${name}=${value.toISOString()}`;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (style !== 'deepObject' && !explode) {
|
||||||
|
let values: string[] = [];
|
||||||
|
Object.entries(value).forEach(([key, v]) => {
|
||||||
|
values = [
|
||||||
|
...values,
|
||||||
|
key,
|
||||||
|
allowReserved ? (v as string) : encodeURIComponent(v as string),
|
||||||
|
];
|
||||||
|
});
|
||||||
|
const joinedValues = values.join(',');
|
||||||
|
switch (style) {
|
||||||
|
case 'form':
|
||||||
|
return `${name}=${joinedValues}`;
|
||||||
|
case 'label':
|
||||||
|
return `.${joinedValues}`;
|
||||||
|
case 'matrix':
|
||||||
|
return `;${name}=${joinedValues}`;
|
||||||
|
default:
|
||||||
|
return joinedValues;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
const separator = separatorObjectExplode(style);
|
||||||
|
const joinedValues = Object.entries(value)
|
||||||
|
.map(([key, v]) =>
|
||||||
|
serializePrimitiveParam({
|
||||||
|
allowReserved,
|
||||||
|
name: style === 'deepObject' ? `${name}[${key}]` : key,
|
||||||
|
value: v as string,
|
||||||
|
}),
|
||||||
|
)
|
||||||
|
.join(separator);
|
||||||
|
return style === 'label' || style === 'matrix'
|
||||||
|
? separator + joinedValues
|
||||||
|
: joinedValues;
|
||||||
|
};
|
||||||
136
frontend/src/lib/api/generated/core/queryKeySerializer.gen.ts
Normal file
136
frontend/src/lib/api/generated/core/queryKeySerializer.gen.ts
Normal file
@@ -0,0 +1,136 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
/**
|
||||||
|
* JSON-friendly union that mirrors what Pinia Colada can hash.
|
||||||
|
*/
|
||||||
|
export type JsonValue =
|
||||||
|
| null
|
||||||
|
| string
|
||||||
|
| number
|
||||||
|
| boolean
|
||||||
|
| JsonValue[]
|
||||||
|
| { [key: string]: JsonValue };
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Replacer that converts non-JSON values (bigint, Date, etc.) to safe substitutes.
|
||||||
|
*/
|
||||||
|
export const queryKeyJsonReplacer = (_key: string, value: unknown) => {
|
||||||
|
if (
|
||||||
|
value === undefined ||
|
||||||
|
typeof value === 'function' ||
|
||||||
|
typeof value === 'symbol'
|
||||||
|
) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
if (typeof value === 'bigint') {
|
||||||
|
return value.toString();
|
||||||
|
}
|
||||||
|
if (value instanceof Date) {
|
||||||
|
return value.toISOString();
|
||||||
|
}
|
||||||
|
return value;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Safely stringifies a value and parses it back into a JsonValue.
|
||||||
|
*/
|
||||||
|
export const stringifyToJsonValue = (input: unknown): JsonValue | undefined => {
|
||||||
|
try {
|
||||||
|
const json = JSON.stringify(input, queryKeyJsonReplacer);
|
||||||
|
if (json === undefined) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
return JSON.parse(json) as JsonValue;
|
||||||
|
} catch {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Detects plain objects (including objects with a null prototype).
|
||||||
|
*/
|
||||||
|
const isPlainObject = (value: unknown): value is Record<string, unknown> => {
|
||||||
|
if (value === null || typeof value !== 'object') {
|
||||||
|
return false;
|
||||||
|
}
|
||||||
|
const prototype = Object.getPrototypeOf(value as object);
|
||||||
|
return prototype === Object.prototype || prototype === null;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Turns URLSearchParams into a sorted JSON object for deterministic keys.
|
||||||
|
*/
|
||||||
|
const serializeSearchParams = (params: URLSearchParams): JsonValue => {
|
||||||
|
const entries = Array.from(params.entries()).sort(([a], [b]) =>
|
||||||
|
a.localeCompare(b),
|
||||||
|
);
|
||||||
|
const result: Record<string, JsonValue> = {};
|
||||||
|
|
||||||
|
for (const [key, value] of entries) {
|
||||||
|
const existing = result[key];
|
||||||
|
if (existing === undefined) {
|
||||||
|
result[key] = value;
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Array.isArray(existing)) {
|
||||||
|
(existing as string[]).push(value);
|
||||||
|
} else {
|
||||||
|
result[key] = [existing, value];
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
return result;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Normalizes any accepted value into a JSON-friendly shape for query keys.
|
||||||
|
*/
|
||||||
|
export const serializeQueryKeyValue = (
|
||||||
|
value: unknown,
|
||||||
|
): JsonValue | undefined => {
|
||||||
|
if (value === null) {
|
||||||
|
return null;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
typeof value === 'string' ||
|
||||||
|
typeof value === 'number' ||
|
||||||
|
typeof value === 'boolean'
|
||||||
|
) {
|
||||||
|
return value;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
value === undefined ||
|
||||||
|
typeof value === 'function' ||
|
||||||
|
typeof value === 'symbol'
|
||||||
|
) {
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof value === 'bigint') {
|
||||||
|
return value.toString();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (value instanceof Date) {
|
||||||
|
return value.toISOString();
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Array.isArray(value)) {
|
||||||
|
return stringifyToJsonValue(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (
|
||||||
|
typeof URLSearchParams !== 'undefined' &&
|
||||||
|
value instanceof URLSearchParams
|
||||||
|
) {
|
||||||
|
return serializeSearchParams(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (isPlainObject(value)) {
|
||||||
|
return stringifyToJsonValue(value);
|
||||||
|
}
|
||||||
|
|
||||||
|
return undefined;
|
||||||
|
};
|
||||||
264
frontend/src/lib/api/generated/core/serverSentEvents.gen.ts
Normal file
264
frontend/src/lib/api/generated/core/serverSentEvents.gen.ts
Normal file
@@ -0,0 +1,264 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
import type { Config } from './types.gen';
|
||||||
|
|
||||||
|
export type ServerSentEventsOptions<TData = unknown> = Omit<
|
||||||
|
RequestInit,
|
||||||
|
'method'
|
||||||
|
> &
|
||||||
|
Pick<Config, 'method' | 'responseTransformer' | 'responseValidator'> & {
|
||||||
|
/**
|
||||||
|
* Fetch API implementation. You can use this option to provide a custom
|
||||||
|
* fetch instance.
|
||||||
|
*
|
||||||
|
* @default globalThis.fetch
|
||||||
|
*/
|
||||||
|
fetch?: typeof fetch;
|
||||||
|
/**
|
||||||
|
* Implementing clients can call request interceptors inside this hook.
|
||||||
|
*/
|
||||||
|
onRequest?: (url: string, init: RequestInit) => Promise<Request>;
|
||||||
|
/**
|
||||||
|
* Callback invoked when a network or parsing error occurs during streaming.
|
||||||
|
*
|
||||||
|
* This option applies only if the endpoint returns a stream of events.
|
||||||
|
*
|
||||||
|
* @param error The error that occurred.
|
||||||
|
*/
|
||||||
|
onSseError?: (error: unknown) => void;
|
||||||
|
/**
|
||||||
|
* Callback invoked when an event is streamed from the server.
|
||||||
|
*
|
||||||
|
* This option applies only if the endpoint returns a stream of events.
|
||||||
|
*
|
||||||
|
* @param event Event streamed from the server.
|
||||||
|
* @returns Nothing (void).
|
||||||
|
*/
|
||||||
|
onSseEvent?: (event: StreamEvent<TData>) => void;
|
||||||
|
serializedBody?: RequestInit['body'];
|
||||||
|
/**
|
||||||
|
* Default retry delay in milliseconds.
|
||||||
|
*
|
||||||
|
* This option applies only if the endpoint returns a stream of events.
|
||||||
|
*
|
||||||
|
* @default 3000
|
||||||
|
*/
|
||||||
|
sseDefaultRetryDelay?: number;
|
||||||
|
/**
|
||||||
|
* Maximum number of retry attempts before giving up.
|
||||||
|
*/
|
||||||
|
sseMaxRetryAttempts?: number;
|
||||||
|
/**
|
||||||
|
* Maximum retry delay in milliseconds.
|
||||||
|
*
|
||||||
|
* Applies only when exponential backoff is used.
|
||||||
|
*
|
||||||
|
* This option applies only if the endpoint returns a stream of events.
|
||||||
|
*
|
||||||
|
* @default 30000
|
||||||
|
*/
|
||||||
|
sseMaxRetryDelay?: number;
|
||||||
|
/**
|
||||||
|
* Optional sleep function for retry backoff.
|
||||||
|
*
|
||||||
|
* Defaults to using `setTimeout`.
|
||||||
|
*/
|
||||||
|
sseSleepFn?: (ms: number) => Promise<void>;
|
||||||
|
url: string;
|
||||||
|
};
|
||||||
|
|
||||||
|
export interface StreamEvent<TData = unknown> {
|
||||||
|
data: TData;
|
||||||
|
event?: string;
|
||||||
|
id?: string;
|
||||||
|
retry?: number;
|
||||||
|
}
|
||||||
|
|
||||||
|
export type ServerSentEventsResult<
|
||||||
|
TData = unknown,
|
||||||
|
TReturn = void,
|
||||||
|
TNext = unknown,
|
||||||
|
> = {
|
||||||
|
stream: AsyncGenerator<
|
||||||
|
TData extends Record<string, unknown> ? TData[keyof TData] : TData,
|
||||||
|
TReturn,
|
||||||
|
TNext
|
||||||
|
>;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const createSseClient = <TData = unknown>({
|
||||||
|
onRequest,
|
||||||
|
onSseError,
|
||||||
|
onSseEvent,
|
||||||
|
responseTransformer,
|
||||||
|
responseValidator,
|
||||||
|
sseDefaultRetryDelay,
|
||||||
|
sseMaxRetryAttempts,
|
||||||
|
sseMaxRetryDelay,
|
||||||
|
sseSleepFn,
|
||||||
|
url,
|
||||||
|
...options
|
||||||
|
}: ServerSentEventsOptions): ServerSentEventsResult<TData> => {
|
||||||
|
let lastEventId: string | undefined;
|
||||||
|
|
||||||
|
const sleep =
|
||||||
|
sseSleepFn ??
|
||||||
|
((ms: number) => new Promise((resolve) => setTimeout(resolve, ms)));
|
||||||
|
|
||||||
|
const createStream = async function* () {
|
||||||
|
let retryDelay: number = sseDefaultRetryDelay ?? 3000;
|
||||||
|
let attempt = 0;
|
||||||
|
const signal = options.signal ?? new AbortController().signal;
|
||||||
|
|
||||||
|
while (true) {
|
||||||
|
if (signal.aborted) break;
|
||||||
|
|
||||||
|
attempt++;
|
||||||
|
|
||||||
|
const headers =
|
||||||
|
options.headers instanceof Headers
|
||||||
|
? options.headers
|
||||||
|
: new Headers(options.headers as Record<string, string> | undefined);
|
||||||
|
|
||||||
|
if (lastEventId !== undefined) {
|
||||||
|
headers.set('Last-Event-ID', lastEventId);
|
||||||
|
}
|
||||||
|
|
||||||
|
try {
|
||||||
|
const requestInit: RequestInit = {
|
||||||
|
redirect: 'follow',
|
||||||
|
...options,
|
||||||
|
body: options.serializedBody,
|
||||||
|
headers,
|
||||||
|
signal,
|
||||||
|
};
|
||||||
|
let request = new Request(url, requestInit);
|
||||||
|
if (onRequest) {
|
||||||
|
request = await onRequest(url, requestInit);
|
||||||
|
}
|
||||||
|
// fetch must be assigned here, otherwise it would throw the error:
|
||||||
|
// TypeError: Failed to execute 'fetch' on 'Window': Illegal invocation
|
||||||
|
const _fetch = options.fetch ?? globalThis.fetch;
|
||||||
|
const response = await _fetch(request);
|
||||||
|
|
||||||
|
if (!response.ok)
|
||||||
|
throw new Error(
|
||||||
|
`SSE failed: ${response.status} ${response.statusText}`,
|
||||||
|
);
|
||||||
|
|
||||||
|
if (!response.body) throw new Error('No body in SSE response');
|
||||||
|
|
||||||
|
const reader = response.body
|
||||||
|
.pipeThrough(new TextDecoderStream())
|
||||||
|
.getReader();
|
||||||
|
|
||||||
|
let buffer = '';
|
||||||
|
|
||||||
|
const abortHandler = () => {
|
||||||
|
try {
|
||||||
|
reader.cancel();
|
||||||
|
} catch {
|
||||||
|
// noop
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
signal.addEventListener('abort', abortHandler);
|
||||||
|
|
||||||
|
try {
|
||||||
|
while (true) {
|
||||||
|
const { done, value } = await reader.read();
|
||||||
|
if (done) break;
|
||||||
|
buffer += value;
|
||||||
|
|
||||||
|
const chunks = buffer.split('\n\n');
|
||||||
|
buffer = chunks.pop() ?? '';
|
||||||
|
|
||||||
|
for (const chunk of chunks) {
|
||||||
|
const lines = chunk.split('\n');
|
||||||
|
const dataLines: Array<string> = [];
|
||||||
|
let eventName: string | undefined;
|
||||||
|
|
||||||
|
for (const line of lines) {
|
||||||
|
if (line.startsWith('data:')) {
|
||||||
|
dataLines.push(line.replace(/^data:\s*/, ''));
|
||||||
|
} else if (line.startsWith('event:')) {
|
||||||
|
eventName = line.replace(/^event:\s*/, '');
|
||||||
|
} else if (line.startsWith('id:')) {
|
||||||
|
lastEventId = line.replace(/^id:\s*/, '');
|
||||||
|
} else if (line.startsWith('retry:')) {
|
||||||
|
const parsed = Number.parseInt(
|
||||||
|
line.replace(/^retry:\s*/, ''),
|
||||||
|
10,
|
||||||
|
);
|
||||||
|
if (!Number.isNaN(parsed)) {
|
||||||
|
retryDelay = parsed;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
let data: unknown;
|
||||||
|
let parsedJson = false;
|
||||||
|
|
||||||
|
if (dataLines.length) {
|
||||||
|
const rawData = dataLines.join('\n');
|
||||||
|
try {
|
||||||
|
data = JSON.parse(rawData);
|
||||||
|
parsedJson = true;
|
||||||
|
} catch {
|
||||||
|
data = rawData;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
if (parsedJson) {
|
||||||
|
if (responseValidator) {
|
||||||
|
await responseValidator(data);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (responseTransformer) {
|
||||||
|
data = await responseTransformer(data);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
onSseEvent?.({
|
||||||
|
data,
|
||||||
|
event: eventName,
|
||||||
|
id: lastEventId,
|
||||||
|
retry: retryDelay,
|
||||||
|
});
|
||||||
|
|
||||||
|
if (dataLines.length) {
|
||||||
|
yield data as any;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
} finally {
|
||||||
|
signal.removeEventListener('abort', abortHandler);
|
||||||
|
reader.releaseLock();
|
||||||
|
}
|
||||||
|
|
||||||
|
break; // exit loop on normal completion
|
||||||
|
} catch (error) {
|
||||||
|
// connection failed or aborted; retry after delay
|
||||||
|
onSseError?.(error);
|
||||||
|
|
||||||
|
if (
|
||||||
|
sseMaxRetryAttempts !== undefined &&
|
||||||
|
attempt >= sseMaxRetryAttempts
|
||||||
|
) {
|
||||||
|
break; // stop after firing error
|
||||||
|
}
|
||||||
|
|
||||||
|
// exponential backoff: double retry each attempt, cap at 30s
|
||||||
|
const backoff = Math.min(
|
||||||
|
retryDelay * 2 ** (attempt - 1),
|
||||||
|
sseMaxRetryDelay ?? 30000,
|
||||||
|
);
|
||||||
|
await sleep(backoff);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
const stream = createStream();
|
||||||
|
|
||||||
|
return { stream };
|
||||||
|
};
|
||||||
118
frontend/src/lib/api/generated/core/types.gen.ts
Normal file
118
frontend/src/lib/api/generated/core/types.gen.ts
Normal file
@@ -0,0 +1,118 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
import type { Auth, AuthToken } from './auth.gen';
|
||||||
|
import type {
|
||||||
|
BodySerializer,
|
||||||
|
QuerySerializer,
|
||||||
|
QuerySerializerOptions,
|
||||||
|
} from './bodySerializer.gen';
|
||||||
|
|
||||||
|
export type HttpMethod =
|
||||||
|
| 'connect'
|
||||||
|
| 'delete'
|
||||||
|
| 'get'
|
||||||
|
| 'head'
|
||||||
|
| 'options'
|
||||||
|
| 'patch'
|
||||||
|
| 'post'
|
||||||
|
| 'put'
|
||||||
|
| 'trace';
|
||||||
|
|
||||||
|
export type Client<
|
||||||
|
RequestFn = never,
|
||||||
|
Config = unknown,
|
||||||
|
MethodFn = never,
|
||||||
|
BuildUrlFn = never,
|
||||||
|
SseFn = never,
|
||||||
|
> = {
|
||||||
|
/**
|
||||||
|
* Returns the final request URL.
|
||||||
|
*/
|
||||||
|
buildUrl: BuildUrlFn;
|
||||||
|
getConfig: () => Config;
|
||||||
|
request: RequestFn;
|
||||||
|
setConfig: (config: Config) => Config;
|
||||||
|
} & {
|
||||||
|
[K in HttpMethod]: MethodFn;
|
||||||
|
} & ([SseFn] extends [never]
|
||||||
|
? { sse?: never }
|
||||||
|
: { sse: { [K in HttpMethod]: SseFn } });
|
||||||
|
|
||||||
|
export interface Config {
|
||||||
|
/**
|
||||||
|
* Auth token or a function returning auth token. The resolved value will be
|
||||||
|
* added to the request payload as defined by its `security` array.
|
||||||
|
*/
|
||||||
|
auth?: ((auth: Auth) => Promise<AuthToken> | AuthToken) | AuthToken;
|
||||||
|
/**
|
||||||
|
* A function for serializing request body parameter. By default,
|
||||||
|
* {@link JSON.stringify()} will be used.
|
||||||
|
*/
|
||||||
|
bodySerializer?: BodySerializer | null;
|
||||||
|
/**
|
||||||
|
* An object containing any HTTP headers that you want to pre-populate your
|
||||||
|
* `Headers` object with.
|
||||||
|
*
|
||||||
|
* {@link https://developer.mozilla.org/docs/Web/API/Headers/Headers#init See more}
|
||||||
|
*/
|
||||||
|
headers?:
|
||||||
|
| RequestInit['headers']
|
||||||
|
| Record<
|
||||||
|
string,
|
||||||
|
| string
|
||||||
|
| number
|
||||||
|
| boolean
|
||||||
|
| (string | number | boolean)[]
|
||||||
|
| null
|
||||||
|
| undefined
|
||||||
|
| unknown
|
||||||
|
>;
|
||||||
|
/**
|
||||||
|
* The request method.
|
||||||
|
*
|
||||||
|
* {@link https://developer.mozilla.org/docs/Web/API/fetch#method See more}
|
||||||
|
*/
|
||||||
|
method?: Uppercase<HttpMethod>;
|
||||||
|
/**
|
||||||
|
* A function for serializing request query parameters. By default, arrays
|
||||||
|
* will be exploded in form style, objects will be exploded in deepObject
|
||||||
|
* style, and reserved characters are percent-encoded.
|
||||||
|
*
|
||||||
|
* This method will have no effect if the native `paramsSerializer()` Axios
|
||||||
|
* API function is used.
|
||||||
|
*
|
||||||
|
* {@link https://swagger.io/docs/specification/serialization/#query View examples}
|
||||||
|
*/
|
||||||
|
querySerializer?: QuerySerializer | QuerySerializerOptions;
|
||||||
|
/**
|
||||||
|
* A function validating request data. This is useful if you want to ensure
|
||||||
|
* the request conforms to the desired shape, so it can be safely sent to
|
||||||
|
* the server.
|
||||||
|
*/
|
||||||
|
requestValidator?: (data: unknown) => Promise<unknown>;
|
||||||
|
/**
|
||||||
|
* A function transforming response data before it's returned. This is useful
|
||||||
|
* for post-processing data, e.g. converting ISO strings into Date objects.
|
||||||
|
*/
|
||||||
|
responseTransformer?: (data: unknown) => Promise<unknown>;
|
||||||
|
/**
|
||||||
|
* A function validating response data. This is useful if you want to ensure
|
||||||
|
* the response conforms to the desired shape, so it can be safely passed to
|
||||||
|
* the transformers and returned to the user.
|
||||||
|
*/
|
||||||
|
responseValidator?: (data: unknown) => Promise<unknown>;
|
||||||
|
}
|
||||||
|
|
||||||
|
type IsExactlyNeverOrNeverUndefined<T> = [T] extends [never]
|
||||||
|
? true
|
||||||
|
: [T] extends [never | undefined]
|
||||||
|
? [undefined] extends [T]
|
||||||
|
? false
|
||||||
|
: true
|
||||||
|
: false;
|
||||||
|
|
||||||
|
export type OmitNever<T extends Record<string, unknown>> = {
|
||||||
|
[K in keyof T as IsExactlyNeverOrNeverUndefined<T[K]> extends true
|
||||||
|
? never
|
||||||
|
: K]: T[K];
|
||||||
|
};
|
||||||
143
frontend/src/lib/api/generated/core/utils.gen.ts
Normal file
143
frontend/src/lib/api/generated/core/utils.gen.ts
Normal file
@@ -0,0 +1,143 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
import type { BodySerializer, QuerySerializer } from './bodySerializer.gen';
|
||||||
|
import {
|
||||||
|
type ArraySeparatorStyle,
|
||||||
|
serializeArrayParam,
|
||||||
|
serializeObjectParam,
|
||||||
|
serializePrimitiveParam,
|
||||||
|
} from './pathSerializer.gen';
|
||||||
|
|
||||||
|
export interface PathSerializer {
|
||||||
|
path: Record<string, unknown>;
|
||||||
|
url: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const PATH_PARAM_RE = /\{[^{}]+\}/g;
|
||||||
|
|
||||||
|
export const defaultPathSerializer = ({ path, url: _url }: PathSerializer) => {
|
||||||
|
let url = _url;
|
||||||
|
const matches = _url.match(PATH_PARAM_RE);
|
||||||
|
if (matches) {
|
||||||
|
for (const match of matches) {
|
||||||
|
let explode = false;
|
||||||
|
let name = match.substring(1, match.length - 1);
|
||||||
|
let style: ArraySeparatorStyle = 'simple';
|
||||||
|
|
||||||
|
if (name.endsWith('*')) {
|
||||||
|
explode = true;
|
||||||
|
name = name.substring(0, name.length - 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
if (name.startsWith('.')) {
|
||||||
|
name = name.substring(1);
|
||||||
|
style = 'label';
|
||||||
|
} else if (name.startsWith(';')) {
|
||||||
|
name = name.substring(1);
|
||||||
|
style = 'matrix';
|
||||||
|
}
|
||||||
|
|
||||||
|
const value = path[name];
|
||||||
|
|
||||||
|
if (value === undefined || value === null) {
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (Array.isArray(value)) {
|
||||||
|
url = url.replace(
|
||||||
|
match,
|
||||||
|
serializeArrayParam({ explode, name, style, value }),
|
||||||
|
);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (typeof value === 'object') {
|
||||||
|
url = url.replace(
|
||||||
|
match,
|
||||||
|
serializeObjectParam({
|
||||||
|
explode,
|
||||||
|
name,
|
||||||
|
style,
|
||||||
|
value: value as Record<string, unknown>,
|
||||||
|
valueOnly: true,
|
||||||
|
}),
|
||||||
|
);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
if (style === 'matrix') {
|
||||||
|
url = url.replace(
|
||||||
|
match,
|
||||||
|
`;${serializePrimitiveParam({
|
||||||
|
name,
|
||||||
|
value: value as string,
|
||||||
|
})}`,
|
||||||
|
);
|
||||||
|
continue;
|
||||||
|
}
|
||||||
|
|
||||||
|
const replaceValue = encodeURIComponent(
|
||||||
|
style === 'label' ? `.${value as string}` : (value as string),
|
||||||
|
);
|
||||||
|
url = url.replace(match, replaceValue);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return url;
|
||||||
|
};
|
||||||
|
|
||||||
|
export const getUrl = ({
|
||||||
|
baseUrl,
|
||||||
|
path,
|
||||||
|
query,
|
||||||
|
querySerializer,
|
||||||
|
url: _url,
|
||||||
|
}: {
|
||||||
|
baseUrl?: string;
|
||||||
|
path?: Record<string, unknown>;
|
||||||
|
query?: Record<string, unknown>;
|
||||||
|
querySerializer: QuerySerializer;
|
||||||
|
url: string;
|
||||||
|
}) => {
|
||||||
|
const pathUrl = _url.startsWith('/') ? _url : `/${_url}`;
|
||||||
|
let url = (baseUrl ?? '') + pathUrl;
|
||||||
|
if (path) {
|
||||||
|
url = defaultPathSerializer({ path, url });
|
||||||
|
}
|
||||||
|
let search = query ? querySerializer(query) : '';
|
||||||
|
if (search.startsWith('?')) {
|
||||||
|
search = search.substring(1);
|
||||||
|
}
|
||||||
|
if (search) {
|
||||||
|
url += `?${search}`;
|
||||||
|
}
|
||||||
|
return url;
|
||||||
|
};
|
||||||
|
|
||||||
|
export function getValidRequestBody(options: {
|
||||||
|
body?: unknown;
|
||||||
|
bodySerializer?: BodySerializer | null;
|
||||||
|
serializedBody?: unknown;
|
||||||
|
}) {
|
||||||
|
const hasBody = options.body !== undefined;
|
||||||
|
const isSerializedBody = hasBody && options.bodySerializer;
|
||||||
|
|
||||||
|
if (isSerializedBody) {
|
||||||
|
if ('serializedBody' in options) {
|
||||||
|
const hasSerializedBody =
|
||||||
|
options.serializedBody !== undefined && options.serializedBody !== '';
|
||||||
|
|
||||||
|
return hasSerializedBody ? options.serializedBody : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// not all clients implement a serializedBody property (i.e. client-axios)
|
||||||
|
return options.body !== '' ? options.body : null;
|
||||||
|
}
|
||||||
|
|
||||||
|
// plain/text body
|
||||||
|
if (hasBody) {
|
||||||
|
return options.body;
|
||||||
|
}
|
||||||
|
|
||||||
|
// no body was provided
|
||||||
|
return undefined;
|
||||||
|
}
|
||||||
10
frontend/src/lib/api/generated/index.ts
Executable file → Normal file
10
frontend/src/lib/api/generated/index.ts
Executable file → Normal file
@@ -1,8 +1,4 @@
|
|||||||
// This file will be auto-generated by running: npm run generate:api
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
// Make sure the backend is running before generating the API client
|
|
||||||
//
|
|
||||||
// To generate: npm run generate:api
|
|
||||||
//
|
|
||||||
// This placeholder prevents import errors before generation
|
|
||||||
|
|
||||||
export {}
|
export type * from './types.gen';
|
||||||
|
export * from './sdk.gen';
|
||||||
|
|||||||
913
frontend/src/lib/api/generated/sdk.gen.ts
Normal file
913
frontend/src/lib/api/generated/sdk.gen.ts
Normal file
@@ -0,0 +1,913 @@
|
|||||||
|
// This file is auto-generated by @hey-api/openapi-ts
|
||||||
|
|
||||||
|
import { type Client, type Options as Options2, type TDataShape, urlSearchParamsBodySerializer } from './client';
|
||||||
|
import { client } from './client.gen';
|
||||||
|
import type { AdminActivateUserData, AdminActivateUserErrors, AdminActivateUserResponses, AdminAddOrganizationMemberData, AdminAddOrganizationMemberErrors, AdminAddOrganizationMemberResponses, AdminBulkUserActionData, AdminBulkUserActionErrors, AdminBulkUserActionResponses, AdminCreateOrganizationData, AdminCreateOrganizationErrors, AdminCreateOrganizationResponses, AdminCreateUserData, AdminCreateUserErrors, AdminCreateUserResponses, AdminDeactivateUserData, AdminDeactivateUserErrors, AdminDeactivateUserResponses, AdminDeleteOrganizationData, AdminDeleteOrganizationErrors, AdminDeleteOrganizationResponses, AdminDeleteUserData, AdminDeleteUserErrors, AdminDeleteUserResponses, AdminGetOrganizationData, AdminGetOrganizationErrors, AdminGetOrganizationResponses, AdminGetUserData, AdminGetUserErrors, AdminGetUserResponses, AdminListOrganizationMembersData, AdminListOrganizationMembersErrors, AdminListOrganizationMembersResponses, AdminListOrganizationsData, AdminListOrganizationsErrors, AdminListOrganizationsResponses, AdminListUsersData, AdminListUsersErrors, AdminListUsersResponses, AdminRemoveOrganizationMemberData, AdminRemoveOrganizationMemberErrors, AdminRemoveOrganizationMemberResponses, AdminUpdateOrganizationData, AdminUpdateOrganizationErrors, AdminUpdateOrganizationResponses, AdminUpdateUserData, AdminUpdateUserErrors, AdminUpdateUserResponses, ChangeCurrentUserPasswordData, ChangeCurrentUserPasswordErrors, ChangeCurrentUserPasswordResponses, CleanupExpiredSessionsData, CleanupExpiredSessionsResponses, ConfirmPasswordResetData, ConfirmPasswordResetErrors, ConfirmPasswordResetResponses, DeleteUserData, DeleteUserErrors, DeleteUserResponses, GetCurrentUserInfoData, GetCurrentUserInfoResponses, GetCurrentUserProfileData, GetCurrentUserProfileResponses, GetMyOrganizationsData, GetMyOrganizationsErrors, GetMyOrganizationsResponses, GetOrganizationData, GetOrganizationErrors, GetOrganizationMembersData, GetOrganizationMembersErrors, GetOrganizationMembersResponses, GetOrganizationResponses, GetUserByIdData, GetUserByIdErrors, GetUserByIdResponses, HealthCheckData, HealthCheckResponses, ListMySessionsData, ListMySessionsResponses, ListUsersData, ListUsersErrors, ListUsersResponses, LoginData, LoginErrors, LoginOauthData, LoginOauthErrors, LoginOauthResponses, LoginResponses, LogoutAllData, LogoutAllResponses, LogoutData, LogoutErrors, LogoutResponses, RefreshTokenData, RefreshTokenErrors, RefreshTokenResponses, RegisterData, RegisterErrors, RegisterResponses, RequestPasswordResetData, RequestPasswordResetErrors, RequestPasswordResetResponses, RevokeSessionData, RevokeSessionErrors, RevokeSessionResponses, RootGetData, RootGetResponses, UpdateCurrentUserData, UpdateCurrentUserErrors, UpdateCurrentUserResponses, UpdateOrganizationData, UpdateOrganizationErrors, UpdateOrganizationResponses, UpdateUserData, UpdateUserErrors, UpdateUserResponses } from './types.gen';
|
||||||
|
|
||||||
|
export type Options<TData extends TDataShape = TDataShape, ThrowOnError extends boolean = boolean> = Options2<TData, ThrowOnError> & {
|
||||||
|
/**
|
||||||
|
* You can provide a client instance returned by `createClient()` instead of
|
||||||
|
* individual options. This might be also useful if you want to implement a
|
||||||
|
* custom client.
|
||||||
|
*/
|
||||||
|
client?: Client;
|
||||||
|
/**
|
||||||
|
* You can pass arbitrary values through the `meta` object. This can be
|
||||||
|
* used to access values that aren't defined as part of the SDK function.
|
||||||
|
*/
|
||||||
|
meta?: Record<string, unknown>;
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Root
|
||||||
|
*/
|
||||||
|
export const rootGet = <ThrowOnError extends boolean = false>(options?: Options<RootGetData, ThrowOnError>) => {
|
||||||
|
return (options?.client ?? client).get<RootGetResponses, unknown, ThrowOnError>({
|
||||||
|
responseType: 'text',
|
||||||
|
url: '/',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Health Check
|
||||||
|
*
|
||||||
|
* Check the health status of the API and its dependencies
|
||||||
|
*/
|
||||||
|
export const healthCheck = <ThrowOnError extends boolean = false>(options?: Options<HealthCheckData, ThrowOnError>) => {
|
||||||
|
return (options?.client ?? client).get<HealthCheckResponses, unknown, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
url: '/health',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Register User
|
||||||
|
*
|
||||||
|
* Register a new user.
|
||||||
|
*
|
||||||
|
* Returns:
|
||||||
|
* The created user information.
|
||||||
|
*/
|
||||||
|
export const register = <ThrowOnError extends boolean = false>(options: Options<RegisterData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).post<RegisterResponses, RegisterErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
url: '/api/v1/auth/register',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Login
|
||||||
|
*
|
||||||
|
* Login with username and password.
|
||||||
|
*
|
||||||
|
* Creates a new session for this device.
|
||||||
|
*
|
||||||
|
* Returns:
|
||||||
|
* Access and refresh tokens.
|
||||||
|
*/
|
||||||
|
export const login = <ThrowOnError extends boolean = false>(options: Options<LoginData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).post<LoginResponses, LoginErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
url: '/api/v1/auth/login',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Login Oauth
|
||||||
|
*
|
||||||
|
* OAuth2-compatible login endpoint, used by the OpenAPI UI.
|
||||||
|
*
|
||||||
|
* Creates a new session for this device.
|
||||||
|
*
|
||||||
|
* Returns:
|
||||||
|
* Access and refresh tokens.
|
||||||
|
*/
|
||||||
|
export const loginOauth = <ThrowOnError extends boolean = false>(options: Options<LoginOauthData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).post<LoginOauthResponses, LoginOauthErrors, ThrowOnError>({
|
||||||
|
...urlSearchParamsBodySerializer,
|
||||||
|
responseType: 'json',
|
||||||
|
url: '/api/v1/auth/login/oauth',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/x-www-form-urlencoded',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Refresh Token
|
||||||
|
*
|
||||||
|
* Refresh access token using a refresh token.
|
||||||
|
*
|
||||||
|
* Validates that the session is still active before issuing new tokens.
|
||||||
|
*
|
||||||
|
* Returns:
|
||||||
|
* New access and refresh tokens.
|
||||||
|
*/
|
||||||
|
export const refreshToken = <ThrowOnError extends boolean = false>(options: Options<RefreshTokenData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).post<RefreshTokenResponses, RefreshTokenErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
url: '/api/v1/auth/refresh',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get Current User Info
|
||||||
|
*
|
||||||
|
* Get current user information.
|
||||||
|
*
|
||||||
|
* Requires authentication.
|
||||||
|
*/
|
||||||
|
export const getCurrentUserInfo = <ThrowOnError extends boolean = false>(options?: Options<GetCurrentUserInfoData, ThrowOnError>) => {
|
||||||
|
return (options?.client ?? client).get<GetCurrentUserInfoResponses, unknown, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/auth/me',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Request Password Reset
|
||||||
|
*
|
||||||
|
* Request a password reset link.
|
||||||
|
*
|
||||||
|
* An email will be sent with a reset link if the email exists.
|
||||||
|
* Always returns success to prevent email enumeration.
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 3 requests/minute
|
||||||
|
*/
|
||||||
|
export const requestPasswordReset = <ThrowOnError extends boolean = false>(options: Options<RequestPasswordResetData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).post<RequestPasswordResetResponses, RequestPasswordResetErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
url: '/api/v1/auth/password-reset/request',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Confirm Password Reset
|
||||||
|
*
|
||||||
|
* Reset password using a token from email.
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 5 requests/minute
|
||||||
|
*/
|
||||||
|
export const confirmPasswordReset = <ThrowOnError extends boolean = false>(options: Options<ConfirmPasswordResetData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).post<ConfirmPasswordResetResponses, ConfirmPasswordResetErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
url: '/api/v1/auth/password-reset/confirm',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logout from Current Device
|
||||||
|
*
|
||||||
|
* Logout from the current device only.
|
||||||
|
*
|
||||||
|
* Other devices will remain logged in.
|
||||||
|
*
|
||||||
|
* Requires the refresh token to identify which session to terminate.
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 10 requests/minute
|
||||||
|
*/
|
||||||
|
export const logout = <ThrowOnError extends boolean = false>(options: Options<LogoutData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).post<LogoutResponses, LogoutErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/auth/logout',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Logout from All Devices
|
||||||
|
*
|
||||||
|
* Logout from ALL devices.
|
||||||
|
*
|
||||||
|
* This will terminate all active sessions for the current user.
|
||||||
|
* You will need to log in again on all devices.
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 5 requests/minute
|
||||||
|
*/
|
||||||
|
export const logoutAll = <ThrowOnError extends boolean = false>(options?: Options<LogoutAllData, ThrowOnError>) => {
|
||||||
|
return (options?.client ?? client).post<LogoutAllResponses, unknown, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/auth/logout-all',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* List Users
|
||||||
|
*
|
||||||
|
* List all users with pagination, filtering, and sorting (admin only).
|
||||||
|
*
|
||||||
|
* **Authentication**: Required (Bearer token)
|
||||||
|
* **Authorization**: Superuser only
|
||||||
|
*
|
||||||
|
* **Filtering**: is_active, is_superuser
|
||||||
|
* **Sorting**: Any user field (email, first_name, last_name, created_at, etc.)
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 60 requests/minute
|
||||||
|
*/
|
||||||
|
export const listUsers = <ThrowOnError extends boolean = false>(options?: Options<ListUsersData, ThrowOnError>) => {
|
||||||
|
return (options?.client ?? client).get<ListUsersResponses, ListUsersErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/users',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get Current User
|
||||||
|
*
|
||||||
|
* Get the current authenticated user's profile.
|
||||||
|
*
|
||||||
|
* **Authentication**: Required (Bearer token)
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 60 requests/minute
|
||||||
|
*/
|
||||||
|
export const getCurrentUserProfile = <ThrowOnError extends boolean = false>(options?: Options<GetCurrentUserProfileData, ThrowOnError>) => {
|
||||||
|
return (options?.client ?? client).get<GetCurrentUserProfileResponses, unknown, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/users/me',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update Current User
|
||||||
|
*
|
||||||
|
* Update the current authenticated user's profile.
|
||||||
|
*
|
||||||
|
* Users can update their own profile information (except is_superuser).
|
||||||
|
*
|
||||||
|
* **Authentication**: Required (Bearer token)
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 30 requests/minute
|
||||||
|
*/
|
||||||
|
export const updateCurrentUser = <ThrowOnError extends boolean = false>(options: Options<UpdateCurrentUserData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).patch<UpdateCurrentUserResponses, UpdateCurrentUserErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/users/me',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Delete User
|
||||||
|
*
|
||||||
|
* Delete a specific user by their ID.
|
||||||
|
*
|
||||||
|
* **Authentication**: Required (Bearer token)
|
||||||
|
* **Authorization**: Superuser only
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 10 requests/minute
|
||||||
|
*
|
||||||
|
* **Note**: This performs a hard delete. Consider implementing soft deletes for production.
|
||||||
|
*/
|
||||||
|
export const deleteUser = <ThrowOnError extends boolean = false>(options: Options<DeleteUserData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).delete<DeleteUserResponses, DeleteUserErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/users/{user_id}',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get User by ID
|
||||||
|
*
|
||||||
|
* Get a specific user by their ID.
|
||||||
|
*
|
||||||
|
* **Authentication**: Required (Bearer token)
|
||||||
|
* **Authorization**:
|
||||||
|
* - Regular users: Can only access their own profile
|
||||||
|
* - Superusers: Can access any profile
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 60 requests/minute
|
||||||
|
*/
|
||||||
|
export const getUserById = <ThrowOnError extends boolean = false>(options: Options<GetUserByIdData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).get<GetUserByIdResponses, GetUserByIdErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/users/{user_id}',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update User
|
||||||
|
*
|
||||||
|
* Update a specific user by their ID.
|
||||||
|
*
|
||||||
|
* **Authentication**: Required (Bearer token)
|
||||||
|
* **Authorization**:
|
||||||
|
* - Regular users: Can only update their own profile (except is_superuser)
|
||||||
|
* - Superusers: Can update any profile
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 30 requests/minute
|
||||||
|
*/
|
||||||
|
export const updateUser = <ThrowOnError extends boolean = false>(options: Options<UpdateUserData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).patch<UpdateUserResponses, UpdateUserErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/users/{user_id}',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Change Current User Password
|
||||||
|
*
|
||||||
|
* Change the current authenticated user's password.
|
||||||
|
*
|
||||||
|
* Requires the current password for verification.
|
||||||
|
*
|
||||||
|
* **Authentication**: Required (Bearer token)
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 5 requests/minute
|
||||||
|
*/
|
||||||
|
export const changeCurrentUserPassword = <ThrowOnError extends boolean = false>(options: Options<ChangeCurrentUserPasswordData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).patch<ChangeCurrentUserPasswordResponses, ChangeCurrentUserPasswordErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/users/me/password',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* List My Active Sessions
|
||||||
|
*
|
||||||
|
* Get a list of all active sessions for the current user.
|
||||||
|
*
|
||||||
|
* This shows where you're currently logged in.
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 30 requests/minute
|
||||||
|
*/
|
||||||
|
export const listMySessions = <ThrowOnError extends boolean = false>(options?: Options<ListMySessionsData, ThrowOnError>) => {
|
||||||
|
return (options?.client ?? client).get<ListMySessionsResponses, unknown, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/sessions/me',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Revoke Specific Session
|
||||||
|
*
|
||||||
|
* Revoke a specific session by ID.
|
||||||
|
*
|
||||||
|
* This logs you out from that particular device.
|
||||||
|
* You can only revoke your own sessions.
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 10 requests/minute
|
||||||
|
*/
|
||||||
|
export const revokeSession = <ThrowOnError extends boolean = false>(options: Options<RevokeSessionData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).delete<RevokeSessionResponses, RevokeSessionErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/sessions/{session_id}',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Cleanup Expired Sessions
|
||||||
|
*
|
||||||
|
* Remove expired sessions for the current user.
|
||||||
|
*
|
||||||
|
* This is a cleanup operation to remove old session records.
|
||||||
|
*
|
||||||
|
* **Rate Limit**: 5 requests/minute
|
||||||
|
*/
|
||||||
|
export const cleanupExpiredSessions = <ThrowOnError extends boolean = false>(options?: Options<CleanupExpiredSessionsData, ThrowOnError>) => {
|
||||||
|
return (options?.client ?? client).delete<CleanupExpiredSessionsResponses, unknown, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/sessions/me/expired',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: List All Users
|
||||||
|
*
|
||||||
|
* Get paginated list of all users with filtering and search (admin only)
|
||||||
|
*/
|
||||||
|
export const adminListUsers = <ThrowOnError extends boolean = false>(options?: Options<AdminListUsersData, ThrowOnError>) => {
|
||||||
|
return (options?.client ?? client).get<AdminListUsersResponses, AdminListUsersErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/users',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: Create User
|
||||||
|
*
|
||||||
|
* Create a new user (admin only)
|
||||||
|
*/
|
||||||
|
export const adminCreateUser = <ThrowOnError extends boolean = false>(options: Options<AdminCreateUserData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).post<AdminCreateUserResponses, AdminCreateUserErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/users',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: Delete User
|
||||||
|
*
|
||||||
|
* Soft delete a user (admin only)
|
||||||
|
*/
|
||||||
|
export const adminDeleteUser = <ThrowOnError extends boolean = false>(options: Options<AdminDeleteUserData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).delete<AdminDeleteUserResponses, AdminDeleteUserErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/users/{user_id}',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: Get User Details
|
||||||
|
*
|
||||||
|
* Get detailed user information (admin only)
|
||||||
|
*/
|
||||||
|
export const adminGetUser = <ThrowOnError extends boolean = false>(options: Options<AdminGetUserData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).get<AdminGetUserResponses, AdminGetUserErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/users/{user_id}',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: Update User
|
||||||
|
*
|
||||||
|
* Update user information (admin only)
|
||||||
|
*/
|
||||||
|
export const adminUpdateUser = <ThrowOnError extends boolean = false>(options: Options<AdminUpdateUserData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).put<AdminUpdateUserResponses, AdminUpdateUserErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/users/{user_id}',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: Activate User
|
||||||
|
*
|
||||||
|
* Activate a user account (admin only)
|
||||||
|
*/
|
||||||
|
export const adminActivateUser = <ThrowOnError extends boolean = false>(options: Options<AdminActivateUserData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).post<AdminActivateUserResponses, AdminActivateUserErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/users/{user_id}/activate',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: Deactivate User
|
||||||
|
*
|
||||||
|
* Deactivate a user account (admin only)
|
||||||
|
*/
|
||||||
|
export const adminDeactivateUser = <ThrowOnError extends boolean = false>(options: Options<AdminDeactivateUserData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).post<AdminDeactivateUserResponses, AdminDeactivateUserErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/users/{user_id}/deactivate',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: Bulk User Action
|
||||||
|
*
|
||||||
|
* Perform bulk actions on multiple users (admin only)
|
||||||
|
*/
|
||||||
|
export const adminBulkUserAction = <ThrowOnError extends boolean = false>(options: Options<AdminBulkUserActionData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).post<AdminBulkUserActionResponses, AdminBulkUserActionErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/users/bulk-action',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: List Organizations
|
||||||
|
*
|
||||||
|
* Get paginated list of all organizations (admin only)
|
||||||
|
*/
|
||||||
|
export const adminListOrganizations = <ThrowOnError extends boolean = false>(options?: Options<AdminListOrganizationsData, ThrowOnError>) => {
|
||||||
|
return (options?.client ?? client).get<AdminListOrganizationsResponses, AdminListOrganizationsErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/organizations',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: Create Organization
|
||||||
|
*
|
||||||
|
* Create a new organization (admin only)
|
||||||
|
*/
|
||||||
|
export const adminCreateOrganization = <ThrowOnError extends boolean = false>(options: Options<AdminCreateOrganizationData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).post<AdminCreateOrganizationResponses, AdminCreateOrganizationErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/organizations',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: Delete Organization
|
||||||
|
*
|
||||||
|
* Delete an organization (admin only)
|
||||||
|
*/
|
||||||
|
export const adminDeleteOrganization = <ThrowOnError extends boolean = false>(options: Options<AdminDeleteOrganizationData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).delete<AdminDeleteOrganizationResponses, AdminDeleteOrganizationErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/organizations/{org_id}',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: Get Organization Details
|
||||||
|
*
|
||||||
|
* Get detailed organization information (admin only)
|
||||||
|
*/
|
||||||
|
export const adminGetOrganization = <ThrowOnError extends boolean = false>(options: Options<AdminGetOrganizationData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).get<AdminGetOrganizationResponses, AdminGetOrganizationErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/organizations/{org_id}',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: Update Organization
|
||||||
|
*
|
||||||
|
* Update organization information (admin only)
|
||||||
|
*/
|
||||||
|
export const adminUpdateOrganization = <ThrowOnError extends boolean = false>(options: Options<AdminUpdateOrganizationData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).put<AdminUpdateOrganizationResponses, AdminUpdateOrganizationErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/organizations/{org_id}',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: List Organization Members
|
||||||
|
*
|
||||||
|
* Get all members of an organization (admin only)
|
||||||
|
*/
|
||||||
|
export const adminListOrganizationMembers = <ThrowOnError extends boolean = false>(options: Options<AdminListOrganizationMembersData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).get<AdminListOrganizationMembersResponses, AdminListOrganizationMembersErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/organizations/{org_id}/members',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: Add Member to Organization
|
||||||
|
*
|
||||||
|
* Add a user to an organization (admin only)
|
||||||
|
*/
|
||||||
|
export const adminAddOrganizationMember = <ThrowOnError extends boolean = false>(options: Options<AdminAddOrganizationMemberData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).post<AdminAddOrganizationMemberResponses, AdminAddOrganizationMemberErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/organizations/{org_id}/members',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Admin: Remove Member from Organization
|
||||||
|
*
|
||||||
|
* Remove a user from an organization (admin only)
|
||||||
|
*/
|
||||||
|
export const adminRemoveOrganizationMember = <ThrowOnError extends boolean = false>(options: Options<AdminRemoveOrganizationMemberData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).delete<AdminRemoveOrganizationMemberResponses, AdminRemoveOrganizationMemberErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/admin/organizations/{org_id}/members/{user_id}',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get My Organizations
|
||||||
|
*
|
||||||
|
* Get all organizations the current user belongs to
|
||||||
|
*/
|
||||||
|
export const getMyOrganizations = <ThrowOnError extends boolean = false>(options?: Options<GetMyOrganizationsData, ThrowOnError>) => {
|
||||||
|
return (options?.client ?? client).get<GetMyOrganizationsResponses, GetMyOrganizationsErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/organizations/me',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get Organization Details
|
||||||
|
*
|
||||||
|
* Get details of an organization the user belongs to
|
||||||
|
*/
|
||||||
|
export const getOrganization = <ThrowOnError extends boolean = false>(options: Options<GetOrganizationData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).get<GetOrganizationResponses, GetOrganizationErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/organizations/{organization_id}',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Update Organization
|
||||||
|
*
|
||||||
|
* Update organization details (admin/owner only)
|
||||||
|
*/
|
||||||
|
export const updateOrganization = <ThrowOnError extends boolean = false>(options: Options<UpdateOrganizationData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).put<UpdateOrganizationResponses, UpdateOrganizationErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/organizations/{organization_id}',
|
||||||
|
...options,
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...options.headers
|
||||||
|
}
|
||||||
|
});
|
||||||
|
};
|
||||||
|
|
||||||
|
/**
|
||||||
|
* Get Organization Members
|
||||||
|
*
|
||||||
|
* Get all members of an organization (members can view)
|
||||||
|
*/
|
||||||
|
export const getOrganizationMembers = <ThrowOnError extends boolean = false>(options: Options<GetOrganizationMembersData, ThrowOnError>) => {
|
||||||
|
return (options.client ?? client).get<GetOrganizationMembersResponses, GetOrganizationMembersErrors, ThrowOnError>({
|
||||||
|
responseType: 'json',
|
||||||
|
security: [
|
||||||
|
{
|
||||||
|
scheme: 'bearer',
|
||||||
|
type: 'http'
|
||||||
|
}
|
||||||
|
],
|
||||||
|
url: '/api/v1/organizations/{organization_id}/members',
|
||||||
|
...options
|
||||||
|
});
|
||||||
|
};
|
||||||
1853
frontend/src/lib/api/generated/types.gen.ts
Normal file
1853
frontend/src/lib/api/generated/types.gen.ts
Normal file
File diff suppressed because it is too large
Load Diff
@@ -21,6 +21,11 @@ describe('Auth Store', () => {
|
|||||||
});
|
});
|
||||||
|
|
||||||
jest.clearAllMocks();
|
jest.clearAllMocks();
|
||||||
|
|
||||||
|
// Reset storage mocks to default successful implementations
|
||||||
|
(storage.saveTokens as jest.Mock).mockResolvedValue(undefined);
|
||||||
|
(storage.getTokens as jest.Mock).mockResolvedValue(null);
|
||||||
|
(storage.clearTokens as jest.Mock).mockResolvedValue(undefined);
|
||||||
});
|
});
|
||||||
|
|
||||||
describe('User validation', () => {
|
describe('User validation', () => {
|
||||||
|
|||||||
Reference in New Issue
Block a user