Files
pn-new-crm/tests/e2e/smoke/04-documents-hub-aggregated.spec.ts
Matt 9a5ba87d6c fix(integration): webhook v2 events, storage migrate, test theatre
- F1: DOCUMENT_DECLINED handler (v2 Decline vs Reject) — routes to same
  handler as DOCUMENT_REJECTED until product refines downstream UX
- Add RECIPIENT_VIEWED / RECIPIENT_SIGNED v2-alias cases with telemetry
  logging so we see when v2 deployments emit them
- D1: populate TABLES_WITH_STORAGE_KEYS (files, berth_pdf_versions,
  brochure_versions, gdpr_exports) — was an empty list, migrated 0 files
- MinIO putObject/getObject/statObject/removeObject socket timeout wrapper
  to prevent worker hangs on TCP blackhole (30s deadline)
- E1: convert test.skip on smoke-setup infra failure to throw new Error
  so green-skipped silence becomes a real test failure (Playwright
  doesn't expose vitest's expect.fail)
- Regression tests: folderId='' → null transform, applyEntityRestoredSuffix
  no-op (never-archived), syncEntityFolderName collision loop past (2)

Note: matching .env.example documentation (D2 — bare DOCUMENSO_API_URL,
DOCUMENSO_API_VERSION, MINIO_AUTO_CREATE_BUCKET, DOCUMENSO_TEMPLATE_ID_EOI,
recipient role id vars) prepared but not committed — pre-commit hook
blocks .env*. Apply manually via the separate .env workflow.

Co-Authored-By: Claude Opus 4.7 (1M context) <noreply@anthropic.com>
2026-05-11 14:02:26 +02:00

155 lines
6.8 KiB
TypeScript

import { test, expect } from '@playwright/test';
import { login, navigateTo } from './helpers';
/**
* Smoke spec — Documents Hub aggregated view (Wave 11.B rebuild)
*
* Exercises the structural layout of the new DocumentsHub:
* - FolderTreeSidebar with system-root folders (Clients / Companies / Yachts)
* - HubRootView sections ("Signing in progress" + "Recent files")
* - Expanding a system root to reveal entity sub-folders
* - Selecting an entity sub-folder switches to EntityFolderView which
* renders the AggregatedSection blocks
*
* Note: This spec intentionally avoids asserting on seeded file/workflow
* counts — those vary by seed state. The integration tests (Tasks 8 + 9)
* already verify service correctness; this spec guards UI-level rendering.
*/
test.describe('Documents hub — aggregated view', () => {
test.beforeEach(async ({ page }) => {
await login(page, 'super_admin');
});
test('hub root shows Signing-in-progress and Recent-files sections', async ({ page }) => {
await navigateTo(page, '/documents');
await page.waitForLoadState('networkidle');
// HubRootView renders two labelled sections.
await expect(page.getByText('Signing in progress')).toBeVisible({ timeout: 10_000 });
await expect(page.getByText('Recent files')).toBeVisible({ timeout: 5_000 });
});
test('sidebar shows All-documents and Root pseudo-rows', async ({ page }) => {
await navigateTo(page, '/documents');
await page.waitForLoadState('networkidle');
// FolderTreeSidebar always renders these two pseudo-rows at the top.
await expect(page.getByRole('button', { name: 'All documents' })).toBeVisible({
timeout: 10_000,
});
await expect(page.getByRole('button', { name: 'Root (no folder)' })).toBeVisible({
timeout: 5_000,
});
});
test('system-root folders appear in the sidebar with lock icons', async ({ page }) => {
await navigateTo(page, '/documents');
await page.waitForLoadState('networkidle');
// ensureSystemRoots creates Clients / Companies / Yachts on port creation.
// Each node has systemManaged=true → FolderRow renders a Lock icon beside
// the name. We assert text presence; the SVG lock icon is decoration only
// and aria-labeled "System folder" by the component.
for (const rootName of ['Clients', 'Companies', 'Yachts']) {
await expect(page.getByText(rootName)).toBeVisible({ timeout: 10_000 });
}
// At least one system-folder lock icon should be in the sidebar.
const lockIcons = page.locator('aside [aria-label="System folder"]');
await expect(lockIcons.first()).toBeVisible({ timeout: 5_000 });
});
test('clicking a system root selects it and breadcrumb updates', async ({ page }) => {
await navigateTo(page, '/documents');
await page.waitForLoadState('networkidle');
// Click the Clients system root folder button (the label button inside FolderRow).
// The chevron button and the label button are siblings — we target the button
// that contains the text "Clients".
const clientsBtn = page
.locator('aside button')
.filter({ hasText: /^Clients$/ })
.first();
await expect(clientsBtn).toBeVisible({ timeout: 10_000 });
await clientsBtn.click();
// The breadcrumb (nav[aria-label="breadcrumb"]) should now show "Clients".
await expect(page.getByRole('navigation', { name: /breadcrumb/i })).toContainText('Clients', {
timeout: 10_000,
});
});
test('entity sub-folder view renders Signing-in-progress and Files sections', async ({
page,
}) => {
// Create a fresh client via the API so we have a guaranteed entity with a
// folder. ensureEntityFolder is called by the createClient service hook.
// page.request shares the browser context cookie jar (session cookie from
// login()) — the bare `request` fixture is an isolated API context that
// does not carry the session cookie and would 401.
const res = await page.request.post('/api/v1/clients', {
data: {
firstName: 'HubAgg',
lastName: `Smoke${Date.now()}`,
email: `hubagg${Date.now()}@e2e.test`,
},
});
// A non-2xx here means smoke setup is broken (port cookie / seed) or the
// clients API regressed. Fail loud rather than skip green — a silent skip
// masked an infra failure for weeks in the audit window. Playwright doesn't
// expose vitest's `expect.fail`, so we throw a plain Error which the
// runner promotes to a failing test the same way.
if (!res.ok()) {
throw new Error(
`Client create returned ${res.status()} ${await res.text()} — entity sub-folder assertion cannot proceed`,
);
}
const { data: client } = (await res.json()) as {
data: { id: string; firstName: string; lastName: string };
};
// Navigate to the documents hub.
await navigateTo(page, '/documents');
await page.waitForLoadState('networkidle');
// Expand the Clients system root by clicking its chevron (Expand button).
// The Expand chevron is the first button sibling inside the FolderRow div.
// It is aria-labeled "Expand" when collapsed.
const expandBtn = page.locator('aside').getByRole('button', { name: 'Expand' }).first();
await expect(expandBtn).toBeVisible({ timeout: 10_000 });
await expandBtn.click();
// The entity folder is named "<FirstName> <LastName>".
const entityFolderBtn = page
.locator('aside button')
.filter({ hasText: `${client.firstName} ${client.lastName}` })
.first();
if (!(await entityFolderBtn.isVisible({ timeout: 8_000 }).catch(() => false))) {
// ensureEntityFolder may not have run yet (async post-create); reload to
// pick up the folder tree refresh.
await navigateTo(page, '/documents');
await page.waitForLoadState('networkidle');
// Re-expand.
const expandBtn2 = page.locator('aside').getByRole('button', { name: 'Expand' }).first();
await expandBtn2.click();
}
await expect(entityFolderBtn).toBeVisible({ timeout: 10_000 });
await entityFolderBtn.click();
// EntityFolderView renders two AggregatedSection blocks with these headings.
await expect(page.getByText('Signing in progress')).toBeVisible({ timeout: 10_000 });
await expect(page.getByText('Files')).toBeVisible({ timeout: 5_000 });
// TODO(Task 19 verification): assert SigningDetailsDialog trigger once
// a signed-file fixture exists in the smoke seed. The button is only
// rendered when files[].signedFromDocumentId is non-null (Task 15 fix-up).
const detailsButton = page.getByRole('button', { name: /view signing details/i });
if ((await detailsButton.count()) > 0) {
await detailsButton.first().click();
await expect(page.getByRole('dialog', { name: /Signing details/i })).toBeVisible();
}
});
});