@sleep2agi/agent-network-dashboard 0.5.1-preview.9 → 0.5.1-preview.90
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/09el212cmtr70.js +1 -0
- package/.next/static/chunks/0apaxz3c96keo.js +4 -0
- package/.next/static/chunks/188mrp1elgpea.js +1 -0
- package/.next/trace +2 -2
- package/.next/trace-build +1 -1
- package/app/components/TopoGraph.tsx +1745 -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-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 → mQHthzMGmjydHu598yl-Z}/_buildManifest.js +0 -0
- /package/.next/static/{egukPz1ctU--4WnT2FpEU → mQHthzMGmjydHu598yl-Z}/_clientMiddlewareManifest.js +0 -0
- /package/.next/static/{egukPz1ctU--4WnT2FpEU → mQHthzMGmjydHu598yl-Z}/_ssgManifest.js +0 -0
|
@@ -0,0 +1,85 @@
|
|
|
1
|
+
/* Round 393 verification: minimap viewport rect rx 0 → 2.
|
|
2
|
+
* Corner-radius cascade family adds a sub-element tier (7th
|
|
3
|
+
* anchor) by softening the cyan-stroked viewport frame inside
|
|
4
|
+
* the R332 rounded minimap container. Small but visible polish
|
|
5
|
+
* — 2-px radius reads as "not sharp" without feeling pillowy on
|
|
6
|
+
* the 30-50px wide viewport rect.
|
|
7
|
+
*
|
|
8
|
+
* Contract:
|
|
9
|
+
* - Minimap only mounts on non-default view → zoom-in twice
|
|
10
|
+
* - viewport rect rx attr === '2'
|
|
11
|
+
* - data-topo-minimap-viewport-rx === '2'
|
|
12
|
+
* - Pre-R393 invariants preserved:
|
|
13
|
+
* * R346 hover-state strokeWidth tween (rest '1.5' / hover '1.75')
|
|
14
|
+
* * R346 opacity tween (rest '0.9' / hover '1')
|
|
15
|
+
* * R379 strokeLinejoin='round'
|
|
16
|
+
* * fill='none' + stroke=pal.legendAccent
|
|
17
|
+
*/
|
|
18
|
+
import { chromium } from 'playwright';
|
|
19
|
+
import { readFileSync } from 'node:fs';
|
|
20
|
+
|
|
21
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
22
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
23
|
+
|
|
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
|
+
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({ json: { messages: [] } }));
|
|
42
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
43
|
+
|
|
44
|
+
const page = await ctx.newPage();
|
|
45
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
46
|
+
await page.waitForSelector('g[data-node]', { timeout: 15000 });
|
|
47
|
+
await page.waitForTimeout(300);
|
|
48
|
+
|
|
49
|
+
await page.click('[data-topo-chrome-zoom-in]');
|
|
50
|
+
await page.click('[data-topo-chrome-zoom-in]');
|
|
51
|
+
await page.waitForSelector('[data-topo-minimap-viewport]', { timeout: 5000 });
|
|
52
|
+
await page.waitForTimeout(400);
|
|
53
|
+
|
|
54
|
+
const probe = await page.evaluate(() => {
|
|
55
|
+
const v = document.querySelector('[data-topo-minimap-viewport]');
|
|
56
|
+
if (!v) return null;
|
|
57
|
+
return {
|
|
58
|
+
rxAttr: v.getAttribute('rx'),
|
|
59
|
+
rxData: v.getAttribute('data-topo-minimap-viewport-rx'),
|
|
60
|
+
strokeWidth: v.getAttribute('stroke-width'),
|
|
61
|
+
opacity: v.getAttribute('opacity'),
|
|
62
|
+
linejoin: v.getAttribute('stroke-linejoin'),
|
|
63
|
+
fill: v.getAttribute('fill'),
|
|
64
|
+
strokePresent: !!v.getAttribute('stroke') && v.getAttribute('stroke') !== 'none',
|
|
65
|
+
};
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
await browser.close();
|
|
69
|
+
|
|
70
|
+
const results = {
|
|
71
|
+
// R393: rx 0 → 2
|
|
72
|
+
rx_attr_2: probe?.rxAttr === '2',
|
|
73
|
+
rx_data_2: probe?.rxData === '2',
|
|
74
|
+
// R346 hover-state rest invariants (no hover yet → rest values)
|
|
75
|
+
rest_strokeWidth_15: probe?.strokeWidth === '1.5',
|
|
76
|
+
rest_opacity_09: probe?.opacity === '0.9',
|
|
77
|
+
// R379 + base invariants
|
|
78
|
+
linejoin_round: probe?.linejoin === 'round',
|
|
79
|
+
fill_none: probe?.fill === 'none',
|
|
80
|
+
stroke_present: probe?.strokePresent === true,
|
|
81
|
+
};
|
|
82
|
+
const ok = Object.values(results).every(Boolean);
|
|
83
|
+
console.log(`${ok ? '✅' : '❌'} minimap viewport rect rx 0 → 2:`, JSON.stringify(results),
|
|
84
|
+
'\n probe:', JSON.stringify(probe, null, 2));
|
|
85
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,103 @@
|
|
|
1
|
+
/* Round 368 verification: recent-panel '+N more flows' footer text
|
|
2
|
+
* fontWeight 400 → 500. Sibling small-text fw lift family with R363
|
|
3
|
+
* recent-row alias + R364 legend-row label + R366 group-label count
|
|
4
|
+
* — all four lifts thicken small-fontSize text (9-11 px) against
|
|
5
|
+
* panel chrome from SVG-default 400 to the font-medium tier (500).
|
|
6
|
+
*
|
|
7
|
+
* The footer only renders when flowLinks.length > 3 (the panel shows
|
|
8
|
+
* the 3 most recent + a footer link to /messages for the overflow).
|
|
9
|
+
*
|
|
10
|
+
* Contract (cyber, 5+ flow links so the footer mounts):
|
|
11
|
+
* - data-recent-panel-more text element font-weight === '500'.
|
|
12
|
+
* - data-recent-panel-more-font-weight === '500'.
|
|
13
|
+
* - Pre-R368 invariants on the same element:
|
|
14
|
+
* * fontSize='9' + fontFamily='monospace' + fontStyle='italic'
|
|
15
|
+
* * letter-spacing='0.2' (R325 rest baseline)
|
|
16
|
+
* * opacity='0.55' (rest, R325 era)
|
|
17
|
+
* * inline transition lists opacity + letter-spacing + fill
|
|
18
|
+
* * data-recent-panel-more attr surfaces the count
|
|
19
|
+
* * data-recent-panel-more-unit tspan with opacity 0.7 (R340)
|
|
20
|
+
*/
|
|
21
|
+
import { chromium } from 'playwright';
|
|
22
|
+
import { readFileSync } from 'node:fs';
|
|
23
|
+
|
|
24
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
25
|
+
const browser = await chromium.launch({ headless: true });
|
|
26
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
27
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
28
|
+
await ctx.addInitScript(() => {
|
|
29
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
30
|
+
});
|
|
31
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
32
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
33
|
+
const r = await route.fetch();
|
|
34
|
+
const b = await r.json();
|
|
35
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
36
|
+
const mk = (alias) => ({
|
|
37
|
+
alias, status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
38
|
+
network_id: nid, project_dir: null,
|
|
39
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
40
|
+
});
|
|
41
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [
|
|
42
|
+
mk('a'), mk('b'), mk('c'), mk('d'), mk('e'),
|
|
43
|
+
] } });
|
|
44
|
+
});
|
|
45
|
+
// 5 distinct flow links → panel shows 3 + footer "+ 2 more flows".
|
|
46
|
+
const now = Date.now();
|
|
47
|
+
const pairs = [['a','b'], ['c','d'], ['a','c'], ['b','d'], ['e','a']];
|
|
48
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages:
|
|
49
|
+
pairs.map(([from, to], idx) => ({
|
|
50
|
+
id: `m${idx}`, from_alias: from, to_alias: to, content: `ping ${idx}`,
|
|
51
|
+
network_id: 'default',
|
|
52
|
+
created_at: new Date(now - (5000 + idx * 1000)).toISOString(),
|
|
53
|
+
}))
|
|
54
|
+
} }));
|
|
55
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
56
|
+
|
|
57
|
+
const page = await ctx.newPage();
|
|
58
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
59
|
+
await page.waitForSelector('[data-recent-panel-more]', { timeout: 15000 });
|
|
60
|
+
await page.waitForTimeout(300);
|
|
61
|
+
|
|
62
|
+
const probe = await page.evaluate(() => {
|
|
63
|
+
const t = document.querySelector('[data-recent-panel-more]');
|
|
64
|
+
const unit = document.querySelector('[data-recent-panel-more-unit]');
|
|
65
|
+
const cs = t ? getComputedStyle(t) : null;
|
|
66
|
+
return {
|
|
67
|
+
fontWeightAttr: t?.getAttribute('font-weight') ?? null,
|
|
68
|
+
fontWeightData: t?.getAttribute('data-recent-panel-more-font-weight') ?? null,
|
|
69
|
+
fontSizeAttr: t?.getAttribute('font-size') ?? null,
|
|
70
|
+
fontFamily: t?.getAttribute('font-family') ?? null,
|
|
71
|
+
fontStyle: t?.getAttribute('font-style') ?? null,
|
|
72
|
+
letterSpacing: t?.getAttribute('letter-spacing') ?? null,
|
|
73
|
+
opacityAttr: t?.getAttribute('opacity') ?? null,
|
|
74
|
+
moreCount: t?.getAttribute('data-recent-panel-more') ?? null,
|
|
75
|
+
hoveredAttr: t?.getAttribute('data-recent-panel-more-hovered') ?? null,
|
|
76
|
+
transition: cs?.transition ?? null,
|
|
77
|
+
unitOpacity: unit?.getAttribute('opacity') ?? null,
|
|
78
|
+
};
|
|
79
|
+
});
|
|
80
|
+
|
|
81
|
+
await browser.close();
|
|
82
|
+
|
|
83
|
+
const hasTrans = (s, prop) =>
|
|
84
|
+
new RegExp(`${prop}\\s+\\d*\\.?\\d*s|${prop}\\s+\\d+ms`, 'i').test(s || '');
|
|
85
|
+
|
|
86
|
+
const results = {
|
|
87
|
+
font_weight_500: probe.fontWeightAttr === '500',
|
|
88
|
+
data_fw_500: probe.fontWeightData === '500',
|
|
89
|
+
font_size_9: probe.fontSizeAttr === '9',
|
|
90
|
+
font_family_mono: /monospace/i.test(probe.fontFamily || ''),
|
|
91
|
+
font_style_italic: probe.fontStyle === 'italic', // pre-R368
|
|
92
|
+
letter_spacing_0_2: probe.letterSpacing === '0.2', // R325 rest
|
|
93
|
+
opacity_0_55_rest: probe.opacityAttr === '0.55', // R325 era rest
|
|
94
|
+
more_count_2: probe.moreCount === '2', // 5 links - 3 shown
|
|
95
|
+
hovered_false_rest: probe.hoveredAttr === 'false',
|
|
96
|
+
trans_has_letterspacing: hasTrans(probe.transition, 'letter-spacing'),
|
|
97
|
+
trans_has_fill: hasTrans(probe.transition, 'fill'),
|
|
98
|
+
unit_opacity_0_7: probe.unitOpacity === '0.7', // R340 unit split
|
|
99
|
+
};
|
|
100
|
+
const ok = Object.values(results).every(Boolean);
|
|
101
|
+
console.log(`${ok ? '✅' : '❌'} more-flows footer fw 400 → 500:`, JSON.stringify(results),
|
|
102
|
+
'\n probe:', probe);
|
|
103
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,112 @@
|
|
|
1
|
+
/* Round 427 verification: node alias label letter-spacing — 3-tier
|
|
2
|
+
* scale rest 0 / hover 0.3 / chat-target 0.5. Extends the hover-
|
|
3
|
+
* letter-spacing family (R344/R345/R347/R351/R420) to per-node alias
|
|
4
|
+
* scope.
|
|
5
|
+
*
|
|
6
|
+
* Contract:
|
|
7
|
+
* - rest: every visible alias-text reports letter-spacing 0px
|
|
8
|
+
* (resolves to keyword 'normal' in computed style on some engines —
|
|
9
|
+
* R218 trap; accept either form)
|
|
10
|
+
* - hover a node: that node's alias-text reports letter-spacing 0.3px,
|
|
11
|
+
* all others stay at rest 0
|
|
12
|
+
* - source-file probe confirms the 3-tier conditional + transition list
|
|
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 (3 nodes) so labels are in card-mode, not dense
|
|
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-alias-text]', { timeout: 15000 });
|
|
48
|
+
await page.waitForTimeout(400);
|
|
49
|
+
|
|
50
|
+
const readAll = () => page.evaluate(() => {
|
|
51
|
+
const ts = [...document.querySelectorAll('[data-node-alias-text]')];
|
|
52
|
+
return ts.map(t => ({
|
|
53
|
+
alias: t.getAttribute('data-node-alias-text'),
|
|
54
|
+
ls: t.style.letterSpacing, // inline style
|
|
55
|
+
ls_c: getComputedStyle(t).letterSpacing, // computed (may be 'normal')
|
|
56
|
+
chat: t.getAttribute('data-node-alias-chat-target'),
|
|
57
|
+
hover: t.getAttribute('data-node-alias-hovered'),
|
|
58
|
+
}));
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
const rest = await readAll();
|
|
62
|
+
|
|
63
|
+
// Hover the first node's group (its bounding rect)
|
|
64
|
+
const firstAlias = rest[0]?.alias;
|
|
65
|
+
let hover = null;
|
|
66
|
+
if (firstAlias) {
|
|
67
|
+
const box = await page.evaluate((alias) => {
|
|
68
|
+
const t = document.querySelector(`[data-node-alias-text="${alias}"]`);
|
|
69
|
+
if (!t) return null;
|
|
70
|
+
// Walk up to the node group (data-node) for the group-hover hit.
|
|
71
|
+
const node = t.closest('[data-node]');
|
|
72
|
+
const target = node || t;
|
|
73
|
+
const b = target.getBoundingClientRect();
|
|
74
|
+
return { x: b.x + b.width / 2, y: b.y + b.height / 2 };
|
|
75
|
+
}, firstAlias);
|
|
76
|
+
if (box) {
|
|
77
|
+
await page.mouse.move(box.x, box.y);
|
|
78
|
+
await page.waitForTimeout(300);
|
|
79
|
+
hover = await readAll();
|
|
80
|
+
await page.mouse.move(0, 0);
|
|
81
|
+
}
|
|
82
|
+
}
|
|
83
|
+
|
|
84
|
+
// Source-file probe
|
|
85
|
+
const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
|
|
86
|
+
const sourceWired = /chatAlias\s*===\s*session\.alias\s*\?\s*'0\.5px'\s*:\s*\n\s*hoveredAlias\s*===\s*session\.alias\s*\?\s*'0\.3px'\s*:\s*'0px'/.test(fileText);
|
|
87
|
+
const sourceTransition = /transition: 'fill 300ms ease-out, letter-spacing 200ms ease-out'/.test(fileText);
|
|
88
|
+
|
|
89
|
+
await browser.close();
|
|
90
|
+
|
|
91
|
+
const isRest = (ls) => ls === '0px' || ls === '' || ls === 'normal';
|
|
92
|
+
const isMid = (ls) => ls === '0.3px';
|
|
93
|
+
|
|
94
|
+
const restAllZero = rest.length > 0 && rest.every(r => isRest(r.ls));
|
|
95
|
+
const hoveredEntry = hover?.find(r => r.alias === firstAlias);
|
|
96
|
+
const othersStillRest = hover ? hover.filter(r => r.alias !== firstAlias).every(r => isRest(r.ls)) : false;
|
|
97
|
+
|
|
98
|
+
const results = {
|
|
99
|
+
rest_mounted_count_gte_3: rest.length >= 3,
|
|
100
|
+
rest_chat_all_false: rest.every(r => r.chat === 'false'),
|
|
101
|
+
rest_hover_all_false: rest.every(r => r.hover === 'false'),
|
|
102
|
+
rest_letter_spacing_zero: restAllZero,
|
|
103
|
+
hover_target_ls_0_3: isMid(hoveredEntry?.ls || ''),
|
|
104
|
+
hover_target_hover_attr: hoveredEntry?.hover === 'true',
|
|
105
|
+
hover_others_stay_rest: othersStillRest,
|
|
106
|
+
source_three_tier_wired: sourceWired,
|
|
107
|
+
source_transition_kept: sourceTransition,
|
|
108
|
+
};
|
|
109
|
+
const ok = Object.values(results).every(Boolean);
|
|
110
|
+
console.log(`${ok ? '✅' : '❌'} node alias letter-spacing 3-tier:`, JSON.stringify(results),
|
|
111
|
+
'\n rest:', JSON.stringify(rest), '\n hover:', JSON.stringify(hover));
|
|
112
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,87 @@
|
|
|
1
|
+
/* Round 407 verification: node halo offline opacity lift — cyber
|
|
2
|
+
* 0.25 → 0.30 and light 0.4 → 0.45. Probes both themes via two
|
|
3
|
+
* browser contexts with a single offline (ghost) session.
|
|
4
|
+
*
|
|
5
|
+
* Contract:
|
|
6
|
+
* - Online node (status: idle): opacity '0.65' cyber / '0.85' light
|
|
7
|
+
* (unchanged by R407 — invariant probe)
|
|
8
|
+
* - Offline node (status: offline + stale last_seen_at):
|
|
9
|
+
* * cyber: opacity attr === '0.3'
|
|
10
|
+
* * light: opacity attr === '0.45'
|
|
11
|
+
* * data-node-halo-offline-opacity === '0.3' / '0.45'
|
|
12
|
+
* - Source-file verification: the conditional
|
|
13
|
+
* `(isLight ? 0.45 : 0.30)` is present.
|
|
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
|
+
const stale = new Date(Date.now() - 30 * 60 * 1000).toISOString();
|
|
21
|
+
|
|
22
|
+
async function probeTheme(theme) {
|
|
23
|
+
const browser = await chromium.launch({ headless: true });
|
|
24
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
25
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
26
|
+
await ctx.addInitScript((t) => {
|
|
27
|
+
try { localStorage.setItem('anet-theme', t); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
28
|
+
}, theme);
|
|
29
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
30
|
+
const r = await route.fetch();
|
|
31
|
+
const b = await r.json();
|
|
32
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
33
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [
|
|
34
|
+
{ alias: 'alpha', status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli', network_id: nid, project_dir: null, created_at: fresh, updated_at: fresh, last_seen_at: fresh },
|
|
35
|
+
{ alias: 'ghost', status: 'offline', model: 'claude-opus-4', runtime: 'claude-code-cli', network_id: nid, project_dir: null, created_at: stale, updated_at: stale, last_seen_at: stale },
|
|
36
|
+
] } });
|
|
37
|
+
});
|
|
38
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
|
|
39
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
40
|
+
const page = await ctx.newPage();
|
|
41
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
42
|
+
await page.waitForSelector('g[data-node]', { timeout: 15000 });
|
|
43
|
+
await page.waitForTimeout(400);
|
|
44
|
+
const probe = await page.evaluate(() => {
|
|
45
|
+
const nodes = Array.from(document.querySelectorAll('g[data-node]'));
|
|
46
|
+
const read = (alias) => {
|
|
47
|
+
const node = nodes.find((n) => n.getAttribute('data-node') === alias);
|
|
48
|
+
if (!node) return null;
|
|
49
|
+
const halo = node.querySelector('[data-node-halo-breath]');
|
|
50
|
+
return halo ? {
|
|
51
|
+
opacityAttr: halo.getAttribute('opacity'),
|
|
52
|
+
offlineData: halo.getAttribute('data-node-halo-offline-opacity'),
|
|
53
|
+
} : null;
|
|
54
|
+
};
|
|
55
|
+
return {
|
|
56
|
+
online: read('alpha'),
|
|
57
|
+
offline: read('ghost'),
|
|
58
|
+
};
|
|
59
|
+
});
|
|
60
|
+
await browser.close();
|
|
61
|
+
return probe;
|
|
62
|
+
}
|
|
63
|
+
|
|
64
|
+
const cyber = await probeTheme('cyber');
|
|
65
|
+
const light = await probeTheme('light');
|
|
66
|
+
|
|
67
|
+
const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
|
|
68
|
+
const sourceHasOfflineLift = /isOnline\s*\?\s*\(isLight\s*\?\s*0\.85\s*:\s*0\.65\)\s*:\s*\(isLight\s*\?\s*0\.45\s*:\s*0\.30\)/.test(fileText);
|
|
69
|
+
|
|
70
|
+
const results = {
|
|
71
|
+
// Online (R407 doesn't touch online — invariant)
|
|
72
|
+
cyber_online_065: cyber?.online?.opacityAttr === '0.65',
|
|
73
|
+
light_online_085: light?.online?.opacityAttr === '0.85',
|
|
74
|
+
cyber_online_no_offline_attr: cyber?.online?.offlineData === null || cyber?.online?.offlineData === undefined,
|
|
75
|
+
// Offline (R407 lift)
|
|
76
|
+
cyber_offline_0_3: cyber?.offline?.opacityAttr === '0.3',
|
|
77
|
+
cyber_offline_data_0_3: cyber?.offline?.offlineData === '0.3',
|
|
78
|
+
light_offline_0_45: light?.offline?.opacityAttr === '0.45',
|
|
79
|
+
light_offline_data_0_45: light?.offline?.offlineData === '0.45',
|
|
80
|
+
// Source-file verification
|
|
81
|
+
source_offline_lift: sourceHasOfflineLift,
|
|
82
|
+
};
|
|
83
|
+
const ok = Object.values(results).every(Boolean);
|
|
84
|
+
console.log(`${ok ? '✅' : '❌'} node halo offline opacity lift:`, JSON.stringify(results),
|
|
85
|
+
'\n cyber:', cyber,
|
|
86
|
+
'\n light:', light);
|
|
87
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -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);
|