feat: add release smoke workflow
This commit is contained in:
146
tests/release-smoke/docker-auth-onboarding.spec.ts
Normal file
146
tests/release-smoke/docker-auth-onboarding.spec.ts
Normal file
@@ -0,0 +1,146 @@
|
||||
import { expect, test, type Page } from "@playwright/test";
|
||||
|
||||
const ADMIN_EMAIL =
|
||||
process.env.PAPERCLIP_RELEASE_SMOKE_EMAIL ??
|
||||
process.env.SMOKE_ADMIN_EMAIL ??
|
||||
"smoke-admin@paperclip.local";
|
||||
const ADMIN_PASSWORD =
|
||||
process.env.PAPERCLIP_RELEASE_SMOKE_PASSWORD ??
|
||||
process.env.SMOKE_ADMIN_PASSWORD ??
|
||||
"paperclip-smoke-password";
|
||||
|
||||
const COMPANY_NAME = `Release-Smoke-${Date.now()}`;
|
||||
const AGENT_NAME = "CEO";
|
||||
const TASK_TITLE = "Release smoke task";
|
||||
|
||||
async function signIn(page: Page) {
|
||||
await page.goto("/");
|
||||
await expect(page).toHaveURL(/\/auth/);
|
||||
|
||||
await page.locator('input[type="email"]').fill(ADMIN_EMAIL);
|
||||
await page.locator('input[type="password"]').fill(ADMIN_PASSWORD);
|
||||
await page.getByRole("button", { name: "Sign In" }).click();
|
||||
|
||||
await expect(page).not.toHaveURL(/\/auth/, { timeout: 20_000 });
|
||||
}
|
||||
|
||||
async function openOnboarding(page: Page) {
|
||||
const wizardHeading = page.locator("h3", { hasText: "Name your company" });
|
||||
const startButton = page.getByRole("button", { name: "Start Onboarding" });
|
||||
|
||||
await expect(wizardHeading.or(startButton)).toBeVisible({ timeout: 20_000 });
|
||||
|
||||
if (await startButton.isVisible()) {
|
||||
await startButton.click();
|
||||
}
|
||||
|
||||
await expect(wizardHeading).toBeVisible({ timeout: 10_000 });
|
||||
}
|
||||
|
||||
test.describe("Docker authenticated onboarding smoke", () => {
|
||||
test("logs in, completes onboarding, and triggers the first CEO run", async ({
|
||||
page,
|
||||
}) => {
|
||||
await signIn(page);
|
||||
await openOnboarding(page);
|
||||
|
||||
await page.locator('input[placeholder="Acme Corp"]').fill(COMPANY_NAME);
|
||||
await page.getByRole("button", { name: "Next" }).click();
|
||||
|
||||
await expect(
|
||||
page.locator("h3", { hasText: "Create your first agent" })
|
||||
).toBeVisible({ timeout: 10_000 });
|
||||
|
||||
await expect(page.locator('input[placeholder="CEO"]')).toHaveValue(AGENT_NAME);
|
||||
await page.getByRole("button", { name: "Process" }).click();
|
||||
await page.locator('input[placeholder="e.g. node, python"]').fill("echo");
|
||||
await page
|
||||
.locator('input[placeholder="e.g. script.js, --flag"]')
|
||||
.fill("release smoke");
|
||||
await page.getByRole("button", { name: "Next" }).click();
|
||||
|
||||
await expect(
|
||||
page.locator("h3", { hasText: "Give it something to do" })
|
||||
).toBeVisible({ timeout: 10_000 });
|
||||
await page
|
||||
.locator('input[placeholder="e.g. Research competitor pricing"]')
|
||||
.fill(TASK_TITLE);
|
||||
await page.getByRole("button", { name: "Next" }).click();
|
||||
|
||||
await expect(
|
||||
page.locator("h3", { hasText: "Ready to launch" })
|
||||
).toBeVisible({ timeout: 10_000 });
|
||||
await expect(page.getByText(COMPANY_NAME)).toBeVisible();
|
||||
await expect(page.getByText(AGENT_NAME)).toBeVisible();
|
||||
await expect(page.getByText(TASK_TITLE)).toBeVisible();
|
||||
|
||||
await page.getByRole("button", { name: "Create & Open Issue" }).click();
|
||||
await expect(page).toHaveURL(/\/issues\//, { timeout: 10_000 });
|
||||
|
||||
const baseUrl = new URL(page.url()).origin;
|
||||
|
||||
const companiesRes = await page.request.get(`${baseUrl}/api/companies`);
|
||||
expect(companiesRes.ok()).toBe(true);
|
||||
const companies = (await companiesRes.json()) as Array<{ id: string; name: string }>;
|
||||
const company = companies.find((entry) => entry.name === COMPANY_NAME);
|
||||
expect(company).toBeTruthy();
|
||||
|
||||
const agentsRes = await page.request.get(
|
||||
`${baseUrl}/api/companies/${company!.id}/agents`
|
||||
);
|
||||
expect(agentsRes.ok()).toBe(true);
|
||||
const agents = (await agentsRes.json()) as Array<{
|
||||
id: string;
|
||||
name: string;
|
||||
role: string;
|
||||
adapterType: string;
|
||||
}>;
|
||||
const ceoAgent = agents.find((entry) => entry.name === AGENT_NAME);
|
||||
expect(ceoAgent).toBeTruthy();
|
||||
expect(ceoAgent!.role).toBe("ceo");
|
||||
expect(ceoAgent!.adapterType).toBe("process");
|
||||
|
||||
const issuesRes = await page.request.get(
|
||||
`${baseUrl}/api/companies/${company!.id}/issues`
|
||||
);
|
||||
expect(issuesRes.ok()).toBe(true);
|
||||
const issues = (await issuesRes.json()) as Array<{
|
||||
id: string;
|
||||
title: string;
|
||||
assigneeAgentId: string | null;
|
||||
}>;
|
||||
const issue = issues.find((entry) => entry.title === TASK_TITLE);
|
||||
expect(issue).toBeTruthy();
|
||||
expect(issue!.assigneeAgentId).toBe(ceoAgent!.id);
|
||||
|
||||
await expect.poll(
|
||||
async () => {
|
||||
const runsRes = await page.request.get(
|
||||
`${baseUrl}/api/companies/${company!.id}/heartbeat-runs?agentId=${ceoAgent!.id}`
|
||||
);
|
||||
expect(runsRes.ok()).toBe(true);
|
||||
const runs = (await runsRes.json()) as Array<{
|
||||
agentId: string;
|
||||
invocationSource: string;
|
||||
status: string;
|
||||
}>;
|
||||
const latestRun = runs.find((entry) => entry.agentId === ceoAgent!.id);
|
||||
return latestRun
|
||||
? {
|
||||
invocationSource: latestRun.invocationSource,
|
||||
status: latestRun.status,
|
||||
}
|
||||
: null;
|
||||
},
|
||||
{
|
||||
timeout: 30_000,
|
||||
intervals: [1_000, 2_000, 5_000],
|
||||
}
|
||||
).toEqual(
|
||||
expect.objectContaining({
|
||||
invocationSource: "assignment",
|
||||
status: expect.stringMatching(/^(queued|running|succeeded)$/),
|
||||
})
|
||||
);
|
||||
});
|
||||
});
|
||||
28
tests/release-smoke/playwright.config.ts
Normal file
28
tests/release-smoke/playwright.config.ts
Normal file
@@ -0,0 +1,28 @@
|
||||
import { defineConfig } from "@playwright/test";
|
||||
|
||||
const BASE_URL =
|
||||
process.env.PAPERCLIP_RELEASE_SMOKE_BASE_URL ?? "http://127.0.0.1:3232";
|
||||
|
||||
export default defineConfig({
|
||||
testDir: ".",
|
||||
testMatch: "**/*.spec.ts",
|
||||
timeout: 90_000,
|
||||
expect: {
|
||||
timeout: 15_000,
|
||||
},
|
||||
retries: process.env.CI ? 1 : 0,
|
||||
use: {
|
||||
baseURL: BASE_URL,
|
||||
headless: true,
|
||||
screenshot: "only-on-failure",
|
||||
trace: "retain-on-failure",
|
||||
},
|
||||
projects: [
|
||||
{
|
||||
name: "chromium",
|
||||
use: { browserName: "chromium" },
|
||||
},
|
||||
],
|
||||
outputDir: "./test-results",
|
||||
reporter: [["list"], ["html", { open: "never", outputFolder: "./playwright-report" }]],
|
||||
});
|
||||
Reference in New Issue
Block a user