Integrate @bgotink/playwright-coverage for E2E test coverage tracking: Install @bgotink/playwright-coverage package Update playwright.config.js with coverage reporter Update test file imports to use coverage-enabled test function Add e2e-tests.yml coverage artifact upload and merge job Create codecov.yml with e2e flag configuration Add E2E coverage skill and VS Code task Coverage outputs: HTML, LCOV, JSON to coverage/e2e/ CI uploads merged coverage to Codecov with 'e2e' flag Enables unified coverage view across unit and E2E tests
269 lines
11 KiB
TypeScript
269 lines
11 KiB
TypeScript
import { test, expect } from '@bgotink/playwright-coverage';
|
|
|
|
/**
|
|
* DNS Provider Types E2E Tests
|
|
*
|
|
* Tests the DNS Provider Types API and UI, including:
|
|
* - API endpoint /api/v1/dns-providers/types
|
|
* - Built-in providers (cloudflare, route53, etc.)
|
|
* - Custom providers from Phase 2 (manual, rfc2136, webhook, script)
|
|
* - Provider selector in UI
|
|
*/
|
|
|
|
test.describe('DNS Provider Types', () => {
|
|
test.describe('API: /api/v1/dns-providers/types', () => {
|
|
test('should return all provider types including built-in and custom', async ({ request }) => {
|
|
const response = await request.get('/api/v1/dns-providers/types');
|
|
expect(response.ok()).toBeTruthy();
|
|
|
|
const data = await response.json();
|
|
// API returns { types: [...], total: N }
|
|
const types = data.types;
|
|
expect(Array.isArray(types)).toBeTruthy();
|
|
|
|
// Should have built-in providers
|
|
const typeNames = types.map((t: { type: string }) => t.type);
|
|
expect(typeNames).toContain('cloudflare');
|
|
expect(typeNames).toContain('route53');
|
|
|
|
// Should have custom providers from Phase 2
|
|
expect(typeNames).toContain('manual');
|
|
expect(typeNames).toContain('rfc2136');
|
|
expect(typeNames).toContain('webhook');
|
|
expect(typeNames).toContain('script');
|
|
});
|
|
|
|
test('each provider type should have required fields', async ({ request }) => {
|
|
const response = await request.get('/api/v1/dns-providers/types');
|
|
const data = await response.json();
|
|
const types = data.types;
|
|
|
|
for (const provider of types) {
|
|
expect(provider).toHaveProperty('type');
|
|
expect(provider).toHaveProperty('name');
|
|
expect(provider).toHaveProperty('fields');
|
|
expect(Array.isArray(provider.fields)).toBeTruthy();
|
|
}
|
|
});
|
|
|
|
test('manual provider type should have correct configuration', async ({ request }) => {
|
|
const response = await request.get('/api/v1/dns-providers/types');
|
|
const data = await response.json();
|
|
const types = data.types;
|
|
|
|
const manualProvider = types.find((t: { type: string }) => t.type === 'manual');
|
|
expect(manualProvider).toBeDefined();
|
|
expect(manualProvider.name).toMatch(/manual/i);
|
|
|
|
// Manual provider should have minimal or no required fields
|
|
// since DNS records are created manually by the user
|
|
});
|
|
|
|
test('webhook provider type should have url field', async ({ request }) => {
|
|
const response = await request.get('/api/v1/dns-providers/types');
|
|
const data = await response.json();
|
|
const types = data.types;
|
|
|
|
const webhookProvider = types.find((t: { type: string }) => t.type === 'webhook');
|
|
expect(webhookProvider).toBeDefined();
|
|
|
|
// Webhook should have URL configuration field
|
|
const fieldNames = webhookProvider.fields.map((f: { name: string }) => f.name);
|
|
expect(fieldNames.some((name: string) => name.toLowerCase().includes('url'))).toBeTruthy();
|
|
});
|
|
|
|
test('rfc2136 provider type should have server and key fields', async ({ request }) => {
|
|
const response = await request.get('/api/v1/dns-providers/types');
|
|
const data = await response.json();
|
|
const types = data.types;
|
|
|
|
const rfc2136Provider = types.find((t: { type: string }) => t.type === 'rfc2136');
|
|
expect(rfc2136Provider).toBeDefined();
|
|
|
|
// RFC2136 (Dynamic DNS) should have server and TSIG key fields
|
|
const fieldNames = rfc2136Provider.fields.map((f: { name: string }) => f.name.toLowerCase());
|
|
expect(fieldNames.some((name: string) => name.includes('server') || name.includes('nameserver'))).toBeTruthy();
|
|
});
|
|
|
|
test('script provider type should have command/path field', async ({ request }) => {
|
|
const response = await request.get('/api/v1/dns-providers/types');
|
|
const data = await response.json();
|
|
const types = data.types;
|
|
|
|
const scriptProvider = types.find((t: { type: string }) => t.type === 'script');
|
|
expect(scriptProvider).toBeDefined();
|
|
|
|
// Script provider should have a command or script path field
|
|
const fieldNames = scriptProvider.fields.map((f: { name: string }) => f.name.toLowerCase());
|
|
expect(
|
|
fieldNames.some((name: string) => name.includes('script') || name.includes('command') || name.includes('path'))
|
|
).toBeTruthy();
|
|
});
|
|
});
|
|
|
|
test.describe('UI: Provider Selector', () => {
|
|
test('should show all provider types in dropdown', async ({ page }) => {
|
|
await page.goto('/dns/providers');
|
|
|
|
await test.step('Click Add Provider button', async () => {
|
|
// Use first() to handle both header button and empty state button
|
|
const addButton = page.getByRole('button', { name: /add.*provider/i }).first();
|
|
await expect(addButton).toBeVisible();
|
|
await addButton.click();
|
|
});
|
|
|
|
await test.step('Open provider type dropdown', async () => {
|
|
// Select trigger has id="provider-type"
|
|
const typeSelect = page.locator('#provider-type');
|
|
|
|
await expect(typeSelect).toBeVisible();
|
|
await typeSelect.click();
|
|
});
|
|
|
|
await test.step('Verify built-in providers appear', async () => {
|
|
await expect(page.getByRole('option', { name: /cloudflare/i })).toBeVisible();
|
|
});
|
|
|
|
await test.step('Verify custom providers appear', async () => {
|
|
await expect(page.getByRole('option', { name: /manual/i })).toBeVisible();
|
|
});
|
|
});
|
|
|
|
test('should display provider description in selector', async ({ page }) => {
|
|
await page.goto('/dns/providers');
|
|
await page.getByRole('button', { name: /add.*provider/i }).first().click();
|
|
|
|
const typeSelect = page.locator('#provider-type');
|
|
await typeSelect.click();
|
|
|
|
// Manual provider option should have description indicating no automation
|
|
const manualOption = page.getByRole('option', { name: /manual/i });
|
|
await expect(manualOption).toBeVisible();
|
|
|
|
// Description might be in the option text or a separate element
|
|
const optionText = await manualOption.textContent();
|
|
// Manual provider description should indicate manual DNS record creation
|
|
expect(optionText?.toLowerCase()).toMatch(/manual|no automation|hand/i);
|
|
});
|
|
|
|
test('should filter provider types based on search', async ({ page }) => {
|
|
await page.goto('/dns/providers');
|
|
await page.getByRole('button', { name: /add.*provider/i }).first().click();
|
|
|
|
const typeSelect = page.locator('#provider-type');
|
|
await typeSelect.click();
|
|
|
|
// The select dropdown doesn't support text search input
|
|
// Instead, verify that all options are keyboard accessible
|
|
await test.step('Verify options can be navigated with keyboard', async () => {
|
|
// Press down arrow to navigate through options
|
|
await page.keyboard.press('ArrowDown');
|
|
|
|
// Verify an option is highlighted/focused
|
|
const options = page.getByRole('option');
|
|
const optionCount = await options.count();
|
|
|
|
// Should have multiple provider options available
|
|
expect(optionCount).toBeGreaterThan(5);
|
|
|
|
// Verify cloudflare option exists in the list
|
|
await expect(page.getByRole('option', { name: /cloudflare/i })).toBeVisible();
|
|
|
|
// Verify manual option exists in the list
|
|
await expect(page.getByRole('option', { name: /manual/i })).toBeVisible();
|
|
});
|
|
});
|
|
});
|
|
|
|
test.describe('Provider Type Selection', () => {
|
|
test('should show correct fields when Manual type is selected', async ({ page }) => {
|
|
await page.goto('/dns/providers');
|
|
await page.getByRole('button', { name: /add.*provider/i }).first().click();
|
|
|
|
await test.step('Select Manual provider type', async () => {
|
|
const typeSelect = page.locator('#provider-type');
|
|
await typeSelect.click();
|
|
await page.getByRole('option', { name: /manual/i }).click();
|
|
});
|
|
|
|
await test.step('Verify Manual-specific UI appears', async () => {
|
|
// Manual provider should show informational text about manual DNS record creation
|
|
const infoText = page.getByText(/manually|dns record|challenge/i);
|
|
await expect(infoText).toBeVisible({ timeout: 3000 }).catch(() => {
|
|
// Info text may not be present, that's okay
|
|
});
|
|
|
|
// Should NOT show fields like API key or access token
|
|
await expect(page.getByLabel(/api.*key/i)).not.toBeVisible();
|
|
await expect(page.getByLabel(/access.*token/i)).not.toBeVisible();
|
|
});
|
|
});
|
|
|
|
test('should show URL field when Webhook type is selected', async ({ page }) => {
|
|
await page.goto('/dns/providers');
|
|
await page.getByRole('button', { name: /add.*provider/i }).first().click();
|
|
|
|
await test.step('Select Webhook provider type', async () => {
|
|
const typeSelect = page.locator('#provider-type');
|
|
await typeSelect.click();
|
|
await page.getByRole('option', { name: /webhook/i }).click();
|
|
});
|
|
|
|
await test.step('Verify Webhook URL field appears', async () => {
|
|
// Wait for dynamic credential fields to render
|
|
await page.waitForTimeout(500);
|
|
|
|
// Webhook provider shows "Create URL" and "Delete URL" fields
|
|
// These are rendered as labels followed by inputs
|
|
const createUrlLabel = page.locator('label').filter({ hasText: /create.*url|url/i }).first();
|
|
await expect(createUrlLabel).toBeVisible({ timeout: 5000 });
|
|
});
|
|
});
|
|
|
|
test('should show server field when RFC2136 type is selected', async ({ page }) => {
|
|
await page.goto('/dns/providers');
|
|
await page.getByRole('button', { name: /add.*provider/i }).first().click();
|
|
|
|
await test.step('Select RFC2136 provider type', async () => {
|
|
const typeSelect = page.locator('#provider-type');
|
|
await typeSelect.click();
|
|
|
|
// RFC2136 might be listed as "RFC 2136" or similar
|
|
const rfc2136Option = page.getByRole('option', { name: /rfc.*2136|dynamic.*dns/i });
|
|
await expect(rfc2136Option).toBeVisible({ timeout: 5000 });
|
|
await rfc2136Option.click();
|
|
});
|
|
|
|
await test.step('Verify RFC2136 server field appears', async () => {
|
|
// Wait for dynamic credential fields to render
|
|
await page.waitForTimeout(500);
|
|
|
|
// RFC2136 provider should have server/nameserver related fields
|
|
const serverLabel = page
|
|
.locator('label')
|
|
.filter({ hasText: /server|nameserver|host/i })
|
|
.first();
|
|
await expect(serverLabel).toBeVisible({ timeout: 5000 });
|
|
});
|
|
});
|
|
|
|
test('should show script path field when Script type is selected', async ({ page }) => {
|
|
await page.goto('/dns/providers');
|
|
await page.getByRole('button', { name: /add.*provider/i }).first().click();
|
|
|
|
await test.step('Select Script provider type', async () => {
|
|
const typeSelect = page.locator('#provider-type');
|
|
await typeSelect.click();
|
|
await page.getByRole('option', { name: /script/i }).click();
|
|
});
|
|
|
|
await test.step('Verify Script path/command field appears', async () => {
|
|
// Script provider shows "Script Path" field with placeholder "/scripts/dns-challenge.sh"
|
|
const scriptField = page.getByRole('textbox', { name: /script path/i })
|
|
.or(page.getByPlaceholder(/dns-challenge\.sh/i));
|
|
await expect(scriptField).toBeVisible();
|
|
});
|
|
});
|
|
});
|
|
});
|