Ajoute une couche end-to-end où un Chromium drive la SPA + API ensemble
contre une DB Postgres séparée, avec Stripe entièrement mocké au niveau
API. 6 scénarios couverts (signup + onboarding + 4 sur le billing trial).
Architecture :
- DB `rubis_test_e2e` séparée, TRUNCATE entre tests (~50 ms reset)
- Routes test-only `/__test__/*` gated par NODE_ENV=test_e2e
(reset, install Stripe mock, fire webhook, lire org state, last-org)
- Stripe mocké via __setStripeForTests — pas d'appel réseau
- Playwright spawn API + SPA automatiquement (webServer config)
- CORS étendu à test_e2e pour le cross-origin localhost:5173 → :3333
Scénarios :
- signup.spec.ts : signup → onboarding 3 étapes → dashboard (assert rubis hero)
- billing-trial.spec.ts :
• démarrer essai 14j → redirect Stripe Checkout (mock)
• fallback Free 2 factures continue l'onboarding
• webhook checkout.completed → org en trialing + trial_ends_at
• retour ?trial=cancel après abandon
• inspection DB : stripeCustomerId posé après start-trial
Scripts :
- pnpm e2e (headless)
- pnpm e2e:headed (Chromium visible)
- pnpm e2e:ui (mode interactif Playwright)
- pnpm e2e:setup (crée + migre rubis_test_e2e via docker exec)
Documentation : docs/tech/e2e-tests.md — architecture, scénarios,
extensions, CI, troubleshooting.
Limites assumées :
- L'UI Stripe Checkout (3DS, formulaire CB) n'est pas testée — externe.
Pour ça : playbook manuel docs/tech/stripe-trial-e2e-playbook.md.
- Le rendu du banner "Essai Pro" n'est pas asserté en E2E à cause de
TanStack Query staleTime — couvert par les tests vitest à la place.
État global du chantier billing : 127 tests japa + 6 Playwright + 11
vitest = couverture multi-niveaux.
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
23 lines
941 B
TypeScript
23 lines
941 B
TypeScript
import { ClientSchema } from '#database/schema'
|
|
import { belongsTo, hasMany } from '@adonisjs/lucid/orm'
|
|
import type { BelongsTo, HasMany } from '@adonisjs/lucid/types/relations'
|
|
import Organization from '#models/organization'
|
|
import Invoice from '#models/invoice'
|
|
|
|
export default class Client extends ClientSchema {
|
|
/**
|
|
* Le champ `address` (existant, string libre) est conservé pour les
|
|
* clients importés avant l'éditeur natif. Les champs structurés
|
|
* (`address_line_1`, `address_zip`, etc.) viennent de la migration
|
|
* `1778800000100_enrich_clients_for_invoicing` + alignement Lucid via
|
|
* `1779000000100_rename_client_address_line_columns`. Le nouveau code
|
|
* lit en priorité les champs structurés et retombe sur `address` s'ils
|
|
* sont vides.
|
|
*/
|
|
@belongsTo(() => Organization)
|
|
declare organization: BelongsTo<typeof Organization>
|
|
|
|
@hasMany(() => Invoice)
|
|
declare invoices: HasMany<typeof Invoice>
|
|
}
|