import { test, expect } from '@playwright/test'; /** * 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(); }); }); }); });