Compare commits
21 Commits
a23e425e35
...
7e4933ee25
| Author | SHA1 | Date | |
|---|---|---|---|
| 7e4933ee25 | |||
| c4573726f6 | |||
| e1c78c532d | |||
| 6688fbb4de | |||
| 06a817c416 | |||
| ede8c94560 | |||
| f05c0e3aca | |||
| 828987b8f1 | |||
| acca2d100f | |||
| 9085e69907 | |||
| 2e6db75472 | |||
| efa3e27455 | |||
| be7ed687e4 | |||
| d883dc3dc0 | |||
| adadfffd79 | |||
| 47b9eecf9f | |||
| e727933e21 | |||
| c340239003 | |||
| d389a9eafc | |||
| 9b6377c787 | |||
| a15de04895 |
24
Dockerfile
24
Dockerfile
@@ -1,9 +1,31 @@
|
|||||||
FROM node:20-bookworm-slim
|
FROM node:20-bookworm-slim
|
||||||
|
|
||||||
|
RUN apt-get update && apt-get install -y \
|
||||||
|
libglib2.0-0 \
|
||||||
|
libnss3 \
|
||||||
|
libnspr4 \
|
||||||
|
libatk1.0-0 \
|
||||||
|
libatk-bridge2.0-0 \
|
||||||
|
libcups2 \
|
||||||
|
libdrm2 \
|
||||||
|
libdbus-1-3 \
|
||||||
|
libxkbcommon0 \
|
||||||
|
libxcomposite1 \
|
||||||
|
libxdamage1 \
|
||||||
|
libxfixes3 \
|
||||||
|
libxrandr2 \
|
||||||
|
libgbm1 \
|
||||||
|
libpango-1.0-0 \
|
||||||
|
libcairo2 \
|
||||||
|
libasound2 \
|
||||||
|
libatspi2.0-0 \
|
||||||
|
&& rm -rf /var/lib/apt/lists/*
|
||||||
|
|
||||||
ENV PLAYWRIGHT_SKIP_BROWSER_DOWNLOAD=1
|
ENV PLAYWRIGHT_SKIP_BROWSER_DOWNLOAD=1
|
||||||
ENV CHROME_DEBUGGING_PORT=9222
|
ENV CHROME_DEBUGGING_PORT=9222
|
||||||
|
|
||||||
WORKDIR /app
|
WORKDIR /app
|
||||||
COPY package*.json ./
|
COPY package*.json ./
|
||||||
RUN npm install
|
RUN npm install && npm install -g playwright && playwright install chromium
|
||||||
COPY . .
|
COPY . .
|
||||||
CMD ["npx", "playwright", "test"]
|
CMD ["npx", "playwright", "test"]
|
||||||
|
|||||||
@@ -6,6 +6,7 @@
|
|||||||
"test:headed": "playwright test --headed"
|
"test:headed": "playwright test --headed"
|
||||||
},
|
},
|
||||||
"devDependencies": {
|
"devDependencies": {
|
||||||
"@playwright/test": "^1.40.0"
|
"@playwright/test": "^1.40.0",
|
||||||
|
"axios": "^1.6.0"
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|||||||
@@ -1,7 +1,7 @@
|
|||||||
import { defineConfig, devices } from '@playwright/test';
|
import { defineConfig, devices } from '@playwright/test';
|
||||||
|
|
||||||
const baseURL = process.env.BASE_URL || 'http://127.0.0.1:3000';
|
const baseURL = process.env.FRONTEND_URL || 'http://frontend:3000';
|
||||||
const webServerURL = process.env.WEB_SERVER_URL || baseURL;
|
const backendURL = process.env.BACKEND_URL || 'http://backend:8000';
|
||||||
|
|
||||||
export default defineConfig({
|
export default defineConfig({
|
||||||
testDir: './tests',
|
testDir: './tests',
|
||||||
@@ -22,7 +22,7 @@ export default defineConfig({
|
|||||||
],
|
],
|
||||||
webServer: {
|
webServer: {
|
||||||
command: 'npm run dev',
|
command: 'npm run dev',
|
||||||
url: webServerURL,
|
url: baseURL,
|
||||||
reuseExistingServer: !process.env.CI,
|
reuseExistingServer: !process.env.CI,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
|
|||||||
@@ -1,6 +0,0 @@
|
|||||||
import { test, expect } from '@playwright/test';
|
|
||||||
|
|
||||||
test('has title', async ({ page }) => {
|
|
||||||
await page.goto('/');
|
|
||||||
await expect(page).toHaveTitle(/HarborForge/);
|
|
||||||
});
|
|
||||||
48
tests/wizard.spec.ts
Normal file
48
tests/wizard.spec.ts
Normal file
@@ -0,0 +1,48 @@
|
|||||||
|
import { test, expect } from '@playwright/test';
|
||||||
|
|
||||||
|
const FRONTEND_URL = process.env.FRONTEND_URL || 'http://frontend:3000';
|
||||||
|
|
||||||
|
test.describe('Setup Wizard', () => {
|
||||||
|
test('complete wizard flow through frontend', async ({ page }) => {
|
||||||
|
// Navigate to frontend - should redirect to wizard since not configured
|
||||||
|
await page.goto(FRONTEND_URL);
|
||||||
|
await page.waitForLoadState('networkidle');
|
||||||
|
|
||||||
|
// Step 0: Welcome - should see wizard page
|
||||||
|
await expect(page.locator('h1')).toContainText('HarborForge', { timeout: 10000 });
|
||||||
|
|
||||||
|
// Check if we're on wizard page (look for Connect to Wizard button)
|
||||||
|
const connectButton = page.locator('button:has-text("Connect to Wizard")');
|
||||||
|
if (await connectButton.isVisible().catch(() => false)) {
|
||||||
|
// We're on wizard page, proceed with wizard flow
|
||||||
|
await connectButton.click();
|
||||||
|
|
||||||
|
// Wait for step 1: Database
|
||||||
|
await page.waitForSelector('h2:has-text("Database configuration")', { timeout: 10000 });
|
||||||
|
await page.fill('input[name="host"]', 'mysql');
|
||||||
|
await page.fill('input[name="port"]', '3306');
|
||||||
|
await page.fill('input[name="user"]', 'harborforge');
|
||||||
|
await page.fill('input[name="password"]', 'harborforge_pass');
|
||||||
|
await page.fill('input[name="database"]', 'harborforge');
|
||||||
|
await page.click('button:has-text("Next")');
|
||||||
|
|
||||||
|
// Wait for step 2: Admin
|
||||||
|
await page.waitForSelector('h2:has-text("Admin account")', { timeout: 10000 });
|
||||||
|
await page.fill('input[name="password"]', 'admin123');
|
||||||
|
await page.fill('input[name="email"]', 'admin@test.com');
|
||||||
|
await page.fill('input[name="full_name"]', 'Test Admin');
|
||||||
|
await page.click('button:has-text("Next")');
|
||||||
|
|
||||||
|
// Wait for step 3: Backend URL (no project step)
|
||||||
|
await page.waitForSelector('h2:has-text("Backend URL")', { timeout: 10000 });
|
||||||
|
await page.fill('input[name="backend_base_url"]', 'http://backend:8000');
|
||||||
|
await page.click('button:has-text("Finish setup")');
|
||||||
|
|
||||||
|
// Wait for step 4: Complete
|
||||||
|
await expect(page.locator('h2')).toContainText('Setup complete!', { timeout: 10000 });
|
||||||
|
} else {
|
||||||
|
// Wizard was already configured, verify we're on main page
|
||||||
|
await expect(page.locator('h1')).toContainText('HarborForge');
|
||||||
|
}
|
||||||
|
});
|
||||||
|
});
|
||||||
Reference in New Issue
Block a user