Files
Charon/tests/dns-provider-types.spec.ts
GitHub Actions 154c43145d chore: add Playwright E2E coverage with Codecov integration
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
2026-01-20 06:11:59 +00:00

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();
});
});
});
});