Files
pn-new-crm/tests/e2e/realapi/sales-journey-signing.spec.ts
Matt 7591231c47
All checks were successful
Build & Push Docker Images / lint (push) Successful in 3m12s
Build & Push Docker Images / build-and-push (push) Successful in 8m24s
test(e2e): add Initiative 4 end-to-end + integration specs
Sales-process coverage (launch-readiness Initiative 4):
- exhaustive: full 7-stage sales journey + illegal-skip rejection + deposit
  total + tenancy/berth-sold; multi-berth EOI berth-range; EOI pathway parity
  (in-app vs Documenso, shared EoiContext); mobile-viewport journey.
- realapi (Documenso-gated, opt-in): generate-and-sign + post-EOI stages.
- integration: Documenso DOCUMENT_COMPLETED webhook idempotency (3x replay ->
  single file/audit write); storage backend swap (s3 <-> filesystem) with a
  real on-disk filesystem round-trip.
- visual: Reports UI snapshot cases (baselines captured separately).

1615 unit/integration pass; tsc + lint clean. Test-only change (specs are not
bundled into the app image) - no app behavior modified.

Co-Authored-By: Claude Opus 4.8 (1M context) <noreply@anthropic.com>
2026-06-04 14:10:35 +02:00

134 lines
5.6 KiB
TypeScript

import 'dotenv/config';
import { test, expect } from '@playwright/test';
import { login, apiHeaders } from '../smoke/helpers';
import { seedFullyLoadedDeal, changeStage, readStage } from '../fixtures/sales-journey';
/**
* End-to-end sales journey — Documenso-dependent half.
*
* Covers the signing legs of the pipeline that need a live Documenso
* instance: generating the EOI through the documenso-template pathway
* and confirming the document is created remotely + persisted locally,
* then walking the interest forward through the post-signing stages.
*
* Skips cleanly when the Documenso env is absent (same guard as
* `tests/e2e/realapi/documenso-real-api.spec.ts`), so this spec is a
* no-op in CI runs that don't opt into the real-API project.
*
* The deterministic stage/UI assertions for the journey live in
* `tests/e2e/exhaustive/10-sales-journey.spec.ts`; this file only adds
* the legs that genuinely require the external signer.
*/
const DOCUMENSO_BASE = process.env.DOCUMENSO_API_URL;
const DOCUMENSO_API_KEY = process.env.DOCUMENSO_API_KEY;
test.describe('realapi: sales journey — Documenso signing legs', () => {
test.skip(!DOCUMENSO_BASE || !DOCUMENSO_API_KEY, 'DOCUMENSO_API_URL / DOCUMENSO_API_KEY not set');
test.beforeEach(async ({ page }) => {
await login(page, 'super_admin');
});
test('generate-and-sign EOI then advance the deal through signing stages', async ({ page }) => {
const headers = await apiHeaders(page);
const deal = await seedFullyLoadedDeal(page, 'Documenso Journey');
// ─── 1. Qualify + move into the EOI stage ─────────────────────────────
await changeStage(page.request, headers, deal.interestId, 'qualified');
await changeStage(page.request, headers, deal.interestId, 'eoi');
expect(await readStage(page.request, headers, deal.interestId)).toBe('eoi');
// ─── 2. Fire generate-and-sign through the documenso-template pathway ──
const signRes = await page.request.post(
'/api/v1/document-templates/documenso-template/generate-and-sign',
{
headers,
data: {
interestId: deal.interestId,
clientId: deal.clientId,
berthId: deal.berthId,
pathway: 'documenso-template',
signers: [],
},
},
);
expect(signRes.ok(), `generate-and-sign: ${signRes.status()} ${await signRes.text()}`).toBe(
true,
);
const body = (await signRes.json()) as {
data: { document: { id: string; documensoId: string | null; status: string } };
};
expect(body.data.document.documensoId, 'documensoId persisted locally').toBeTruthy();
expect(body.data.document.status).toBe('sent');
// ─── 3. Confirm the document exists on the Documenso side ─────────────
const documensoId = body.data.document.documensoId!;
const documensoRes = await page.request.get(
`${DOCUMENSO_BASE}/api/v1/documents/${documensoId}`,
{ headers: { Authorization: `Bearer ${DOCUMENSO_API_KEY}` } },
);
expect(
documensoRes.ok(),
`documenso GET: ${documensoRes.status()} ${await documensoRes.text()}`,
).toBe(true);
const documensoDoc = (await documensoRes.json()) as { id: number; status: string };
expect(String(documensoDoc.id)).toBe(documensoId);
expect(['PENDING', 'DRAFT'], `unexpected status ${documensoDoc.status}`).toContain(
documensoDoc.status,
);
// ─── 4. Walk the post-EOI legs of the pipeline ────────────────────────
// The webhook-driven signed → reservation → contract advances happen
// on real signing events (which Documenso has no machine-driveable
// "sign on behalf of recipient" endpoint for). We exercise the same
// forward path the webhook would, recording the deposit en route so
// the deposit_paid gate is met realistically.
await changeStage(page.request, headers, deal.interestId, 'reservation');
const depositRes = await page.request.post(`/api/v1/interests/${deal.interestId}/payments`, {
headers,
data: {
interestId: deal.interestId,
paymentType: 'deposit',
amount: '25000',
currency: 'EUR',
receivedAt: new Date().toISOString(),
},
});
expect(depositRes.ok(), `deposit: ${depositRes.status()} ${await depositRes.text()}`).toBe(
true,
);
await changeStage(page.request, headers, deal.interestId, 'deposit_paid');
await changeStage(page.request, headers, deal.interestId, 'contract');
expect(await readStage(page.request, headers, deal.interestId)).toBe('contract');
// ─── 5. Complete: tenancy + berth sold ────────────────────────────────
const tenancyRes = await page.request.post(`/api/v1/berths/${deal.berthId}/tenancies`, {
headers,
data: {
berthId: deal.berthId,
clientId: deal.clientId,
yachtId: deal.yachtId,
interestId: deal.interestId,
startDate: new Date().toISOString(),
tenureType: 'permanent',
},
});
expect(
tenancyRes.ok(),
`create tenancy: ${tenancyRes.status()} ${await tenancyRes.text()}`,
).toBe(true);
const statusRes = await page.request.patch(`/api/v1/berths/${deal.berthId}/status`, {
headers,
data: { status: 'sold', reason: 'Contract signed', interestId: deal.interestId },
});
expect(statusRes.ok(), `berth → sold: ${statusRes.status()}`).toBe(true);
expect(((await statusRes.json()) as { data: { status: string } }).data.status).toBe('sold');
});
});