mirror of
https://github.com/Kpa-clawbot/meshcore-analyzer.git
synced 2026-05-22 10:25:13 +00:00
5cb9b9e732
## Summary Adds Playwright E2E coverage for `public/home.js` and `public/path-inspector.js` per the umbrella issue #1297 B5 page-modules batch. Both files were flagged in the 2026-05-19 frontend coverage audit as page modules with only 1 E2E mention — well below the >=50% statement coverage target. ## Files added - `test-home-coverage-e2e.js` — 12 steps exercising: - first-time chooser → `showChooser` + `setLevel` - experienced-user render → `renderHome` + `loadStats` - search → suggestions → claim → `setupSearch` + `addMyNode` - My Mesh card render + click → `loadHealth` detail - card remove → localStorage cleared - level toggle → checklist accordion expand - `test-path-inspector-coverage-e2e.js` — 10 steps exercising: - page chrome (input/submit/help text) - all 4 validation branches (empty, non-hex, odd-length, mixed lengths) - Enter-key submit + URL `?prefixes=` replacement - valid prefixes → results/no-results render - candidate row toggle + Show on Map → `#/map` hand-off - deep-link `?prefixes=2c` auto-fill + auto-submit Both wired into `.github/workflows/deploy.yml` after the #1279 entries. ## Why the existing `test-path-inspector-e2e.js` is not enough The existing file uses the `@playwright/test` runner (`npx playwright test …`). CI's `e2e-test` step runs every coverage test as `node test-*-e2e.js` directly — the `@playwright/test`-style file is never invoked by CI and contributes zero to the frontend coverage roll-up. ## TDD note Per AGENTS.md exemption: pure coverage tests on existing UI surfaces, no production code modified (`git diff origin/master --stat` shows only the two new test files plus the workflow wiring). Zero behavior change → no red-then-green commit required. ## Verified - Both tests pass locally against a fresh Go server backed by `test-fixtures/e2e-fixture.db` (12/12 + 10/10). - Preflight (`bash ~/.openclaw/skills/pr-preflight/scripts/run-all.sh origin/master`): all gates clean. Refs #1297 Co-authored-by: iavor-bot <bot@corescope>
198 lines
8.9 KiB
JavaScript
198 lines
8.9 KiB
JavaScript
#!/usr/bin/env node
|
|
/* Coverage E2E for public/path-inspector.js (#1297 B5).
|
|
*
|
|
* The existing test-path-inspector-e2e.js uses @playwright/test runner
|
|
* which is not wired into CI's `e2e-test` step (CI runs raw
|
|
* `node test-…-e2e.js`). This file uses the plain chromium-launch
|
|
* pattern compatible with CI and exercises the standalone tools page
|
|
* surface end-to-end:
|
|
*
|
|
* - navigate to /#/tools/path-inspector
|
|
* - assert page chrome renders (input, submit btn, help text)
|
|
* - validation paths: empty input → error; mixed prefix lengths →
|
|
* error; non-hex input → error
|
|
* - valid prefixes (1-byte) → submit → API round-trip completes,
|
|
* URL gets ?prefixes=… appended, results table OR no-results
|
|
* state renders
|
|
* - if a candidate exists, expand its evidence row by clicking the
|
|
* non-button cells and exercise "Show on Map" (asserts route
|
|
* hand-off via window._pendingPathInspectorRoute / nav to #/map)
|
|
* - deep-link auto-fill: /#/tools/path-inspector?prefixes=2c
|
|
* auto-runs and renders the input value
|
|
*
|
|
* Target: lift public/path-inspector.js coverage >= 50% by exercising
|
|
* init/parsePrefixes/validatePrefixes/submit/renderResults/showOnMap
|
|
* branches.
|
|
*
|
|
* Usage: BASE_URL=http://localhost:13581 node test-path-inspector-coverage-e2e.js
|
|
*/
|
|
'use strict';
|
|
const { chromium } = require('playwright');
|
|
|
|
const BASE = process.env.BASE_URL || 'http://localhost:13581';
|
|
|
|
let passed = 0, failed = 0;
|
|
async function step(name, fn) {
|
|
try { await fn(); passed++; console.log(' \u2713 ' + name); }
|
|
catch (e) { failed++; console.error(' \u2717 ' + name + ': ' + e.message); }
|
|
}
|
|
function assert(c, m) { if (!c) throw new Error(m || 'assertion failed'); }
|
|
|
|
async function goPI(page, qs) {
|
|
const url = BASE + '/#/tools/path-inspector' + (qs ? '?' + qs : '');
|
|
await page.goto(url, { waitUntil: 'domcontentloaded' });
|
|
await page.waitForSelector('.path-inspector-page', { timeout: 8000 });
|
|
}
|
|
|
|
(async () => {
|
|
const requireChromium = process.env.CHROMIUM_REQUIRE === '1';
|
|
let browser;
|
|
try {
|
|
browser = await chromium.launch({
|
|
headless: true,
|
|
executablePath: process.env.CHROMIUM_PATH || undefined,
|
|
args: ['--no-sandbox', '--disable-gpu', '--disable-dev-shm-usage'],
|
|
});
|
|
} catch (err) {
|
|
if (requireChromium) {
|
|
console.error('test-path-inspector-coverage-e2e.js: FAIL — Chromium required but unavailable: ' + err.message);
|
|
process.exit(1);
|
|
}
|
|
console.log('test-path-inspector-coverage-e2e.js: SKIP (Chromium unavailable: ' + err.message.split('\n')[0] + ')');
|
|
process.exit(0);
|
|
}
|
|
|
|
const ctx = await browser.newContext({ viewport: { width: 1400, height: 900 } });
|
|
const page = await ctx.newPage();
|
|
page.setDefaultTimeout(15000);
|
|
page.on('pageerror', (e) => console.error('[pageerror]', e.message));
|
|
|
|
console.log('\n=== path-inspector.js coverage E2E against ' + BASE + ' ===');
|
|
|
|
// ── 1. Page chrome ──
|
|
await step('page renders input, submit button, and help text', async () => {
|
|
await goPI(page);
|
|
const input = await page.$('#path-inspector-input');
|
|
const btn = await page.$('#path-inspector-submit');
|
|
assert(input, '#path-inspector-input missing');
|
|
assert(btn, '#path-inspector-submit missing');
|
|
const helpHasCode = await page.$('.help-text code');
|
|
assert(helpHasCode, 'help-text <code> example missing');
|
|
});
|
|
|
|
// ── 2. Validation branches ──
|
|
await step('empty input shows "Enter at least one prefix." error', async () => {
|
|
await goPI(page);
|
|
await page.click('#path-inspector-submit');
|
|
const err = (await page.textContent('#path-inspector-error') || '').trim();
|
|
assert(/at least one prefix/i.test(err), 'expected validation error, got: ' + err);
|
|
});
|
|
|
|
await step('non-hex input shows "Invalid hex" error', async () => {
|
|
await goPI(page);
|
|
await page.fill('#path-inspector-input', 'zz');
|
|
await page.click('#path-inspector-submit');
|
|
const err = (await page.textContent('#path-inspector-error') || '').trim();
|
|
assert(/invalid hex/i.test(err), 'expected "Invalid hex" error, got: ' + err);
|
|
});
|
|
|
|
await step('odd-length prefix shows error', async () => {
|
|
await goPI(page);
|
|
await page.fill('#path-inspector-input', 'abc');
|
|
await page.click('#path-inspector-submit');
|
|
const err = (await page.textContent('#path-inspector-error') || '').trim();
|
|
assert(/odd-length/i.test(err), 'expected odd-length error, got: ' + err);
|
|
});
|
|
|
|
await step('mixed prefix lengths show error', async () => {
|
|
await goPI(page);
|
|
await page.fill('#path-inspector-input', '2c,aabb');
|
|
await page.click('#path-inspector-submit');
|
|
const err = (await page.textContent('#path-inspector-error') || '').trim();
|
|
assert(/mixed/i.test(err), 'expected mixed-length error, got: ' + err);
|
|
});
|
|
|
|
// ── 3. Enter-key submit path ──
|
|
await step('Enter key in input triggers submit (URL gets ?prefixes=…)', async () => {
|
|
await goPI(page);
|
|
await page.fill('#path-inspector-input', '2c');
|
|
await page.press('#path-inspector-input', 'Enter');
|
|
// The history.replaceState appends ?prefixes=2c to the hash.
|
|
await page.waitForFunction(() => location.hash.includes('prefixes='), { timeout: 4000 });
|
|
const hash = await page.evaluate(() => location.hash);
|
|
assert(hash.includes('prefixes=2c'), 'expected hash to contain prefixes=2c, got: ' + hash);
|
|
});
|
|
|
|
// ── 4. Submit produces a result (table or no-results) ──
|
|
await step('valid 1-byte prefixes → results table OR no-results renders', async () => {
|
|
await goPI(page);
|
|
await page.fill('#path-inspector-input', '2c,a1');
|
|
await page.click('#path-inspector-submit');
|
|
await page.waitForFunction(() => {
|
|
const r = document.getElementById('path-inspector-results');
|
|
const e = document.getElementById('path-inspector-error');
|
|
return (r && (r.querySelector('.path-inspector-table') || r.querySelector('.no-results'))) ||
|
|
(e && e.textContent.trim().length > 0);
|
|
}, { timeout: 8000 });
|
|
// If error path, surface it so the test fails informatively.
|
|
const err = (await page.textContent('#path-inspector-error') || '').trim();
|
|
if (err) assert(false, 'unexpected error from valid input: ' + err);
|
|
});
|
|
|
|
// ── 5. Candidate interactions (only if table rendered) ──
|
|
await step('if candidates returned: clicking row toggles evidence row', async () => {
|
|
await goPI(page);
|
|
await page.fill('#path-inspector-input', '2c,a1');
|
|
await page.click('#path-inspector-submit');
|
|
await page.waitForSelector('.path-inspector-table, .no-results', { timeout: 6000 });
|
|
const hasTable = await page.$('.path-inspector-table');
|
|
if (!hasTable) return; // No candidates in fixture; that's still coverage of renderResults() empty branch.
|
|
// Click on the # cell of the first candidate row (NOT the Show on Map button).
|
|
const firstRow = await page.$('.path-inspector-table tbody tr:not(.evidence-row)');
|
|
if (firstRow) {
|
|
// Click the first <td> (the index cell) to avoid the Show on Map button.
|
|
await firstRow.evaluate((row) => {
|
|
const td = row.querySelector('td');
|
|
if (td) td.click();
|
|
});
|
|
await page.waitForTimeout(200);
|
|
// Evidence row should toggle .collapsed off (or stay off if already shown).
|
|
const evidenceRow = await page.$('.evidence-row');
|
|
assert(evidenceRow, 'evidence row should exist when candidates render');
|
|
}
|
|
});
|
|
|
|
await step('"Show on Map" hands off to map page (hash → #/map)', async () => {
|
|
await goPI(page);
|
|
await page.fill('#path-inspector-input', '2c,a1');
|
|
await page.click('#path-inspector-submit');
|
|
await page.waitForSelector('.path-inspector-table, .no-results', { timeout: 6000 });
|
|
const hasTable = await page.$('.path-inspector-table');
|
|
if (!hasTable) return;
|
|
const btn = await page.$('.path-inspector-table button[data-idx]');
|
|
if (!btn) return;
|
|
await btn.click();
|
|
await page.waitForFunction(() => location.hash.startsWith('#/map'), { timeout: 4000 });
|
|
const hash = await page.evaluate(() => location.hash);
|
|
assert(hash.startsWith('#/map'), 'expected nav to #/map, got: ' + hash);
|
|
});
|
|
|
|
// ── 6. Deep-link auto-fill ──
|
|
await step('deep link ?prefixes=2c auto-fills the input', async () => {
|
|
await goPI(page, 'prefixes=2c');
|
|
const val = await page.inputValue('#path-inspector-input');
|
|
assert(val === '2c', 'expected input prefilled with "2c", got: ' + val);
|
|
// And auto-submit kicks off a request that produces results or no-results.
|
|
await page.waitForFunction(() => {
|
|
const r = document.getElementById('path-inspector-results');
|
|
const e = document.getElementById('path-inspector-error');
|
|
return (r && (r.querySelector('.path-inspector-table') || r.querySelector('.no-results'))) ||
|
|
(e && e.textContent.trim().length > 0);
|
|
}, { timeout: 8000 });
|
|
});
|
|
|
|
await browser.close();
|
|
console.log('\n--- ' + passed + ' passed, ' + failed + ' failed ---\n');
|
|
process.exit(failed > 0 ? 1 : 0);
|
|
})().catch((e) => { console.error(e); process.exit(1); });
|