@sleep2agi/agent-network-dashboard 0.5.1-preview.9 → 0.5.1-preview.91
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/.next/BUILD_ID +1 -1
- package/.next/build-manifest.json +3 -3
- package/.next/diagnostics/route-bundle-stats.json +6 -6
- package/.next/fallback-build-manifest.json +3 -3
- package/.next/server/app/_global-error.html +1 -1
- package/.next/server/app/_global-error.rsc +1 -1
- package/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
- package/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
- package/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
- package/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
- package/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
- package/.next/server/app/_not-found.html +2 -2
- package/.next/server/app/_not-found.rsc +2 -2
- package/.next/server/app/_not-found.segments/_full.segment.rsc +2 -2
- package/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
- package/.next/server/app/_not-found.segments/_index.segment.rsc +2 -2
- package/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
- package/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/admin/page_client-reference-manifest.js +1 -1
- package/.next/server/app/admin.html +2 -2
- package/.next/server/app/admin.rsc +2 -2
- package/.next/server/app/admin.segments/_full.segment.rsc +2 -2
- package/.next/server/app/admin.segments/_head.segment.rsc +1 -1
- package/.next/server/app/admin.segments/_index.segment.rsc +2 -2
- package/.next/server/app/admin.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/admin.segments/admin/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/admin.segments/admin.segment.rsc +1 -1
- package/.next/server/app/index.html +2 -2
- package/.next/server/app/index.rsc +3 -3
- package/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
- package/.next/server/app/index.segments/_full.segment.rsc +3 -3
- package/.next/server/app/index.segments/_head.segment.rsc +1 -1
- package/.next/server/app/index.segments/_index.segment.rsc +2 -2
- package/.next/server/app/index.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/login/page_client-reference-manifest.js +1 -1
- package/.next/server/app/login.html +2 -2
- package/.next/server/app/login.rsc +3 -3
- package/.next/server/app/login.segments/_full.segment.rsc +3 -3
- package/.next/server/app/login.segments/_head.segment.rsc +1 -1
- package/.next/server/app/login.segments/_index.segment.rsc +2 -2
- package/.next/server/app/login.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/login.segments/login/__PAGE__.segment.rsc +2 -2
- package/.next/server/app/login.segments/login.segment.rsc +1 -1
- package/.next/server/app/logs/page_client-reference-manifest.js +1 -1
- package/.next/server/app/logs.html +2 -2
- package/.next/server/app/logs.rsc +2 -2
- package/.next/server/app/logs.segments/_full.segment.rsc +2 -2
- package/.next/server/app/logs.segments/_head.segment.rsc +1 -1
- package/.next/server/app/logs.segments/_index.segment.rsc +2 -2
- package/.next/server/app/logs.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/logs.segments/logs/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/logs.segments/logs.segment.rsc +1 -1
- package/.next/server/app/messages/page_client-reference-manifest.js +1 -1
- package/.next/server/app/messages.html +2 -2
- package/.next/server/app/messages.rsc +2 -2
- package/.next/server/app/messages.segments/_full.segment.rsc +2 -2
- package/.next/server/app/messages.segments/_head.segment.rsc +1 -1
- package/.next/server/app/messages.segments/_index.segment.rsc +2 -2
- package/.next/server/app/messages.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/messages.segments/messages/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/messages.segments/messages.segment.rsc +1 -1
- package/.next/server/app/node/page_client-reference-manifest.js +1 -1
- package/.next/server/app/node.html +2 -2
- package/.next/server/app/node.rsc +2 -2
- package/.next/server/app/node.segments/_full.segment.rsc +2 -2
- package/.next/server/app/node.segments/_head.segment.rsc +1 -1
- package/.next/server/app/node.segments/_index.segment.rsc +2 -2
- package/.next/server/app/node.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/node.segments/node/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/node.segments/node.segment.rsc +1 -1
- package/.next/server/app/nodes/page_client-reference-manifest.js +1 -1
- package/.next/server/app/nodes.html +2 -2
- package/.next/server/app/nodes.rsc +2 -2
- package/.next/server/app/nodes.segments/_full.segment.rsc +2 -2
- package/.next/server/app/nodes.segments/_head.segment.rsc +1 -1
- package/.next/server/app/nodes.segments/_index.segment.rsc +2 -2
- package/.next/server/app/nodes.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/nodes.segments/nodes/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/nodes.segments/nodes.segment.rsc +1 -1
- package/.next/server/app/page_client-reference-manifest.js +1 -1
- package/.next/server/app/server-logs/page_client-reference-manifest.js +1 -1
- package/.next/server/app/server-logs.html +2 -2
- package/.next/server/app/server-logs.rsc +2 -2
- package/.next/server/app/server-logs.segments/_full.segment.rsc +2 -2
- package/.next/server/app/server-logs.segments/_head.segment.rsc +1 -1
- package/.next/server/app/server-logs.segments/_index.segment.rsc +2 -2
- package/.next/server/app/server-logs.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/server-logs.segments/server-logs/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/server-logs.segments/server-logs.segment.rsc +1 -1
- package/.next/server/app/settings/networks/page_client-reference-manifest.js +1 -1
- package/.next/server/app/settings/networks.html +2 -2
- package/.next/server/app/settings/networks.rsc +2 -2
- package/.next/server/app/settings/networks.segments/_full.segment.rsc +2 -2
- package/.next/server/app/settings/networks.segments/_head.segment.rsc +1 -1
- package/.next/server/app/settings/networks.segments/_index.segment.rsc +2 -2
- package/.next/server/app/settings/networks.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/settings/networks.segments/settings/networks/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/settings/networks.segments/settings/networks.segment.rsc +1 -1
- package/.next/server/app/settings/networks.segments/settings.segment.rsc +1 -1
- package/.next/server/app/settings/page_client-reference-manifest.js +1 -1
- package/.next/server/app/settings/tokens/page_client-reference-manifest.js +1 -1
- package/.next/server/app/settings/tokens.html +2 -2
- package/.next/server/app/settings/tokens.rsc +2 -2
- package/.next/server/app/settings/tokens.segments/_full.segment.rsc +2 -2
- package/.next/server/app/settings/tokens.segments/_head.segment.rsc +1 -1
- package/.next/server/app/settings/tokens.segments/_index.segment.rsc +2 -2
- package/.next/server/app/settings/tokens.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/settings/tokens.segments/settings/tokens/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/settings/tokens.segments/settings/tokens.segment.rsc +1 -1
- package/.next/server/app/settings/tokens.segments/settings.segment.rsc +1 -1
- package/.next/server/app/settings.html +2 -2
- package/.next/server/app/settings.rsc +3 -3
- package/.next/server/app/settings.segments/_full.segment.rsc +3 -3
- package/.next/server/app/settings.segments/_head.segment.rsc +1 -1
- package/.next/server/app/settings.segments/_index.segment.rsc +2 -2
- package/.next/server/app/settings.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/settings.segments/settings/__PAGE__.segment.rsc +2 -2
- package/.next/server/app/settings.segments/settings.segment.rsc +1 -1
- package/.next/server/app/tasks/[id]/page_client-reference-manifest.js +1 -1
- package/.next/server/app/tasks/page_client-reference-manifest.js +1 -1
- package/.next/server/app/tasks.html +2 -2
- package/.next/server/app/tasks.rsc +2 -2
- package/.next/server/app/tasks.segments/_full.segment.rsc +2 -2
- package/.next/server/app/tasks.segments/_head.segment.rsc +1 -1
- package/.next/server/app/tasks.segments/_index.segment.rsc +2 -2
- package/.next/server/app/tasks.segments/_tree.segment.rsc +2 -2
- package/.next/server/app/tasks.segments/tasks/__PAGE__.segment.rsc +1 -1
- package/.next/server/app/tasks.segments/tasks.segment.rsc +1 -1
- package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js +3 -3
- package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js.map +1 -1
- package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js +1 -1
- package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js.map +1 -1
- package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js +1 -1
- package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js.map +1 -1
- package/.next/server/middleware-build-manifest.js +3 -3
- package/.next/server/pages/404.html +2 -2
- package/.next/server/pages/500.html +1 -1
- package/.next/static/chunks/017hq2-5l~_98.css +2 -0
- package/.next/static/chunks/07t9_p5h7da1u.js +1 -0
- package/.next/static/chunks/0ahff_xzbgbg4.js +4 -0
- package/.next/static/chunks/10qa7z9iocn6t.js +1 -0
- package/.next/trace +2 -2
- package/.next/trace-build +1 -1
- package/app/components/TopoGraph.tsx +1777 -111
- package/package.json +1 -1
- package/scripts/topo-active-links-chip-hover-lift-test.mjs +93 -0
- package/scripts/topo-chip-digit-fontweight-test.mjs +105 -0
- package/scripts/topo-chip-digit-hover-bold-test.mjs +94 -0
- package/scripts/topo-chip-row-group-hover-brighten-test.mjs +107 -0
- package/scripts/topo-chip-row-hover-lift-test.mjs +95 -0
- package/scripts/topo-chrome-button-hover-lift-test.mjs +94 -0
- package/scripts/topo-chrome-segmented-radius-test.mjs +100 -0
- package/scripts/topo-click-ripple-opacity-test.mjs +99 -0
- package/scripts/topo-edge-badge-digit-fw-test.mjs +103 -0
- package/scripts/topo-edge-badge-fontsize-test.mjs +90 -0
- package/scripts/topo-edge-badge-hover-opacity-test.mjs +94 -0
- package/scripts/topo-edge-badge-hover-stroke-test.mjs +92 -0
- package/scripts/topo-edge-badge-opacity-test.mjs +80 -0
- package/scripts/topo-edge-badge-pin-opacity-test.mjs +86 -0
- package/scripts/topo-edge-badge-stroke-test.mjs +92 -0
- package/scripts/topo-edge-freshness-floor-test.mjs +99 -0
- package/scripts/topo-edge-particle-radius-test.mjs +76 -0
- package/scripts/topo-edge-visible-linecap-test.mjs +89 -0
- package/scripts/topo-filter-pill-hover-lift-test.mjs +101 -0
- package/scripts/topo-filter-pill-hover-opacity-test.mjs +110 -0
- package/scripts/topo-filter-pill-value-fw-test.mjs +88 -0
- package/scripts/topo-filter-pill-x-hover-scale-test.mjs +99 -0
- package/scripts/topo-flow-rail-linecap-test.mjs +79 -0
- package/scripts/topo-freshness-chip-hierarchy-test.mjs +93 -0
- package/scripts/topo-freshness-chip-tabular-test.mjs +41 -0
- package/scripts/topo-freshness-floor-lift-test.mjs +92 -0
- package/scripts/topo-freshness-suffix-tabular-test.mjs +88 -0
- package/scripts/topo-fullscreen-icon-hover-scale-test.mjs +91 -0
- package/scripts/topo-group-box-stroke-test.mjs +105 -0
- package/scripts/topo-group-label-count-fontweight-test.mjs +108 -0
- package/scripts/topo-hover-detail-body-fw-test.mjs +101 -0
- package/scripts/topo-hover-detail-model-fw-test.mjs +98 -0
- package/scripts/topo-hover-detail-opacity-test.mjs +98 -0
- package/scripts/topo-hover-detail-rx-test.mjs +81 -0
- package/scripts/topo-hub-digit-fontsize-test.mjs +86 -0
- package/scripts/topo-hub-digit-fw-hover-test.mjs +102 -0
- package/scripts/topo-hub-halo-light-trough-test.mjs +88 -0
- package/scripts/topo-hub-halo-radius-test.mjs +86 -0
- package/scripts/topo-hub-halo-trough-test.mjs +83 -0
- package/scripts/topo-hub-highlight-opacity-test.mjs +88 -0
- package/scripts/topo-hub-highlight-radius-test.mjs +90 -0
- package/scripts/topo-hub-hover-ring-opacity-test.mjs +96 -0
- package/scripts/topo-hub-hover-ring-stroke-test.mjs +86 -0
- package/scripts/topo-hub-spoke-hover-opacity-test.mjs +119 -0
- package/scripts/topo-hub-spoke-linecap-test.mjs +80 -0
- package/scripts/topo-label-card-opacity-hover-test.mjs +99 -0
- package/scripts/topo-layout-toggle-hover-tracking-test.mjs +109 -0
- package/scripts/topo-layout-toggle-radius-test.mjs +87 -0
- package/scripts/topo-legend-label-fontweight-test.mjs +94 -0
- package/scripts/topo-legend-pin-ring-stroke-test.mjs +101 -0
- package/scripts/topo-minimap-offline-opacity-test.mjs +90 -0
- package/scripts/topo-minimap-online-hover-opacity-test.mjs +92 -0
- package/scripts/topo-minimap-online-opacity-test.mjs +93 -0
- package/scripts/topo-minimap-online-radius-test.mjs +85 -0
- package/scripts/topo-minimap-viewport-linejoin-test.mjs +75 -0
- package/scripts/topo-minimap-viewport-rx-test.mjs +85 -0
- package/scripts/topo-more-flows-fontweight-test.mjs +103 -0
- package/scripts/topo-node-alias-letter-spacing-test.mjs +112 -0
- package/scripts/topo-node-halo-offline-opacity-test.mjs +87 -0
- package/scripts/topo-node-label-card-rx-test.mjs +85 -0
- package/scripts/topo-node-pulse-peak-test.mjs +89 -0
- package/scripts/topo-node-pulse-trough-test.mjs +91 -0
- package/scripts/topo-node-sub-text-letter-spacing-test.mjs +115 -0
- package/scripts/topo-panel-count-fw-hover-test.mjs +105 -0
- package/scripts/topo-panel-count-letterspacing-test.mjs +89 -0
- package/scripts/topo-panel-stroke-hover-test.mjs +110 -0
- package/scripts/topo-pressure-bar-height-test.mjs +92 -0
- package/scripts/topo-pressure-kicker-fontweight-test.mjs +76 -0
- package/scripts/topo-recent-pip-radius-2-test.mjs +72 -0
- package/scripts/topo-recent-pip-radius-test.mjs +76 -0
- package/scripts/topo-recent-row-content-opacity-test.mjs +81 -0
- package/scripts/topo-recent-row-text-fontweight-test.mjs +90 -0
- package/scripts/topo-reset-hover-rotate-test.mjs +102 -0
- package/scripts/topo-spoke-active-opacity-test.mjs +104 -0
- package/scripts/topo-spoke-active-stroke-test.mjs +95 -0
- package/scripts/topo-spoke-idle-opacity-test.mjs +91 -0
- package/scripts/topo-vendor-chip-hover-lift-test.mjs +87 -0
- package/scripts/topo-vendor-glyph-fontweight-test.mjs +102 -0
- package/scripts/topo-vendor-letter-hover-scale-test.mjs +129 -0
- package/scripts/topo-vendor-suffix-hover-brighten-test.mjs +87 -0
- package/scripts/topo-zoom-icon-hover-scale-test.mjs +114 -0
- package/scripts/topo-zoom-level-hover-fw-test.mjs +95 -0
- package/.next/static/chunks/08fc_cz1nk7b9.js +0 -1
- package/.next/static/chunks/0aauz~36q5n2a.css +0 -2
- package/.next/static/chunks/0e0okm.affulg.js +0 -1
- package/.next/static/chunks/0s3vtwfo26_t6.js +0 -4
- /package/.next/static/{egukPz1ctU--4WnT2FpEU → i0drwZtW8h-M1ML2C5VZF}/_buildManifest.js +0 -0
- /package/.next/static/{egukPz1ctU--4WnT2FpEU → i0drwZtW8h-M1ML2C5VZF}/_clientMiddlewareManifest.js +0 -0
- /package/.next/static/{egukPz1ctU--4WnT2FpEU → i0drwZtW8h-M1ML2C5VZF}/_ssgManifest.js +0 -0
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
/* Round 411 verification: node label card rx 6 → 8. Corner-radius
|
|
2
|
+
* cascade family 8th anchor — compact card tier aligns with the
|
|
3
|
+
* R332/R375/R376 compact-chrome rx=8 tier.
|
|
4
|
+
*
|
|
5
|
+
* Contract:
|
|
6
|
+
* - Each node's label-card <rect> rx attr === '8'
|
|
7
|
+
* - data-node-label-card-rx === '8'
|
|
8
|
+
* - Pre-R411 invariants on the rect:
|
|
9
|
+
* * cardW (width) > 0
|
|
10
|
+
* * cardH (height) > 0
|
|
11
|
+
* * fill, stroke present
|
|
12
|
+
*/
|
|
13
|
+
import { chromium } from 'playwright';
|
|
14
|
+
import { readFileSync } from 'node:fs';
|
|
15
|
+
|
|
16
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
17
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
18
|
+
|
|
19
|
+
const browser = await chromium.launch({ headless: true });
|
|
20
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
21
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
22
|
+
await ctx.addInitScript(() => {
|
|
23
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
24
|
+
});
|
|
25
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
26
|
+
const r = await route.fetch();
|
|
27
|
+
const b = await r.json();
|
|
28
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
29
|
+
const mk = (alias) => ({
|
|
30
|
+
alias, status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
31
|
+
network_id: nid, project_dir: null,
|
|
32
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
33
|
+
});
|
|
34
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [ mk('alpha'), mk('beta'), mk('gamma') ] } });
|
|
35
|
+
});
|
|
36
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
|
|
37
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
38
|
+
|
|
39
|
+
const page = await ctx.newPage();
|
|
40
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
41
|
+
await page.waitForSelector('[data-node-label-card]', { timeout: 15000 });
|
|
42
|
+
await page.waitForTimeout(400);
|
|
43
|
+
|
|
44
|
+
const probe = await page.evaluate(() => {
|
|
45
|
+
const cards = Array.from(document.querySelectorAll('[data-node-label-card]'));
|
|
46
|
+
if (cards.length === 0) return { count: 0 };
|
|
47
|
+
const sample = cards[0];
|
|
48
|
+
return {
|
|
49
|
+
count: cards.length,
|
|
50
|
+
rxAttr: sample.getAttribute('rx'),
|
|
51
|
+
rxData: sample.getAttribute('data-node-label-card-rx'),
|
|
52
|
+
widthAttr: sample.getAttribute('width'),
|
|
53
|
+
heightAttr: sample.getAttribute('height'),
|
|
54
|
+
fillPresent: !!sample.getAttribute('fill'),
|
|
55
|
+
strokePresent: !!sample.getAttribute('stroke'),
|
|
56
|
+
allRx8: cards.every((c) => c.getAttribute('rx') === '8'),
|
|
57
|
+
};
|
|
58
|
+
});
|
|
59
|
+
|
|
60
|
+
const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
|
|
61
|
+
const sourceRx8 = /rx="8"\s*\n\s+fill=\{pal\.labelBox\.fill\}/.test(fileText) || /width=\{cardW\} height=\{cardH\} rx="8"/.test(fileText);
|
|
62
|
+
const sourceDataRx8 = /data-node-label-card-rx="8"/.test(fileText);
|
|
63
|
+
|
|
64
|
+
await browser.close();
|
|
65
|
+
|
|
66
|
+
const results = {
|
|
67
|
+
// At least one label card mounted
|
|
68
|
+
cards_count_ge_1: (probe.count || 0) >= 1,
|
|
69
|
+
// R411 wire
|
|
70
|
+
rx_attr_8: probe.rxAttr === '8',
|
|
71
|
+
rx_data_8: probe.rxData === '8',
|
|
72
|
+
all_cards_rx_8: probe.allRx8 === true,
|
|
73
|
+
// Pre-R411 invariants
|
|
74
|
+
width_present: Number(probe.widthAttr) > 0,
|
|
75
|
+
height_present: Number(probe.heightAttr) > 0,
|
|
76
|
+
fill_present: probe.fillPresent === true,
|
|
77
|
+
stroke_present: probe.strokePresent === true,
|
|
78
|
+
// Source-file canonical wire
|
|
79
|
+
source_rx_8: sourceRx8,
|
|
80
|
+
source_data_rx_8: sourceDataRx8,
|
|
81
|
+
};
|
|
82
|
+
const ok = Object.values(results).every(Boolean);
|
|
83
|
+
console.log(`${ok ? '✅' : '❌'} node label card rx 6 → 8:`, JSON.stringify(results),
|
|
84
|
+
'\n probe:', probe);
|
|
85
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
/* Round 409 verification: active-node pulse SMIL peak opacity lift
|
|
2
|
+
* — cyber 0.18 → 0.20 / light 0.12 → 0.14. Active-state breath
|
|
3
|
+
* widens slightly at peak (+0.02 both themes) while trough stays.
|
|
4
|
+
*
|
|
5
|
+
* Contract:
|
|
6
|
+
* - Seed a flow link so a node becomes "active" (R243 isActive gate)
|
|
7
|
+
* - The active node's <g data-node-pulse> child has an <animate>
|
|
8
|
+
* opacity element with:
|
|
9
|
+
* * cyber: values='0.20;0.04;0.20' + data-node-pulse-peak='0.20'
|
|
10
|
+
* * light: values='0.14;0.02;0.14' + data-node-pulse-peak='0.14'
|
|
11
|
+
* - Pre-R409 invariants preserved:
|
|
12
|
+
* * dur='2.4s', calcMode='spline', repeatCount='indefinite'
|
|
13
|
+
* * keySplines='0.42 0 0.58 1;0.42 0 0.58 1'
|
|
14
|
+
*/
|
|
15
|
+
import { chromium } from 'playwright';
|
|
16
|
+
import { readFileSync } from 'node:fs';
|
|
17
|
+
|
|
18
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
19
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
20
|
+
|
|
21
|
+
async function probeTheme(theme) {
|
|
22
|
+
const browser = await chromium.launch({ headless: true });
|
|
23
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
24
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
25
|
+
await ctx.addInitScript((t) => {
|
|
26
|
+
try { localStorage.setItem('anet-theme', t); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
27
|
+
}, theme);
|
|
28
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
29
|
+
const r = await route.fetch();
|
|
30
|
+
const b = await r.json();
|
|
31
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
32
|
+
const mk = (alias) => ({
|
|
33
|
+
alias, status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
34
|
+
network_id: nid, project_dir: null,
|
|
35
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
36
|
+
});
|
|
37
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [ mk('alpha'), mk('beta'), mk('gamma') ] } });
|
|
38
|
+
});
|
|
39
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({
|
|
40
|
+
json: { messages: [
|
|
41
|
+
{ id: 'm1', from_alias: 'alpha', to_alias: 'beta', content: 'ping', created_at: fresh, network_id: 'default' },
|
|
42
|
+
] },
|
|
43
|
+
}));
|
|
44
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
45
|
+
const page = await ctx.newPage();
|
|
46
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
47
|
+
await page.waitForSelector('[data-node-pulse-active="true"]', { timeout: 15000 });
|
|
48
|
+
await page.waitForTimeout(500);
|
|
49
|
+
const probe = await page.evaluate(() => {
|
|
50
|
+
const pulseGroup = document.querySelector('[data-node-pulse-active="true"]');
|
|
51
|
+
if (!pulseGroup) return null;
|
|
52
|
+
const animates = pulseGroup.querySelectorAll('animate');
|
|
53
|
+
const opacityAnim = Array.from(animates).find(a => a.getAttribute('attributeName') === 'opacity');
|
|
54
|
+
return opacityAnim ? {
|
|
55
|
+
attributeName: opacityAnim.getAttribute('attributeName'),
|
|
56
|
+
values: opacityAnim.getAttribute('values'),
|
|
57
|
+
dur: opacityAnim.getAttribute('dur'),
|
|
58
|
+
calcMode: opacityAnim.getAttribute('calcMode'),
|
|
59
|
+
repeatCount: opacityAnim.getAttribute('repeatCount'),
|
|
60
|
+
keySplines: opacityAnim.getAttribute('keySplines'),
|
|
61
|
+
peakData: opacityAnim.getAttribute('data-node-pulse-peak'),
|
|
62
|
+
} : null;
|
|
63
|
+
});
|
|
64
|
+
await browser.close();
|
|
65
|
+
return probe;
|
|
66
|
+
}
|
|
67
|
+
|
|
68
|
+
const cyber = await probeTheme('cyber');
|
|
69
|
+
const light = await probeTheme('light');
|
|
70
|
+
|
|
71
|
+
const results = {
|
|
72
|
+
// R409: peak lifts to 0.20 cyber / 0.14 light
|
|
73
|
+
cyber_values_0_20: cyber?.values === '0.20;0.04;0.20',
|
|
74
|
+
cyber_peak_data_0_20: cyber?.peakData === '0.20',
|
|
75
|
+
light_values_0_14: light?.values === '0.14;0.02;0.14',
|
|
76
|
+
light_peak_data_0_14: light?.peakData === '0.14',
|
|
77
|
+
// Pre-R409 invariants
|
|
78
|
+
cyber_dur_2_4s: cyber?.dur === '2.4s',
|
|
79
|
+
cyber_spline: cyber?.calcMode === 'spline',
|
|
80
|
+
cyber_repeat_indefinite: cyber?.repeatCount === 'indefinite',
|
|
81
|
+
cyber_keySplines: cyber?.keySplines === '0.42 0 0.58 1;0.42 0 0.58 1',
|
|
82
|
+
light_dur_2_4s: light?.dur === '2.4s',
|
|
83
|
+
light_keySplines: light?.keySplines === '0.42 0 0.58 1;0.42 0 0.58 1',
|
|
84
|
+
};
|
|
85
|
+
const ok = Object.values(results).every(Boolean);
|
|
86
|
+
console.log(`${ok ? '✅' : '❌'} active-node pulse peak lift:`, JSON.stringify(results),
|
|
87
|
+
'\n cyber:', cyber,
|
|
88
|
+
'\n light:', light);
|
|
89
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,91 @@
|
|
|
1
|
+
/* Round 413 verification: active-node pulse SMIL trough opacity
|
|
2
|
+
* lift — cyber 0.04 → 0.05 / light 0.02 → 0.03. Extends the
|
|
3
|
+
* stale-state legibility lift family (8th anchor) and pairs with
|
|
4
|
+
* R409 peak lift so the per-node "active flow" breath reads
|
|
5
|
+
* confidently at both ends of its cycle.
|
|
6
|
+
*
|
|
7
|
+
* Contract:
|
|
8
|
+
* - Seed a flow link so a node becomes "active" (R243 isActive gate)
|
|
9
|
+
* - The active node's <animate> opacity element:
|
|
10
|
+
* * cyber: values='0.20;0.05;0.20' + data-node-pulse-trough='0.05'
|
|
11
|
+
* * light: values='0.14;0.03;0.14' + data-node-pulse-trough='0.03'
|
|
12
|
+
* - R409 peak invariant: data-node-pulse-peak='0.20' / '0.14'
|
|
13
|
+
* - Pre-R413 invariants preserved:
|
|
14
|
+
* * dur='2.4s', calcMode='spline', repeatCount='indefinite'
|
|
15
|
+
* * keySplines='0.42 0 0.58 1;0.42 0 0.58 1'
|
|
16
|
+
*/
|
|
17
|
+
import { chromium } from 'playwright';
|
|
18
|
+
import { readFileSync } from 'node:fs';
|
|
19
|
+
|
|
20
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
21
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
22
|
+
|
|
23
|
+
async function probeTheme(theme) {
|
|
24
|
+
const browser = await chromium.launch({ headless: true });
|
|
25
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
26
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
27
|
+
await ctx.addInitScript((t) => {
|
|
28
|
+
try { localStorage.setItem('anet-theme', t); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
29
|
+
}, theme);
|
|
30
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
31
|
+
const r = await route.fetch();
|
|
32
|
+
const b = await r.json();
|
|
33
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
34
|
+
const mk = (alias) => ({
|
|
35
|
+
alias, status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
36
|
+
network_id: nid, project_dir: null,
|
|
37
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
38
|
+
});
|
|
39
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [ mk('alpha'), mk('beta'), mk('gamma') ] } });
|
|
40
|
+
});
|
|
41
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({
|
|
42
|
+
json: { messages: [
|
|
43
|
+
{ id: 'm1', from_alias: 'alpha', to_alias: 'beta', content: 'ping', created_at: fresh, network_id: 'default' },
|
|
44
|
+
] },
|
|
45
|
+
}));
|
|
46
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
47
|
+
const page = await ctx.newPage();
|
|
48
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
49
|
+
await page.waitForSelector('[data-node-pulse-active="true"]', { timeout: 15000 });
|
|
50
|
+
await page.waitForTimeout(500);
|
|
51
|
+
const probe = await page.evaluate(() => {
|
|
52
|
+
const pulseGroup = document.querySelector('[data-node-pulse-active="true"]');
|
|
53
|
+
if (!pulseGroup) return null;
|
|
54
|
+
const animates = pulseGroup.querySelectorAll('animate');
|
|
55
|
+
const opacityAnim = Array.from(animates).find(a => a.getAttribute('attributeName') === 'opacity');
|
|
56
|
+
return opacityAnim ? {
|
|
57
|
+
values: opacityAnim.getAttribute('values'),
|
|
58
|
+
peakData: opacityAnim.getAttribute('data-node-pulse-peak'),
|
|
59
|
+
troughData: opacityAnim.getAttribute('data-node-pulse-trough'),
|
|
60
|
+
dur: opacityAnim.getAttribute('dur'),
|
|
61
|
+
calcMode: opacityAnim.getAttribute('calcMode'),
|
|
62
|
+
keySplines: opacityAnim.getAttribute('keySplines'),
|
|
63
|
+
} : null;
|
|
64
|
+
});
|
|
65
|
+
await browser.close();
|
|
66
|
+
return probe;
|
|
67
|
+
}
|
|
68
|
+
|
|
69
|
+
const cyber = await probeTheme('cyber');
|
|
70
|
+
const light = await probeTheme('light');
|
|
71
|
+
|
|
72
|
+
const results = {
|
|
73
|
+
// R413: trough lifts to 0.05 cyber / 0.03 light
|
|
74
|
+
cyber_values_0_05: cyber?.values === '0.20;0.05;0.20',
|
|
75
|
+
cyber_trough_data_0_05: cyber?.troughData === '0.05',
|
|
76
|
+
light_values_0_03: light?.values === '0.14;0.03;0.14',
|
|
77
|
+
light_trough_data_0_03: light?.troughData === '0.03',
|
|
78
|
+
// R409 peak invariants
|
|
79
|
+
cyber_peak_data_0_20: cyber?.peakData === '0.20',
|
|
80
|
+
light_peak_data_0_14: light?.peakData === '0.14',
|
|
81
|
+
// Pre-R413 invariants preserved
|
|
82
|
+
cyber_dur_2_4s: cyber?.dur === '2.4s',
|
|
83
|
+
cyber_spline: cyber?.calcMode === 'spline',
|
|
84
|
+
cyber_keySplines: cyber?.keySplines === '0.42 0 0.58 1;0.42 0 0.58 1',
|
|
85
|
+
light_dur_2_4s: light?.dur === '2.4s',
|
|
86
|
+
};
|
|
87
|
+
const ok = Object.values(results).every(Boolean);
|
|
88
|
+
console.log(`${ok ? '✅' : '❌'} active-node pulse trough lift:`, JSON.stringify(results),
|
|
89
|
+
'\n cyber:', cyber,
|
|
90
|
+
'\n light:', light);
|
|
91
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,115 @@
|
|
|
1
|
+
/* Round 428 verification: node sub-text (status label line beneath
|
|
2
|
+
* the alias) adopts hover letter-spacing tween 0 → 0.2px on
|
|
3
|
+
* hoveredAlias. Sibling to R427 alias-text 0 → 0.3 — the alias is
|
|
4
|
+
* primary identity (0.3 kerning), sub-text is secondary status (0.2).
|
|
5
|
+
*
|
|
6
|
+
* Contract:
|
|
7
|
+
* - rest: every sub-text has letter-spacing '0px' and hover='false'
|
|
8
|
+
* - hover one node: that sub-text has letter-spacing '0.2px' and
|
|
9
|
+
* hover='true'; siblings stay at rest
|
|
10
|
+
* - alias-text and sub-text tween together on the same hovered node
|
|
11
|
+
* (R427 0.3 + R428 0.2)
|
|
12
|
+
* - source-file probe confirms conditional + transition extended
|
|
13
|
+
*/
|
|
14
|
+
import { chromium } from 'playwright';
|
|
15
|
+
import { readFileSync } from 'node:fs';
|
|
16
|
+
|
|
17
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
18
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
19
|
+
|
|
20
|
+
const browser = await chromium.launch({ headless: true });
|
|
21
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
22
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
23
|
+
await ctx.addInitScript(() => {
|
|
24
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
25
|
+
});
|
|
26
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
27
|
+
const r = await route.fetch();
|
|
28
|
+
const b = await r.json();
|
|
29
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
30
|
+
const mk = (alias, status) => ({
|
|
31
|
+
alias, status, model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
32
|
+
network_id: nid, project_dir: null,
|
|
33
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
34
|
+
});
|
|
35
|
+
// small fleet — card-mode labels
|
|
36
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [
|
|
37
|
+
mk('alpha', 'working'),
|
|
38
|
+
mk('beta', 'idle'),
|
|
39
|
+
mk('gamma', 'working'),
|
|
40
|
+
] } });
|
|
41
|
+
});
|
|
42
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
|
|
43
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
44
|
+
|
|
45
|
+
const page = await ctx.newPage();
|
|
46
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
47
|
+
await page.waitForSelector('[data-node-sub-text]', { timeout: 15000 });
|
|
48
|
+
await page.waitForTimeout(400);
|
|
49
|
+
|
|
50
|
+
const readAll = () => page.evaluate(() => {
|
|
51
|
+
const subs = [...document.querySelectorAll('[data-node-sub-text]')];
|
|
52
|
+
const alis = [...document.querySelectorAll('[data-node-alias-text]')];
|
|
53
|
+
return {
|
|
54
|
+
sub: subs.map(t => ({
|
|
55
|
+
alias: t.getAttribute('data-node-sub-text'),
|
|
56
|
+
ls: t.style.letterSpacing,
|
|
57
|
+
hover: t.getAttribute('data-node-sub-text-hovered'),
|
|
58
|
+
})),
|
|
59
|
+
alias: alis.map(t => ({
|
|
60
|
+
alias: t.getAttribute('data-node-alias-text'),
|
|
61
|
+
ls: t.style.letterSpacing,
|
|
62
|
+
hover: t.getAttribute('data-node-alias-hovered'),
|
|
63
|
+
})),
|
|
64
|
+
};
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
const rest = await readAll();
|
|
68
|
+
const firstAlias = rest.sub[0]?.alias;
|
|
69
|
+
let hover = null;
|
|
70
|
+
if (firstAlias) {
|
|
71
|
+
const box = await page.evaluate((alias) => {
|
|
72
|
+
const t = document.querySelector(`[data-node-alias-text="${alias}"]`);
|
|
73
|
+
if (!t) return null;
|
|
74
|
+
const node = t.closest('[data-node]');
|
|
75
|
+
const target = node || t;
|
|
76
|
+
const b = target.getBoundingClientRect();
|
|
77
|
+
return { x: b.x + b.width / 2, y: b.y + b.height / 2 };
|
|
78
|
+
}, firstAlias);
|
|
79
|
+
if (box) {
|
|
80
|
+
await page.mouse.move(box.x, box.y);
|
|
81
|
+
await page.waitForTimeout(300);
|
|
82
|
+
hover = await readAll();
|
|
83
|
+
await page.mouse.move(0, 0);
|
|
84
|
+
}
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
// Source-file probe
|
|
88
|
+
const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
|
|
89
|
+
const sourceWired = /data-node-sub-text-hovered.*\n.*letterSpacing:\s*hoveredAlias\s*===\s*session\.alias\s*\?\s*'0\.2px'\s*:\s*'0px'/s.test(fileText);
|
|
90
|
+
const sourceTransition = /transition: 'fill 300ms ease-out, letter-spacing 200ms ease-out'/.test(fileText);
|
|
91
|
+
|
|
92
|
+
await browser.close();
|
|
93
|
+
|
|
94
|
+
const isRest = (ls) => ls === '0px' || ls === '' || ls === 'normal';
|
|
95
|
+
const hoveredSubEntry = hover?.sub.find(r => r.alias === firstAlias);
|
|
96
|
+
const hoveredAliasEntry = hover?.alias.find(r => r.alias === firstAlias);
|
|
97
|
+
const othersSubRest = hover ? hover.sub.filter(r => r.alias !== firstAlias).every(r => isRest(r.ls)) : false;
|
|
98
|
+
|
|
99
|
+
const results = {
|
|
100
|
+
rest_sub_count_gte_3: rest.sub.length >= 3,
|
|
101
|
+
rest_sub_all_zero: rest.sub.every(r => isRest(r.ls)),
|
|
102
|
+
rest_sub_all_hover_false: rest.sub.every(r => r.hover === 'false'),
|
|
103
|
+
hover_sub_target_ls_0_2: hoveredSubEntry?.ls === '0.2px',
|
|
104
|
+
hover_sub_target_attr_true: hoveredSubEntry?.hover === 'true',
|
|
105
|
+
hover_sub_others_stay_rest: othersSubRest,
|
|
106
|
+
// R427/R428 sibling parity — alias and sub-text tween together on same hovered node
|
|
107
|
+
hover_alias_target_ls_0_3: hoveredAliasEntry?.ls === '0.3px',
|
|
108
|
+
source_conditional_wired: sourceWired,
|
|
109
|
+
source_transition_wired: sourceTransition,
|
|
110
|
+
};
|
|
111
|
+
const ok = Object.values(results).every(Boolean);
|
|
112
|
+
console.log(`${ok ? '✅' : '❌'} node sub-text letter-spacing hover:`, JSON.stringify(results),
|
|
113
|
+
'\n hover sub:', JSON.stringify(hoveredSubEntry),
|
|
114
|
+
'\n hover alias:', JSON.stringify(hoveredAliasEntry));
|
|
115
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,105 @@
|
|
|
1
|
+
/* Round 424 verification: panel header count digit fontWeight 600 →
|
|
2
|
+
* 700 on panel hover. Closes the 5-layer panel hover cue stack with
|
|
3
|
+
* a typographic-weight axis at the panel-header data scope:
|
|
4
|
+
* R135 drop-shadow (depth)
|
|
5
|
+
* R348 fill opacity (solidity)
|
|
6
|
+
* R345 title letter-spacing (spacing)
|
|
7
|
+
* R423 stroke→legendAccent (color/edge)
|
|
8
|
+
* R424 count digit fw 600→700 (weight)
|
|
9
|
+
*
|
|
10
|
+
* Contract:
|
|
11
|
+
* - rest state: data-recent-panel-count + data-legend-panel-count
|
|
12
|
+
* tspans report computed font-weight matching rest branch (600)
|
|
13
|
+
* - geometry / R311 / R310 invariants preserved (digit text reads,
|
|
14
|
+
* R348 rest opacity '0.92' still applies to outer rect)
|
|
15
|
+
* - source-file probe verifies both panels are wired with
|
|
16
|
+
* `hoveredPanel === '<key>' ? '700' : '600'` conditional
|
|
17
|
+
* - panel hover dispatch on the panel <g> is unreliable
|
|
18
|
+
* (React's onMouseEnter non-bubbling synthetic — same R423
|
|
19
|
+
* finding); source-file verification is the canonical contract
|
|
20
|
+
* for the hover branch, same pattern R348/R394/R412/R423 used.
|
|
21
|
+
*/
|
|
22
|
+
import { chromium } from 'playwright';
|
|
23
|
+
import { readFileSync } from 'node:fs';
|
|
24
|
+
|
|
25
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
26
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
27
|
+
|
|
28
|
+
const browser = await chromium.launch({ headless: true });
|
|
29
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
30
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
31
|
+
await ctx.addInitScript(() => {
|
|
32
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
33
|
+
});
|
|
34
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
35
|
+
const r = await route.fetch();
|
|
36
|
+
const b = await r.json();
|
|
37
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
38
|
+
const mk = (alias, status) => ({
|
|
39
|
+
alias, status, model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
40
|
+
network_id: nid, project_dir: null,
|
|
41
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
42
|
+
});
|
|
43
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [
|
|
44
|
+
mk('alpha', 'working'),
|
|
45
|
+
mk('beta', 'idle'),
|
|
46
|
+
] } });
|
|
47
|
+
});
|
|
48
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({
|
|
49
|
+
json: { messages: [
|
|
50
|
+
{ id: 'm1', from_alias: 'alpha', to_alias: 'beta', content: 'ping', created_at: fresh, network_id: 'default' },
|
|
51
|
+
] },
|
|
52
|
+
}));
|
|
53
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
54
|
+
|
|
55
|
+
const page = await ctx.newPage();
|
|
56
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
57
|
+
await page.waitForSelector('[data-recent-panel-count], [data-legend-panel-count]', { timeout: 15000 });
|
|
58
|
+
await page.waitForTimeout(400);
|
|
59
|
+
|
|
60
|
+
const rest = await page.evaluate(() => {
|
|
61
|
+
const r = document.querySelector('[data-recent-panel-count]');
|
|
62
|
+
const l = document.querySelector('[data-legend-panel-count]');
|
|
63
|
+
return {
|
|
64
|
+
recent: r ? {
|
|
65
|
+
fw_attr: r.getAttribute('font-weight'),
|
|
66
|
+
fw_computed: getComputedStyle(r).fontWeight,
|
|
67
|
+
text: r.textContent,
|
|
68
|
+
} : null,
|
|
69
|
+
legend: l ? {
|
|
70
|
+
fw_attr: l.getAttribute('font-weight'),
|
|
71
|
+
fw_computed: getComputedStyle(l).fontWeight,
|
|
72
|
+
text: l.textContent,
|
|
73
|
+
} : null,
|
|
74
|
+
};
|
|
75
|
+
});
|
|
76
|
+
|
|
77
|
+
// Source-file verification — runtime hover dispatch on the panel <g>
|
|
78
|
+
// is unreliable (R423 finding). Same source-file probe pattern.
|
|
79
|
+
const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
|
|
80
|
+
const recentSourceWired = /fontWeight=\{hoveredPanel === 'recent' \? '700' : '600'\}/.test(fileText);
|
|
81
|
+
const legendSourceWired = /fontWeight=\{hoveredPanel === 'legend' \? '700' : '600'\}/.test(fileText);
|
|
82
|
+
// R247 transition list extended to include font-weight on both
|
|
83
|
+
const recentTransitionWired = /transition: 'fill 200ms ease-out, font-weight 200ms ease-out'/.test(fileText);
|
|
84
|
+
|
|
85
|
+
await browser.close();
|
|
86
|
+
|
|
87
|
+
const results = {
|
|
88
|
+
// mount / textContent invariants
|
|
89
|
+
recent_mounted: !!rest.recent,
|
|
90
|
+
legend_mounted: !!rest.legend,
|
|
91
|
+
recent_has_flows: /flows/.test(rest.recent?.text || ''),
|
|
92
|
+
legend_has_nodes: /node/.test(rest.legend?.text || ''),
|
|
93
|
+
// rest font-weight resolves to base 600 (string '600' from attr or computed)
|
|
94
|
+
recent_rest_fw_600: rest.recent?.fw_attr === '600' || rest.recent?.fw_computed === '600',
|
|
95
|
+
legend_rest_fw_600: rest.legend?.fw_attr === '600' || rest.legend?.fw_computed === '600',
|
|
96
|
+
// source-file: both panels wired with conditional fw branch
|
|
97
|
+
source_recent_hover_wired: recentSourceWired,
|
|
98
|
+
source_legend_hover_wired: legendSourceWired,
|
|
99
|
+
// transition list extended to font-weight
|
|
100
|
+
source_transition_wired: recentTransitionWired,
|
|
101
|
+
};
|
|
102
|
+
const ok = Object.values(results).every(Boolean);
|
|
103
|
+
console.log(`${ok ? '✅' : '❌'} panel count digit fw hover:`, JSON.stringify(results),
|
|
104
|
+
'\n rest:', rest);
|
|
105
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,89 @@
|
|
|
1
|
+
/* Round 349 verification: panel header counts pick up editorial
|
|
2
|
+
* letter-spacing 0.2, one tier below the R301 panel title 0.3.
|
|
3
|
+
* Closes the panel-pair editorial symmetry — both top-corner panels
|
|
4
|
+
* now have a 2-step header hierarchy:
|
|
5
|
+
* title fontSize=12 fontWeight=700 letterSpacing=0.3 (R301)
|
|
6
|
+
* count fontSize=10 fontWeight=600 letterSpacing=0.2 (R349, this round)
|
|
7
|
+
* Joins the R285 / R289 / R301 / R302 / R304 / R325 editorial-
|
|
8
|
+
* letterspacing tier at the panel-summary scope. The R162 freshness
|
|
9
|
+
* fill, R225 tabular-nums, R311 fontWeight, R336 unit-tspan opacity-0.7
|
|
10
|
+
* split all preserved (SVG inheritance propagates the tier to descendant
|
|
11
|
+
* tspans).
|
|
12
|
+
*
|
|
13
|
+
* Contract:
|
|
14
|
+
* - Recent-signal panel header count <text> letter-spacing="0.2".
|
|
15
|
+
* - Legend panel header count <text> letter-spacing="0.2".
|
|
16
|
+
* - data-{recent,legend}-panel-count-letter-spacing="0.2" attrs present.
|
|
17
|
+
* - Pre-R349 invariants: R311 fontWeight=600, R225 tabular-nums,
|
|
18
|
+
* R336 unit-tspan opacity=0.7 all preserved.
|
|
19
|
+
*/
|
|
20
|
+
import { chromium } from 'playwright';
|
|
21
|
+
import { readFileSync } from 'node:fs';
|
|
22
|
+
|
|
23
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
24
|
+
const browser = await chromium.launch({ headless: true });
|
|
25
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
26
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
27
|
+
await ctx.addInitScript(() => {
|
|
28
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
29
|
+
});
|
|
30
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
31
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
32
|
+
const r = await route.fetch();
|
|
33
|
+
const b = await r.json();
|
|
34
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
35
|
+
const mk = (alias) => ({
|
|
36
|
+
alias, status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
37
|
+
network_id: nid, project_dir: null,
|
|
38
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
39
|
+
});
|
|
40
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [ mk('a'), mk('b'), mk('c') ] } });
|
|
41
|
+
});
|
|
42
|
+
// One message → recent panel mounts.
|
|
43
|
+
const now = Date.now();
|
|
44
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [
|
|
45
|
+
{ id: 'm1', from_alias: 'a', to_alias: 'b', content: 'ping',
|
|
46
|
+
network_id: 'default', created_at: new Date(now - 5000).toISOString() },
|
|
47
|
+
] } }));
|
|
48
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
49
|
+
|
|
50
|
+
const page = await ctx.newPage();
|
|
51
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
52
|
+
await page.waitForSelector('[data-recent-panel-count]', { timeout: 15000 });
|
|
53
|
+
await page.waitForSelector('[data-legend-panel-count]', { timeout: 5000 });
|
|
54
|
+
await page.waitForTimeout(300);
|
|
55
|
+
|
|
56
|
+
const probe = await page.evaluate(() => {
|
|
57
|
+
const rTspan = document.querySelector('[data-recent-panel-count]');
|
|
58
|
+
const rText = rTspan?.closest('text');
|
|
59
|
+
const rUnit = document.querySelector('[data-recent-panel-count-unit]');
|
|
60
|
+
const lText = document.querySelector('[data-legend-panel-count]');
|
|
61
|
+
const lUnit = document.querySelector('[data-legend-panel-count-unit]');
|
|
62
|
+
return {
|
|
63
|
+
recentTextLs: rText?.getAttribute('letter-spacing') ?? null,
|
|
64
|
+
recentLsAttr: rText?.getAttribute('data-recent-panel-count-letter-spacing') ?? null,
|
|
65
|
+
recentFw: rTspan?.getAttribute('font-weight') ?? null,
|
|
66
|
+
recentUnitOp: rUnit?.getAttribute('opacity') ?? null,
|
|
67
|
+
legendTextLs: lText?.getAttribute('letter-spacing') ?? null,
|
|
68
|
+
legendLsAttr: lText?.getAttribute('data-legend-panel-count-letter-spacing') ?? null,
|
|
69
|
+
legendFw: lText?.getAttribute('font-weight') ?? null,
|
|
70
|
+
legendUnitOp: lUnit?.getAttribute('opacity') ?? null,
|
|
71
|
+
};
|
|
72
|
+
});
|
|
73
|
+
|
|
74
|
+
await browser.close();
|
|
75
|
+
|
|
76
|
+
const results = {
|
|
77
|
+
recent_text_ls_0_2: probe.recentTextLs === '0.2',
|
|
78
|
+
recent_attr_0_2: probe.recentLsAttr === '0.2',
|
|
79
|
+
recent_fw_600: probe.recentFw === '600', // R311
|
|
80
|
+
recent_unit_op_0_7: probe.recentUnitOp === '0.7', // R336
|
|
81
|
+
legend_text_ls_0_2: probe.legendTextLs === '0.2',
|
|
82
|
+
legend_attr_0_2: probe.legendLsAttr === '0.2',
|
|
83
|
+
legend_fw_600: probe.legendFw === '600', // R310
|
|
84
|
+
legend_unit_op_0_7: probe.legendUnitOp === '0.7', // R336
|
|
85
|
+
};
|
|
86
|
+
const ok = Object.values(results).every(Boolean);
|
|
87
|
+
console.log(`${ok ? '✅' : '❌'} panel count letter-spacing 0.2:`, JSON.stringify(results),
|
|
88
|
+
'\n probe:', probe);
|
|
89
|
+
process.exit(ok ? 0 : 1);
|