Files
sciagent/fe0/e2e/backup-admin-download.spec.ts
Thinh Lam 688fac73e9
CI/CD / backend (push) Failing after 2m8s
CI/CD / frontend (push) Failing after 1m40s
CI/CD / deploy (push) Has been skipped
sciagent code + Gitea Actions CI/CD
Co-Authored-By: Claude Opus 4.8 <noreply@anthropic.com>
2026-06-30 09:38:30 +07:00

156 lines
5.7 KiB
TypeScript

/**
* Full-stack backup smoke: API seeds submit (Postgres + MinIO), browser downloads ZIP as admin.
*
* Prerequisites (same as be0/tests/test_backup_e2e.py):
* - docker compose: postgres, minio, be0, fe0 (or host services on expected ports)
* - DB migrated through 009; MinIO buckets exist
* - DB migration 013 (email verification): Playwright cannot capture verify tokens; either
* use API/integration path for full coverage or extend this spec with a token source.
*
* be0 must be started with AUTH_ADMIN_EMAILS containing the same address as E2E_ADMIN_EMAIL.
*
* Run:
* export E2E_BACKUP=1
* export E2E_BASE_URL=http://localhost:8081
* export E2E_ADMIN_EMAIL=e2e-backup-admin@ump.edu.vn
* # In docker-compose be0 environment: AUTH_ADMIN_EMAILS=e2e-backup-admin@ump.edu.vn
* cd fe0 && npm install && npx playwright install chromium
* npm run test:e2e -- e2e/backup-admin-download.spec.ts
*/
import { test, expect } from "@playwright/test";
import { randomUUID } from "node:crypto";
const PASSWORD = "Testpass1!";
const MIN_PDF = Buffer.concat([
Buffer.from("%PDF-1.4\n%\xe2\xe3\xcf\xd3\n1 0 obj<<>>endobj\ntrailer<<>>\n%%EOF\n"),
Buffer.alloc(120, 0x30),
]);
function registerStaffFields(): Record<string, string> {
const suffix = randomUUID().replace(/-/g, "").slice(0, 8).toUpperCase();
return {
employeeId: `CB-${suffix}`,
academicTitleCode: "master",
unitNameFreetext: "Khoa kiểm thử",
jobTitle: "Cán bộ",
};
}
const stackEnabled = process.env.E2E_BACKUP === "1";
const adminEmail = process.env.E2E_ADMIN_EMAIL?.trim() ?? "";
test.describe("Admin backup ZIP (frontend → API → DB → MinIO)", () => {
test("admin can download backup after applicant submits PDF", async ({ page, request }) => {
test.skip(!stackEnabled, "Set E2E_BACKUP=1");
test.skip(!adminEmail, "Set E2E_ADMIN_EMAIL and list it in be0 AUTH_ADMIN_EMAILS");
const applicantEmail = `e2e-fe-app-${randomUUID().slice(0, 10)}@ump.edu.vn`;
const registerJson = (body: object) =>
request.post("/api/v1/auth/register", {
headers: { "Content-Type": "application/json" },
data: JSON.stringify(body),
});
const loginJson = (email: string) =>
request.post("/api/v1/auth/login", {
headers: { "Content-Type": "application/json" },
data: JSON.stringify({ email, password: PASSWORD }),
});
let r = await registerJson({
fullName: "E2E Applicant",
email: applicantEmail,
password: PASSWORD,
passwordConfirm: PASSWORD,
...registerStaffFields(),
});
expect(r.ok(), await r.text()).toBeTruthy();
r = await loginJson(applicantEmail);
expect(r.ok(), await r.text()).toBeTruthy();
const applicantToken = (await r.json()) as { accessToken: string };
expect(applicantToken.accessToken).toBeTruthy();
r = await request.post("/api/applications/new", {
headers: {
Authorization: `Bearer ${applicantToken.accessToken}`,
"Content-Type": "application/json",
},
data: JSON.stringify({ name: "E2E FE backup" }),
});
expect(r.ok(), await r.text()).toBeTruthy();
const created = (await r.json()) as { id?: string; application?: { draft_case_id?: string; id?: string } };
const caseId = String(created.application?.draft_case_id ?? "").trim();
let applicationId = String(created.id ?? created.application?.id ?? "").trim();
expect(caseId, JSON.stringify(created)).toBeTruthy();
expect(applicationId).toBeTruthy();
const meta = {
initiativeCaseId: caseId,
initiativeName: "E2E FE Backup",
authorName: "Applicant",
authorEmail: applicantEmail,
subjectId: "s1",
groupId: "g1",
topicType: "Hồ sơ PDF",
};
r = await request.post("/api/applications/submit", {
headers: { Authorization: `Bearer ${applicantToken.accessToken}` },
multipart: {
file: {
name: "e2e.pdf",
mimeType: "application/pdf",
buffer: MIN_PDF,
},
metadata: JSON.stringify(meta),
},
});
expect(r.ok(), await r.text()).toBeTruthy();
const submitted = (await r.json()) as { id?: string };
applicationId = String(submitted.id ?? applicationId);
r = await registerJson({
fullName: "E2E Admin",
email: adminEmail,
password: PASSWORD,
passwordConfirm: PASSWORD,
...registerStaffFields(),
});
if (r.ok()) {
const regBody = (await r.json()) as { user?: { roles?: string[] } };
expect(regBody.user?.roles).toContain("admin");
} else {
expect([400, 409, 422]).toContain(r.status());
}
r = await loginJson(adminEmail);
expect(r.ok(), await r.text()).toBeTruthy();
await page.goto("/login");
await page.locator("#login-email").fill(adminEmail);
await page.locator("#login-password").fill(PASSWORD);
await page.getByRole("button", { name: "Đăng nhập" }).click();
await page.waitForURL(/\/dashboard/u, { timeout: 30_000 });
await page.goto(`/dashboard/admin/applications/review?applicationId=${encodeURIComponent(applicationId)}`);
await expect(page.getByRole("heading", { name: /Xem hồ sơ đã nộp \(quản trị\)/u })).toBeVisible({
timeout: 30_000,
});
const backupBtn = page.getByRole("button", { name: /Tải bản sao lưu/u });
await expect(backupBtn).toBeVisible();
const [download] = await Promise.all([page.waitForEvent("download"), backupBtn.click()]);
expect(download.suggestedFilename()).toMatch(/\.zip$/i);
const path = await download.path();
expect(path).toBeTruthy();
const fs = await import("node:fs/promises");
const buf = await fs.readFile(path!);
expect(buf.length).toBeGreaterThan(100);
expect(buf[0]).toBe(0x50);
expect(buf[1]).toBe(0x4b);
});
});