Monorepo Turborepo (pnpm workspaces) avec 3 packages :
- apps/web : SPA React 19 + Vite 8 + Tailwind v4 (CSS-first)
• TanStack Router (file-based, auto code-splitting), Query, Form
• Radix primitives bruts + CVA + clsx + tailwind-merge
• MSW pour mocker l'API tant qu'Adonis n'est pas branché
• Polices Bricolage Grotesque + Inter self-hostées via fontsource
• Tokens marque (rubis, cream, ink) exposés via @theme
• Primitives maison : Gem, Brand, Eyebrow, Button, Input, Field
• Route /login full flow : TanStack Form + Zod + mutation Query
- apps/api : Adonis 7 (kit api, scaffold via create-adonisjs)
• Auth access tokens (Bearer) — cf. ADR-017
• Tuyau core déjà câblé pour la génération de types
• Routes /api/v1/auth/{signup,login} + /api/v1/account/{profile,logout}
• Minimal — uniquement le pont front ↔ back
- packages/shared : types TS + schemas Zod + constantes
• Source unique de vérité partagée api ↔ web
• Domaines : User, Org, Auth, Client, Invoice, Plan
Tooling racine : Turbo, ESLint v9 flat, Prettier, husky, lint-staged.
CLAUDE.md et docs/decisions.md mis à jour avec ADR-014 à ADR-018
(stack, monorepo, PG existant, Bearer tokens, MinIO existant)
et le pointeur vers docs/tech/architecture.md.
Logo Rubis déplacé de landing/assets/ vers /assets/ (source unique
réutilisée par la landing et l'app).
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
70 lines
2.3 KiB
TypeScript
70 lines
2.3 KiB
TypeScript
import { HttpContext } from '@adonisjs/core/http'
|
|
import { BaseSerializer } from '@adonisjs/core/transformers'
|
|
import { type SimplePaginatorMetaKeys } from '@adonisjs/lucid/types/querybuilder'
|
|
|
|
/**
|
|
* Custom serializer for API responses that ensures consistent JSON structure
|
|
* across all API endpoints. Wraps response data in a 'data' property and handles
|
|
* pagination metadata for Lucid ORM query results.
|
|
*/
|
|
class ApiSerializer extends BaseSerializer<{
|
|
Wrap: 'data'
|
|
PaginationMetaData: SimplePaginatorMetaKeys
|
|
}> {
|
|
/**
|
|
* Wraps all serialized data under this key in the response object.
|
|
* Example: { data: [...] } instead of returning raw arrays/objects
|
|
*/
|
|
wrap: 'data' = 'data'
|
|
|
|
/**
|
|
* Validates and defines pagination metadata structure for paginated responses.
|
|
* Ensures that pagination info from Lucid queries is properly formatted.
|
|
*
|
|
* @throws Error if metadata doesn't match Lucid's pagination structure
|
|
*/
|
|
definePaginationMetaData(metaData: unknown): SimplePaginatorMetaKeys {
|
|
if (!this.isLucidPaginatorMetaData(metaData)) {
|
|
throw new Error(
|
|
'Invalid pagination metadata. Expected metadata to contain Lucid pagination keys'
|
|
)
|
|
}
|
|
return metaData
|
|
}
|
|
}
|
|
|
|
/**
|
|
* Single instance of ApiSerializer used across the application
|
|
*/
|
|
const serializer = new ApiSerializer()
|
|
const serialize = Object.assign(
|
|
function (this: HttpContext, ...[data, resolver]: Parameters<ApiSerializer['serialize']>) {
|
|
return serializer.serialize(data, resolver ?? this.containerResolver)
|
|
},
|
|
{
|
|
withoutWrapping(
|
|
this: HttpContext,
|
|
...[data, resolver]: Parameters<ApiSerializer['serializeWithoutWrapping']>
|
|
) {
|
|
return serializer.serializeWithoutWrapping(data, resolver ?? this.containerResolver)
|
|
},
|
|
}
|
|
) as ApiSerializer['serialize'] & { withoutWrapping: ApiSerializer['serializeWithoutWrapping'] }
|
|
|
|
/**
|
|
* Adds the serialize method to all HttpContext instances.
|
|
* Usage in controllers: return ctx.serialize(data)
|
|
* This ensures all API responses follow the same structure with data wrapping.
|
|
*/
|
|
HttpContext.instanceProperty('serialize', serialize)
|
|
|
|
/**
|
|
* Module augmentation to add the serialize method to HttpContext.
|
|
* This allows controllers to use ctx.serialize() for consistent API responses.
|
|
*/
|
|
declare module '@adonisjs/core/http' {
|
|
export interface HttpContext {
|
|
serialize: typeof serialize
|
|
}
|
|
}
|