Permet de faire vivre Rubis en accéléré pour démontrer le produit à des
prospects, SANS impacter la prod. Les vrais users ont demoMode=false par
défaut → toute la logique démo est court-circuitée.
Architecture (priorité : zéro impact prod, codebase propre)
Phase 1 — Abstraction Clock
- Migration : organizations.demo_mode + virtual_now + demo_speed_factor
(défaut false/null/1, zéro effet sur les orgs existantes)
- services/clock.ts : now(orgId?) → DateTime.utc() en prod, virtualNow
en démo. Cache mémoire 250ms pour pas spammer la DB. Helpers
setVirtualNow / setDemoMode pour les transitions.
- Refacto 7 fichiers : relance_scheduler, checkin_scheduler, dashboard,
send_relance_job, send_checkin_job, mail_dispatcher (buildRelanceVars
daysLate), activity_recorder, checkin_controller, invoices_controller
(buildTimeline + markPaid). DateTime.now() → clock.now(orgId).
- Tests existants (51) passent identique → preuve que la prod est intacte.
Phase 2 — Capture emails + dispatch
- Migration : demo_captured_emails (kind, to, from, subject, body, sent_at,
meta) — index sur (org, sent_at desc) pour l'inbox.
- services/demo/capture.ts : captureEmailIfDemo() — UNIQUE point de fork
dans la prod (deux lignes dans mail_dispatcher : if captured return).
Hors démo, fonction retourne false → flux Resend inchangé.
- services/demo/dispatch.ts : tickAndDispatch(orgId, target) → bump
virtual_now, trouve les tasks dues (relance + checkin), invoke les
handlers existants synchronement (skip BullMQ, propre). Retourne les
events fired pour l'UI.
- POST /api/v1/demo/{start,end,tick} + GET /demo/{state,inbox}, toutes
protégées par requireDemoOrg() (403 si demoMode=false).
Phase 3 — UI horloge "vivante"
- lib/demo.ts : useDemoState, useDemoTick (boucle rAF locale qui avance
virtualNow à `speed * elapsed` jours/sec, sync backend toutes les
250ms, auto-pause sur fired events). Pas de boutons +1j/+3j —
l'horloge tourne vraiment.
- DemoClock (top-right, fixed) : date pleine en font display, rail
rubis-glow avec pastille ◆ qui glisse vers le prochain event,
play/pause + sélecteur 1x/2x/5x. Auto-cachée si demoMode=false.
- DemoEmailSlide : slide-over droite quand event fires — affiche
l'email capturé (de/à/sujet/body) façon vrai client mail. Pause
forcée tant que tous les events ne sont pas acquittés ("comme si
le temps était vraiment passé").
- DemoToggle dans /parametres : démarrer/quitter le mode démo, avec
copy explicite ("emails capturés, pas envoyés à de vrais clients").
Le code démo vit isolé dans services/demo/, controllers/demo_controller.ts,
components/demo/, lib/demo.ts. La prod ne référence ces fichiers QUE
via captureEmailIfDemo dans mail_dispatcher.
Co-Authored-By: Claude Opus 4.7 <noreply@anthropic.com>
107 lines
3.4 KiB
TypeScript
107 lines
3.4 KiB
TypeScript
import CheckinTask from '#models/checkin_task'
|
|
import Invoice from '#models/invoice'
|
|
import { getQueue } from '#services/queue'
|
|
import { generateCheckinToken } from '#services/checkin_token'
|
|
import * as clock from '#services/clock'
|
|
import app from '@adonisjs/core/services/app'
|
|
import type { TransactionClientContract } from '@adonisjs/lucid/types/database'
|
|
|
|
const CHECKIN_QUEUE = 'checkins'
|
|
|
|
function shouldEnqueue(): boolean {
|
|
return app.getEnvironment() !== 'test'
|
|
}
|
|
|
|
/**
|
|
* Programme un check-in pour une facture.
|
|
*
|
|
* V1 : 1 check-in par facture, envoyé à `dueDate` (pile à l'échéance).
|
|
* Si dueDate est dans le passé → envoie immédiat (à `now + 1min`),
|
|
* pour que les factures importées en retard reçoivent quand même un
|
|
* check-in.
|
|
*
|
|
* Le token est généré ici (plain) — on retourne le plain pour permettre
|
|
* au caller de le passer dans des emails de test si besoin, mais en
|
|
* pratique seul le hash est stocké et lu via SendCheckinJob.
|
|
*
|
|
* Idempotent par invoice : si une CheckinTask `scheduled` existe déjà,
|
|
* on la cancelle d'abord puis on en crée une nouvelle (cas re-scheduling
|
|
* après changement de dueDate).
|
|
*
|
|
* En tests : la task DB est créée mais l'enqueue BullMQ est skippé
|
|
* (les tx auto-rollback laisseraient des jobs orphelins en Redis sinon).
|
|
*/
|
|
export async function scheduleCheckinForInvoice(
|
|
invoice: Invoice,
|
|
trx?: TransactionClientContract
|
|
): Promise<{ task: CheckinTask; plain: string } | null> {
|
|
// Cancel l'éventuelle CheckinTask scheduled précédente.
|
|
const existing = await CheckinTask.query(trx ? { client: trx } : undefined)
|
|
.where('invoice_id', invoice.id)
|
|
.where('status', 'scheduled')
|
|
const queue = shouldEnqueue() ? getQueue(CHECKIN_QUEUE) : null
|
|
for (const t of existing) {
|
|
if (queue) await queue.remove(`checkin-${t.id}`).catch(() => {})
|
|
t.useTransaction(trx ?? (null as never))
|
|
t.status = 'expired'
|
|
await t.save()
|
|
}
|
|
|
|
const now = await clock.now(invoice.organizationId)
|
|
const sendAtRaw = invoice.dueDate
|
|
const sendAt = sendAtRaw < now ? now.plus({ minutes: 1 }) : sendAtRaw
|
|
|
|
const { plain, hashed } = generateCheckinToken()
|
|
|
|
const task = await CheckinTask.create(
|
|
{
|
|
organizationId: invoice.organizationId,
|
|
invoiceId: invoice.id,
|
|
sendAt,
|
|
tokenHash: hashed,
|
|
status: 'scheduled',
|
|
sentAt: null,
|
|
answeredAt: null,
|
|
answer: null,
|
|
},
|
|
trx ? { client: trx } : undefined
|
|
)
|
|
|
|
if (queue) {
|
|
const delay = Math.max(0, sendAt.toMillis() - now.toMillis())
|
|
await queue.add(
|
|
'send-checkin',
|
|
{ taskId: task.id, plain },
|
|
{
|
|
delay,
|
|
jobId: `checkin-${task.id}`,
|
|
attempts: 3,
|
|
backoff: { type: 'exponential', delay: 30_000 },
|
|
}
|
|
)
|
|
}
|
|
|
|
return { task, plain }
|
|
}
|
|
|
|
/**
|
|
* Annule le check-in scheduled d'une facture (appelé par mark-paid).
|
|
*/
|
|
export async function cancelCheckinForInvoice(
|
|
invoiceId: string,
|
|
trx?: TransactionClientContract
|
|
): Promise<void> {
|
|
const tasks = await CheckinTask.query(trx ? { client: trx } : undefined)
|
|
.where('invoice_id', invoiceId)
|
|
.where('status', 'scheduled')
|
|
if (tasks.length === 0) return
|
|
|
|
const queue = shouldEnqueue() ? getQueue(CHECKIN_QUEUE) : null
|
|
for (const t of tasks) {
|
|
if (queue) await queue.remove(`checkin-${t.id}`).catch(() => {})
|
|
t.useTransaction(trx ?? (null as never))
|
|
t.status = 'expired'
|
|
await t.save()
|
|
}
|
|
}
|