mirror of
https://github.com/AutoMaker-Org/automaker.git
synced 2026-01-31 20:03:37 +00:00
- Added `dev:test` script to package.json for streamlined testing without file watching. - Introduced `kill-test-servers` script to ensure no existing servers are running on test ports before executing tests. - Enhanced Playwright configuration to use mock agent for tests, ensuring consistent API responses and disabling rate limiting. - Updated various test files to include authentication steps and handle login screens, improving reliability and reducing flakiness in tests. - Added `global-setup` for e2e tests to ensure proper initialization before test execution.
233 lines
8.1 KiB
TypeScript
233 lines
8.1 KiB
TypeScript
import { Page } from '@playwright/test';
|
|
import { clickElement } from '../core/interactions';
|
|
import { waitForElement } from '../core/waiting';
|
|
import { authenticateForTests } from '../api/client';
|
|
|
|
/**
|
|
* Navigate to the board/kanban view
|
|
* Note: Navigates directly to /board since index route shows WelcomeView
|
|
*/
|
|
export async function navigateToBoard(page: Page): Promise<void> {
|
|
// Authenticate before navigating
|
|
await authenticateForTests(page);
|
|
|
|
// Navigate directly to /board route
|
|
await page.goto('/board');
|
|
await page.waitForLoadState('load');
|
|
|
|
// Check if we're on the login screen and handle it
|
|
const loginInput = page
|
|
.locator('[data-testid="login-api-key-input"], input[type="password"][placeholder*="API key"]')
|
|
.first();
|
|
const isLoginScreen = await loginInput.isVisible({ timeout: 2000 }).catch(() => false);
|
|
if (isLoginScreen) {
|
|
const apiKey = process.env.AUTOMAKER_API_KEY || 'test-api-key-for-e2e-tests';
|
|
await loginInput.fill(apiKey);
|
|
await page.waitForTimeout(100);
|
|
await page
|
|
.locator('[data-testid="login-submit-button"], button:has-text("Login")')
|
|
.first()
|
|
.click();
|
|
await page.waitForURL('**/board', { timeout: 5000 });
|
|
await page.waitForLoadState('load');
|
|
}
|
|
|
|
// Wait for the board view to be visible
|
|
await waitForElement(page, 'board-view', { timeout: 10000 });
|
|
}
|
|
|
|
/**
|
|
* Navigate to the context view
|
|
* Note: Navigates directly to /context since index route shows WelcomeView
|
|
*/
|
|
export async function navigateToContext(page: Page): Promise<void> {
|
|
// Authenticate before navigating
|
|
await authenticateForTests(page);
|
|
|
|
// Navigate directly to /context route
|
|
await page.goto('/context');
|
|
await page.waitForLoadState('load');
|
|
|
|
// Check if we're on the login screen and handle it
|
|
const loginInputCtx = page
|
|
.locator('[data-testid="login-api-key-input"], input[type="password"][placeholder*="API key"]')
|
|
.first();
|
|
const isLoginScreenCtx = await loginInputCtx.isVisible({ timeout: 2000 }).catch(() => false);
|
|
if (isLoginScreenCtx) {
|
|
const apiKey = process.env.AUTOMAKER_API_KEY || 'test-api-key-for-e2e-tests';
|
|
await loginInputCtx.fill(apiKey);
|
|
await page.waitForTimeout(100);
|
|
await page
|
|
.locator('[data-testid="login-submit-button"], button:has-text("Login")')
|
|
.first()
|
|
.click();
|
|
await page.waitForURL('**/context', { timeout: 5000 });
|
|
await page.waitForLoadState('load');
|
|
}
|
|
|
|
// Wait for loading to complete (if present)
|
|
const loadingElement = page.locator('[data-testid="context-view-loading"]');
|
|
try {
|
|
const loadingVisible = await loadingElement.isVisible({ timeout: 2000 });
|
|
if (loadingVisible) {
|
|
// Wait for loading to disappear (context view will appear)
|
|
await loadingElement.waitFor({ state: 'hidden', timeout: 10000 });
|
|
}
|
|
} catch {
|
|
// Loading element not found or already hidden, continue
|
|
}
|
|
|
|
// Wait for the context view to be visible
|
|
// Increase timeout to handle slower server startup
|
|
await waitForElement(page, 'context-view', { timeout: 15000 });
|
|
}
|
|
|
|
/**
|
|
* Navigate to the spec view
|
|
* Note: Navigates directly to /spec since index route shows WelcomeView
|
|
*/
|
|
export async function navigateToSpec(page: Page): Promise<void> {
|
|
// Authenticate before navigating
|
|
await authenticateForTests(page);
|
|
|
|
// Navigate directly to /spec route
|
|
await page.goto('/spec');
|
|
await page.waitForLoadState('load');
|
|
|
|
// Wait for loading state to complete first (if present)
|
|
const loadingElement = page.locator('[data-testid="spec-view-loading"]');
|
|
try {
|
|
const loadingVisible = await loadingElement.isVisible({ timeout: 2000 });
|
|
if (loadingVisible) {
|
|
// Wait for loading to disappear (spec view or empty state will appear)
|
|
await loadingElement.waitFor({ state: 'hidden', timeout: 10000 });
|
|
}
|
|
} catch {
|
|
// Loading element not found or already hidden, continue
|
|
}
|
|
|
|
// Wait for either the main spec view or empty state to be visible
|
|
// The spec-view element appears when loading is complete and spec exists
|
|
// The spec-view-empty element appears when loading is complete and spec doesn't exist
|
|
await Promise.race([
|
|
waitForElement(page, 'spec-view', { timeout: 10000 }).catch(() => null),
|
|
waitForElement(page, 'spec-view-empty', { timeout: 10000 }).catch(() => null),
|
|
]);
|
|
}
|
|
|
|
/**
|
|
* Navigate to the agent view
|
|
* Note: Navigates directly to /agent since index route shows WelcomeView
|
|
*/
|
|
export async function navigateToAgent(page: Page): Promise<void> {
|
|
// Authenticate before navigating
|
|
await authenticateForTests(page);
|
|
|
|
// Navigate directly to /agent route
|
|
await page.goto('/agent');
|
|
await page.waitForLoadState('load');
|
|
|
|
// Check if we're on the login screen and handle it
|
|
const loginInputAgent = page
|
|
.locator('[data-testid="login-api-key-input"], input[type="password"][placeholder*="API key"]')
|
|
.first();
|
|
const isLoginScreenAgent = await loginInputAgent.isVisible({ timeout: 2000 }).catch(() => false);
|
|
if (isLoginScreenAgent) {
|
|
const apiKey = process.env.AUTOMAKER_API_KEY || 'test-api-key-for-e2e-tests';
|
|
await loginInputAgent.fill(apiKey);
|
|
await page.waitForTimeout(100);
|
|
await page
|
|
.locator('[data-testid="login-submit-button"], button:has-text("Login")')
|
|
.first()
|
|
.click();
|
|
await page.waitForURL('**/agent', { timeout: 5000 });
|
|
await page.waitForLoadState('load');
|
|
}
|
|
|
|
// Wait for the agent view to be visible
|
|
await waitForElement(page, 'agent-view', { timeout: 10000 });
|
|
}
|
|
|
|
/**
|
|
* Navigate to the settings view
|
|
* Note: Navigates directly to /settings since index route shows WelcomeView
|
|
*/
|
|
export async function navigateToSettings(page: Page): Promise<void> {
|
|
// Authenticate before navigating
|
|
await authenticateForTests(page);
|
|
|
|
// Navigate directly to /settings route
|
|
await page.goto('/settings');
|
|
await page.waitForLoadState('load');
|
|
|
|
// Wait for the settings view to be visible
|
|
await waitForElement(page, 'settings-view', { timeout: 10000 });
|
|
}
|
|
|
|
/**
|
|
* Navigate to the setup view directly
|
|
* Note: This function uses setupFirstRun from project/setup to avoid circular dependency
|
|
*/
|
|
export async function navigateToSetup(page: Page): Promise<void> {
|
|
// Dynamic import to avoid circular dependency
|
|
const { setupFirstRun } = await import('../project/setup');
|
|
await setupFirstRun(page);
|
|
await page.goto('/');
|
|
await page.waitForLoadState('load');
|
|
await waitForElement(page, 'setup-view', { timeout: 10000 });
|
|
}
|
|
|
|
/**
|
|
* Navigate to the welcome view (clear project selection)
|
|
*/
|
|
export async function navigateToWelcome(page: Page): Promise<void> {
|
|
// Authenticate before navigating
|
|
await authenticateForTests(page);
|
|
|
|
await page.goto('/');
|
|
await page.waitForLoadState('load');
|
|
|
|
// Check if we're on the login screen and handle it
|
|
const loginInputWelcome = page
|
|
.locator('[data-testid="login-api-key-input"], input[type="password"][placeholder*="API key"]')
|
|
.first();
|
|
const isLoginScreenWelcome = await loginInputWelcome
|
|
.isVisible({ timeout: 2000 })
|
|
.catch(() => false);
|
|
if (isLoginScreenWelcome) {
|
|
const apiKey = process.env.AUTOMAKER_API_KEY || 'test-api-key-for-e2e-tests';
|
|
await loginInputWelcome.fill(apiKey);
|
|
await page.waitForTimeout(100);
|
|
await page
|
|
.locator('[data-testid="login-submit-button"], button:has-text("Login")')
|
|
.first()
|
|
.click();
|
|
await page.waitForURL('**/', { timeout: 5000 });
|
|
await page.waitForLoadState('load');
|
|
}
|
|
|
|
await waitForElement(page, 'welcome-view', { timeout: 10000 });
|
|
}
|
|
|
|
/**
|
|
* Navigate to a specific view using the sidebar navigation
|
|
*/
|
|
export async function navigateToView(page: Page, viewId: string): Promise<void> {
|
|
const navSelector = viewId === 'settings' ? 'settings-button' : `nav-${viewId}`;
|
|
await clickElement(page, navSelector);
|
|
await page.waitForTimeout(100);
|
|
}
|
|
|
|
/**
|
|
* Get the current view from the URL or store (checks which view is active)
|
|
*/
|
|
export async function getCurrentView(page: Page): Promise<string | null> {
|
|
// Get the current view from zustand store via localStorage
|
|
const storage = await page.evaluate(() => {
|
|
const item = localStorage.getItem('automaker-storage');
|
|
return item ? JSON.parse(item) : null;
|
|
});
|
|
|
|
return storage?.state?.currentView || null;
|
|
}
|