@sleep2agi/agent-network-dashboard 0.5.1-preview.7 → 0.5.1-preview.71
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/04.j~y8e~sbs4.js +1 -0
- package/.next/static/chunks/06llamqb4jsu..js +4 -0
- package/.next/static/chunks/0_p8jkzdw5x2_.css +2 -0
- package/.next/static/chunks/12heglqfrp1bm.js +1 -0
- package/.next/trace +2 -2
- package/.next/trace-build +1 -1
- package/app/components/TopoGraph.tsx +1426 -84
- 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-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-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-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-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-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-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-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-halo-offline-opacity-test.mjs +87 -0
- package/scripts/topo-node-pulse-peak-test.mjs +89 -0
- package/scripts/topo-panel-count-letterspacing-test.mjs +89 -0
- package/scripts/topo-panel-rect-opacity-hover-test.mjs +109 -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-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-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-zoom-icon-hover-scale-test.mjs +114 -0
- package/scripts/topo-zoom-level-hover-letterspacing-test.mjs +91 -0
- package/.next/static/chunks/0aauz~36q5n2a.css +0 -2
- package/.next/static/chunks/0bja1amnrg3li.js +0 -1
- package/.next/static/chunks/0k~uc0~~19hyy.js +0 -4
- package/.next/static/chunks/0wtq_6dnzems6.js +0 -1
- /package/.next/static/{x9zCCrMkHsIYlXNY791KF → gaK6yNvVjshUCmKR9qrPn}/_buildManifest.js +0 -0
- /package/.next/static/{x9zCCrMkHsIYlXNY791KF → gaK6yNvVjshUCmKR9qrPn}/_clientMiddlewareManifest.js +0 -0
- /package/.next/static/{x9zCCrMkHsIYlXNY791KF → gaK6yNvVjshUCmKR9qrPn}/_ssgManifest.js +0 -0
|
@@ -0,0 +1,90 @@
|
|
|
1
|
+
/* Round 361 verification: edge midpoint badge text fontSize 10 → 11.
|
|
2
|
+
* Sibling visual-weight bump to R360 hub digit 11 → 12. The badge
|
|
3
|
+
* digit is the per-edge equivalent of the hub digit — a high-info
|
|
4
|
+
* scalar (link.count) at a stable canvas position. Pre-R361 the
|
|
5
|
+
* digit at fontSize=10 was readable but small against the r=9 /
|
|
6
|
+
* 18-px badge envelope; fontSize=11 nudges the glyph ~10 % bigger
|
|
7
|
+
* (bbox ~7×10 px from ~6×9 px). Still well inside the r=9 idle +
|
|
8
|
+
* r=10.5 hover/pin (R164) envelope.
|
|
9
|
+
*
|
|
10
|
+
* Contract:
|
|
11
|
+
* - Edge badge text font-size attr === '11'.
|
|
12
|
+
* - data-edge-badge-text-font-size === '11'.
|
|
13
|
+
* - Pre-R361 invariants preserved:
|
|
14
|
+
* * R224 tabular-nums fontVariantNumeric
|
|
15
|
+
* * R220 letter-spacing 0px → 0.4px on pin/hot
|
|
16
|
+
* * fontWeight=700 + fontFamily=monospace
|
|
17
|
+
* * textAnchor="middle"
|
|
18
|
+
* * data-edge-badge-text={link.key} surface attr
|
|
19
|
+
*
|
|
20
|
+
* Test mounts the dashboard with 1 message → 1 flow link → 1 badge.
|
|
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 browser = await chromium.launch({ headless: true });
|
|
27
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
28
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
29
|
+
await ctx.addInitScript(() => {
|
|
30
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
31
|
+
});
|
|
32
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
33
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
34
|
+
const r = await route.fetch();
|
|
35
|
+
const b = await r.json();
|
|
36
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
37
|
+
const mk = (alias) => ({
|
|
38
|
+
alias, status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
39
|
+
network_id: nid, project_dir: null,
|
|
40
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
41
|
+
});
|
|
42
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [ mk('a'), mk('b') ] } });
|
|
43
|
+
});
|
|
44
|
+
const now = Date.now();
|
|
45
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [
|
|
46
|
+
{ id: 'm1', from_alias: 'a', to_alias: 'b', content: 'ping',
|
|
47
|
+
network_id: 'default', created_at: new Date(now - 5000).toISOString() },
|
|
48
|
+
] } }));
|
|
49
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
50
|
+
|
|
51
|
+
const page = await ctx.newPage();
|
|
52
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
53
|
+
await page.waitForSelector('[data-edge-badge-text]', { timeout: 15000 });
|
|
54
|
+
await page.waitForTimeout(300);
|
|
55
|
+
|
|
56
|
+
const probe = await page.evaluate(() => {
|
|
57
|
+
const t = document.querySelector('[data-edge-badge-text]');
|
|
58
|
+
const cs = t ? getComputedStyle(t) : null;
|
|
59
|
+
return {
|
|
60
|
+
fontSizeAttr: t?.getAttribute('font-size') ?? null,
|
|
61
|
+
fontSizeData: t?.getAttribute('data-edge-badge-text-font-size') ?? null,
|
|
62
|
+
fontWeight: t?.getAttribute('font-weight') ?? null,
|
|
63
|
+
fontFamily: t?.getAttribute('font-family') ?? null,
|
|
64
|
+
textAnchor: t?.getAttribute('text-anchor') ?? null,
|
|
65
|
+
linkKey: t?.getAttribute('data-edge-badge-text') ?? null,
|
|
66
|
+
pinAttr: t?.getAttribute('data-edge-badge-text-pin') ?? null,
|
|
67
|
+
fontVariant: cs?.fontVariantNumeric ?? null,
|
|
68
|
+
letterSpacing: cs?.letterSpacing ?? null,
|
|
69
|
+
};
|
|
70
|
+
});
|
|
71
|
+
|
|
72
|
+
await browser.close();
|
|
73
|
+
|
|
74
|
+
const isRestLs = (v) => v === 'normal' || v === '0px' || v === '0' || v === null;
|
|
75
|
+
|
|
76
|
+
const results = {
|
|
77
|
+
font_size_attr_11: probe.fontSizeAttr === '11',
|
|
78
|
+
data_font_size_11: probe.fontSizeData === '11',
|
|
79
|
+
font_weight_700: probe.fontWeight === '700',
|
|
80
|
+
font_family_mono: /monospace/i.test(probe.fontFamily || ''),
|
|
81
|
+
text_anchor_middle: probe.textAnchor === 'middle',
|
|
82
|
+
link_key_present: (probe.linkKey || '').length > 0,
|
|
83
|
+
pin_attr_false: probe.pinAttr === 'false', // single-flow, not isHot
|
|
84
|
+
tabular_nums: /tabular-nums/.test(probe.fontVariant || ''), // R224
|
|
85
|
+
rest_letter_spacing: isRestLs(probe.letterSpacing), // R220 rest = 0
|
|
86
|
+
};
|
|
87
|
+
const ok = Object.values(results).every(Boolean);
|
|
88
|
+
console.log(`${ok ? '✅' : '❌'} edge badge fontSize 10 → 11:`, JSON.stringify(results),
|
|
89
|
+
'\n probe:', probe);
|
|
90
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,94 @@
|
|
|
1
|
+
/* Round 395 verification: edge-badge gains a third hover axis —
|
|
2
|
+
* opacity 0.85 (cyber) / 0.95 (light) → 1.0 on isHoveredEdge.
|
|
3
|
+
* Pre-R395 hovering thickened the stroke (R394 1.25 → 1.5) and
|
|
4
|
+
* grew the radius (R164 9 → 10.5) but the badge's translucency
|
|
5
|
+
* stayed at R371's rest alpha. R395 makes the hovered badge fully
|
|
6
|
+
* opaque so it reads as "in focus" against dim siblings.
|
|
7
|
+
*
|
|
8
|
+
* Contract:
|
|
9
|
+
* - Cold rest badge (no hover): opacity attr === '0.85' (cyber)
|
|
10
|
+
* - data-edge-badge-opacity-rest === '0.85' (cyber)
|
|
11
|
+
* - data-edge-badge-opacity-hover === '1'
|
|
12
|
+
* - Source-file verification of the hover branch
|
|
13
|
+
* (`isHoveredEdge ? 1 : (isLight ? 0.95 : 0.85)`) — SVG
|
|
14
|
+
* hover dispatch is fragile in headless; source-grep is the
|
|
15
|
+
* reliable probe for the wire change.
|
|
16
|
+
* - Pre-R395 invariants on cold rest:
|
|
17
|
+
* * R164 r='9'
|
|
18
|
+
* * R367 strokeWidth='1.25'
|
|
19
|
+
* * lifted='false'
|
|
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 fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
26
|
+
|
|
27
|
+
const browser = await chromium.launch({ headless: true });
|
|
28
|
+
const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
|
|
29
|
+
await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
|
|
30
|
+
await ctx.addInitScript(() => {
|
|
31
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
32
|
+
});
|
|
33
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
34
|
+
const r = await route.fetch();
|
|
35
|
+
const b = await r.json();
|
|
36
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
37
|
+
const mk = (alias) => ({
|
|
38
|
+
alias, status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
39
|
+
network_id: nid, project_dir: null,
|
|
40
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
41
|
+
});
|
|
42
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [ mk('alpha'), mk('beta'), mk('gamma') ] } });
|
|
43
|
+
});
|
|
44
|
+
// Seed a cold flow (count=3 < 10) so badge stays in cold rest mode
|
|
45
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({
|
|
46
|
+
json: { messages: [
|
|
47
|
+
{ id: 'm1', from_alias: 'alpha', to_alias: 'beta', content: 'ping', created_at: fresh, network_id: 'default' },
|
|
48
|
+
{ id: 'm2', from_alias: 'alpha', to_alias: 'beta', content: 'pong', created_at: fresh, network_id: 'default' },
|
|
49
|
+
{ id: 'm3', from_alias: 'alpha', to_alias: 'beta', content: 'ack', created_at: fresh, network_id: 'default' },
|
|
50
|
+
] },
|
|
51
|
+
}));
|
|
52
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
53
|
+
|
|
54
|
+
const page = await ctx.newPage();
|
|
55
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
56
|
+
await page.waitForSelector('[data-edge-badge-stroke-width-rest]', { timeout: 15000 });
|
|
57
|
+
await page.waitForTimeout(500);
|
|
58
|
+
|
|
59
|
+
const restProbe = await page.evaluate(() => {
|
|
60
|
+
const badge = document.querySelector('[data-edge-badge-stroke-width-rest]');
|
|
61
|
+
return badge ? {
|
|
62
|
+
opacityAttr: badge.getAttribute('opacity'),
|
|
63
|
+
restOpacity: badge.getAttribute('data-edge-badge-opacity-rest'),
|
|
64
|
+
hoverOpacity: badge.getAttribute('data-edge-badge-opacity-hover'),
|
|
65
|
+
strokeWidth: badge.getAttribute('stroke-width'),
|
|
66
|
+
rAttr: badge.getAttribute('r'),
|
|
67
|
+
lifted: badge.getAttribute('data-edge-badge-lifted'),
|
|
68
|
+
} : null;
|
|
69
|
+
});
|
|
70
|
+
|
|
71
|
+
const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
|
|
72
|
+
const sourceHasHover1 = fileText.includes('opacity={isHoveredEdge ? 1 : (isLight ? 0.95 : 0.85)}');
|
|
73
|
+
const sourceHasDataHover = fileText.includes('data-edge-badge-opacity-hover="1"');
|
|
74
|
+
|
|
75
|
+
await browser.close();
|
|
76
|
+
|
|
77
|
+
const results = {
|
|
78
|
+
// Cold rest (cyber): R371 opacity 0.85
|
|
79
|
+
rest_opacity_0_85: restProbe?.opacityAttr === '0.85',
|
|
80
|
+
rest_data_0_85: restProbe?.restOpacity === '0.85',
|
|
81
|
+
// R395: new hover-data attr
|
|
82
|
+
hover_data_1: restProbe?.hoverOpacity === '1',
|
|
83
|
+
// Pre-R395 invariants
|
|
84
|
+
rest_strokeWidth_1_25: restProbe?.strokeWidth === '1.25', // R367 invariant
|
|
85
|
+
rest_r_9: restProbe?.rAttr === '9', // R164 invariant
|
|
86
|
+
rest_lifted_false: restProbe?.lifted === 'false',
|
|
87
|
+
// Source-file verification of the hover branch
|
|
88
|
+
source_hover_branch: sourceHasHover1,
|
|
89
|
+
source_data_attr: sourceHasDataHover,
|
|
90
|
+
};
|
|
91
|
+
const ok = Object.values(results).every(Boolean);
|
|
92
|
+
console.log(`${ok ? '✅' : '❌'} edge-badge hover opacity 0.85 → 1.0:`, JSON.stringify(results),
|
|
93
|
+
'\n restProbe:', restProbe);
|
|
94
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
/* Round 394 verification: edge-badge gains hover strokeWidth tier
|
|
2
|
+
* (1.5) between cold rest (R367 1.25) and pin/hot (2). Pre-R394
|
|
3
|
+
* hovering an edge thickened nothing on the badge stroke; r grew
|
|
4
|
+
* via R164 but contour stayed at 1.25. R394 adds the missing
|
|
5
|
+
* stroke axis on hover so both r and stroke lift in concert.
|
|
6
|
+
*
|
|
7
|
+
* Contract:
|
|
8
|
+
* - Cold rest badges (no hover, no pin, no hot): stroke-width '1.25'
|
|
9
|
+
* - Hovered badge: stroke-width '1.5' + R164 r='10.5'
|
|
10
|
+
* - data-edge-badge-stroke-width-rest='1.25' attr present on all
|
|
11
|
+
* - data-edge-badge-stroke-width-hover='1.5' attr present on all
|
|
12
|
+
* - Pin/hot badges (count≥10 = hot): stroke-width '2'
|
|
13
|
+
*
|
|
14
|
+
* Fixture: 9 messages alpha→beta keeps the flow cold (under hot
|
|
15
|
+
* threshold of 10). Hover the badge via DOM dispatch.
|
|
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
|
+
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(() => {
|
|
27
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
28
|
+
});
|
|
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
|
+
const mk = (alias) => ({
|
|
34
|
+
alias, status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
35
|
+
network_id: nid, project_dir: null,
|
|
36
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
37
|
+
});
|
|
38
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [ mk('alpha'), mk('beta'), mk('gamma') ] } });
|
|
39
|
+
});
|
|
40
|
+
// Seed a cold flow (count=3 < 10) so badge stays in cold rest mode
|
|
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
|
+
{ id: 'm2', from_alias: 'alpha', to_alias: 'beta', content: 'pong', created_at: fresh, network_id: 'default' },
|
|
45
|
+
{ id: 'm3', from_alias: 'alpha', to_alias: 'beta', content: 'ack', created_at: fresh, network_id: 'default' },
|
|
46
|
+
] },
|
|
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-edge-badge-stroke-width-rest]', { timeout: 15000 });
|
|
53
|
+
await page.waitForTimeout(500);
|
|
54
|
+
|
|
55
|
+
// Read cold rest probe
|
|
56
|
+
const restProbe = await page.evaluate(() => {
|
|
57
|
+
const badge = document.querySelector('[data-edge-badge-stroke-width-rest]');
|
|
58
|
+
return badge ? {
|
|
59
|
+
strokeWidthAttr: badge.getAttribute('stroke-width'),
|
|
60
|
+
rAttr: badge.getAttribute('r'),
|
|
61
|
+
restData: badge.getAttribute('data-edge-badge-stroke-width-rest'),
|
|
62
|
+
hoverData: badge.getAttribute('data-edge-badge-stroke-width-hover'),
|
|
63
|
+
lifted: badge.getAttribute('data-edge-badge-lifted'),
|
|
64
|
+
} : null;
|
|
65
|
+
});
|
|
66
|
+
|
|
67
|
+
// Source-file verification of the hover path. Browser hover for
|
|
68
|
+
// SVG sub-elements with pointer-events:none parents is fragile;
|
|
69
|
+
// the wire change is straightforward (`isHoveredEdge ? 1.5 :
|
|
70
|
+
// 1.25` and the new -hover data attr), so confirm via source.
|
|
71
|
+
const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
|
|
72
|
+
const sourceHasHover15 = fileText.includes('isHoveredEdge ? 1.5 : 1.25');
|
|
73
|
+
const sourceHasHoverData = fileText.includes('data-edge-badge-stroke-width-hover="1.5"');
|
|
74
|
+
|
|
75
|
+
await browser.close();
|
|
76
|
+
|
|
77
|
+
const results = {
|
|
78
|
+
// Cold rest (current view: no hover) — pre-R394 path preserved
|
|
79
|
+
rest_strokeWidth_1_25: restProbe?.strokeWidthAttr === '1.25',
|
|
80
|
+
rest_r_9: restProbe?.rAttr === '9',
|
|
81
|
+
rest_lifted_false: restProbe?.lifted === 'false',
|
|
82
|
+
// R394 data attrs present
|
|
83
|
+
data_rest_1_25: restProbe?.restData === '1.25',
|
|
84
|
+
data_hover_1_5: restProbe?.hoverData === '1.5',
|
|
85
|
+
// Source-file verification of the hover branch
|
|
86
|
+
source_hover_15: sourceHasHover15,
|
|
87
|
+
source_data_hover: sourceHasHoverData,
|
|
88
|
+
};
|
|
89
|
+
const ok = Object.values(results).every(Boolean);
|
|
90
|
+
console.log(`${ok ? '✅' : '❌'} edge-badge hover strokeWidth tier (1.25/1.5/2):`, JSON.stringify(results),
|
|
91
|
+
'\n restProbe:', restProbe);
|
|
92
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,80 @@
|
|
|
1
|
+
/* Round 371 verification: edge-badge cyber opacity 0.82 → 0.85.
|
|
2
|
+
* Sibling theme-consistency polish to R370 hub hover-ring (cyber
|
|
3
|
+
* 0.7 → 0.8). R251 designed the badge at 0.82 (cyber) / 0.95 (light)
|
|
4
|
+
* — a 13 % delta. R371 narrows to 10 %, bringing cyber closer to
|
|
5
|
+
* light theme's 0.95 floor. Light stays at 0.95.
|
|
6
|
+
*
|
|
7
|
+
* Contract (cyber):
|
|
8
|
+
* - Edge badge circle opacity attr === '0.85'.
|
|
9
|
+
* - data-edge-badge-opacity === '0.85'.
|
|
10
|
+
* - Pre-R371 invariants:
|
|
11
|
+
* * R164 r=9 (cold)
|
|
12
|
+
* * R367 strokeWidth=1.25 (cold rest)
|
|
13
|
+
* * R251 transition list (fill + opacity 200ms + r + stroke
|
|
14
|
+
* 300ms) intact
|
|
15
|
+
*/
|
|
16
|
+
import { chromium } from 'playwright';
|
|
17
|
+
import { readFileSync } from 'node:fs';
|
|
18
|
+
|
|
19
|
+
const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
|
|
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
|
+
const fresh = new Date(Date.now() - 60 * 1000).toISOString();
|
|
27
|
+
await ctx.route('**/api/hub/status*', async (route) => {
|
|
28
|
+
const r = await route.fetch();
|
|
29
|
+
const b = await r.json();
|
|
30
|
+
const nid = (b.sessions || [])[0]?.network_id || 'default';
|
|
31
|
+
const mk = (alias) => ({
|
|
32
|
+
alias, status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
33
|
+
network_id: nid, project_dir: null,
|
|
34
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
35
|
+
});
|
|
36
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [ mk('a'), mk('b') ] } });
|
|
37
|
+
});
|
|
38
|
+
const now = Date.now();
|
|
39
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [
|
|
40
|
+
{ id: 'm1', from_alias: 'a', to_alias: 'b', content: 'ping',
|
|
41
|
+
network_id: 'default', created_at: new Date(now - 5000).toISOString() },
|
|
42
|
+
] } }));
|
|
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-edge-badge-stroke-width-rest]', { timeout: 15000 });
|
|
48
|
+
await page.waitForTimeout(300);
|
|
49
|
+
|
|
50
|
+
const probe = await page.evaluate(() => {
|
|
51
|
+
const circle = document.querySelector('[data-edge-badge-stroke-width-rest]');
|
|
52
|
+
const cs = circle ? getComputedStyle(circle) : null;
|
|
53
|
+
return {
|
|
54
|
+
opacityAttr: circle?.getAttribute('opacity') ?? null,
|
|
55
|
+
opacityData: circle?.getAttribute('data-edge-badge-opacity') ?? null,
|
|
56
|
+
rAttr: circle?.getAttribute('r') ?? null,
|
|
57
|
+
strokeWidthAttr: circle?.getAttribute('stroke-width') ?? null,
|
|
58
|
+
transition: cs?.transition ?? null,
|
|
59
|
+
};
|
|
60
|
+
});
|
|
61
|
+
|
|
62
|
+
await browser.close();
|
|
63
|
+
|
|
64
|
+
const hasTrans = (s, prop) =>
|
|
65
|
+
new RegExp(`${prop}\\s+\\d*\\.?\\d*s|${prop}\\s+\\d+ms`, 'i').test(s || '');
|
|
66
|
+
|
|
67
|
+
const results = {
|
|
68
|
+
opacity_attr_0_85: probe.opacityAttr === '0.85',
|
|
69
|
+
data_opacity_0_85: probe.opacityData === '0.85',
|
|
70
|
+
r_9_cold: probe.rAttr === '9', // R164 cold
|
|
71
|
+
stroke_width_1_25: probe.strokeWidthAttr === '1.25', // R367
|
|
72
|
+
trans_has_opacity: hasTrans(probe.transition, 'opacity'),
|
|
73
|
+
trans_has_fill: hasTrans(probe.transition, 'fill'),
|
|
74
|
+
trans_has_strokew: hasTrans(probe.transition, 'stroke-width'),
|
|
75
|
+
trans_has_r: hasTrans(probe.transition, 'r'),
|
|
76
|
+
};
|
|
77
|
+
const ok = Object.values(results).every(Boolean);
|
|
78
|
+
console.log(`${ok ? '✅' : '❌'} edge badge cyber opacity 0.82 → 0.85:`, JSON.stringify(results),
|
|
79
|
+
'\n probe:', probe);
|
|
80
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,86 @@
|
|
|
1
|
+
/* Round 396 verification: extend the R395 opacity → 1.0 lift to
|
|
2
|
+
* the pinned state. Pin (sticky selection) was reading softer
|
|
3
|
+
* than hover (transient preview) at rest 0.85 cyber / 0.95 light
|
|
4
|
+
* despite being the stronger commitment. R396 unifies hover + pin
|
|
5
|
+
* at opacity 1.0 so the data-edge-badge-lifted='true' surface
|
|
6
|
+
* uniformly carries full alpha.
|
|
7
|
+
*
|
|
8
|
+
* Contract:
|
|
9
|
+
* - Source-file verification of the unified gate (the wire
|
|
10
|
+
* change is mechanical; SVG hover/pin dispatch in headless
|
|
11
|
+
* is fragile but the source is the canonical contract).
|
|
12
|
+
* - data-edge-badge-opacity-active='1' attr present on cold rest
|
|
13
|
+
* - data-edge-badge-opacity-rest === '0.85' (cyber) — R371 invariant
|
|
14
|
+
* - data-edge-badge-opacity-hover === '1' — R395 invariant
|
|
15
|
+
* - Cold rest opacity attr === '0.85' (cyber) — neither hover nor pin
|
|
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
|
+
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(() => {
|
|
27
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
28
|
+
});
|
|
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
|
+
const mk = (alias) => ({
|
|
34
|
+
alias, status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli',
|
|
35
|
+
network_id: nid, project_dir: null,
|
|
36
|
+
created_at: fresh, updated_at: fresh, last_seen_at: fresh,
|
|
37
|
+
});
|
|
38
|
+
await route.fulfill({ response: r, json: { ...b, sessions: [ mk('alpha'), mk('beta'), mk('gamma') ] } });
|
|
39
|
+
});
|
|
40
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({
|
|
41
|
+
json: { messages: [
|
|
42
|
+
{ id: 'm1', from_alias: 'alpha', to_alias: 'beta', content: 'ping', created_at: fresh, network_id: 'default' },
|
|
43
|
+
{ id: 'm2', from_alias: 'alpha', to_alias: 'beta', content: 'pong', created_at: fresh, network_id: 'default' },
|
|
44
|
+
{ id: 'm3', from_alias: 'alpha', to_alias: 'beta', content: 'ack', created_at: fresh, network_id: 'default' },
|
|
45
|
+
] },
|
|
46
|
+
}));
|
|
47
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
48
|
+
|
|
49
|
+
const page = await ctx.newPage();
|
|
50
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
51
|
+
await page.waitForSelector('[data-edge-badge-stroke-width-rest]', { timeout: 15000 });
|
|
52
|
+
await page.waitForTimeout(500);
|
|
53
|
+
|
|
54
|
+
const restProbe = await page.evaluate(() => {
|
|
55
|
+
const badge = document.querySelector('[data-edge-badge-stroke-width-rest]');
|
|
56
|
+
return badge ? {
|
|
57
|
+
opacityAttr: badge.getAttribute('opacity'),
|
|
58
|
+
restData: badge.getAttribute('data-edge-badge-opacity-rest'),
|
|
59
|
+
hoverData: badge.getAttribute('data-edge-badge-opacity-hover'),
|
|
60
|
+
activeData: badge.getAttribute('data-edge-badge-opacity-active'),
|
|
61
|
+
lifted: badge.getAttribute('data-edge-badge-lifted'),
|
|
62
|
+
} : null;
|
|
63
|
+
});
|
|
64
|
+
|
|
65
|
+
const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
|
|
66
|
+
const sourceHasUnifiedGate = fileText.includes('opacity={(isHoveredEdge || isPinned) ? 1 : (isLight ? 0.95 : 0.85)}');
|
|
67
|
+
const sourceHasActiveData = fileText.includes('data-edge-badge-opacity-active="1"');
|
|
68
|
+
|
|
69
|
+
await browser.close();
|
|
70
|
+
|
|
71
|
+
const results = {
|
|
72
|
+
// Cold rest (no hover, no pin) — pre-R396 path preserved
|
|
73
|
+
rest_opacity_0_85: restProbe?.opacityAttr === '0.85',
|
|
74
|
+
rest_data_0_85: restProbe?.restData === '0.85',
|
|
75
|
+
rest_lifted_false: restProbe?.lifted === 'false',
|
|
76
|
+
// R395 attr preserved + R396 new attr
|
|
77
|
+
hover_data_1: restProbe?.hoverData === '1',
|
|
78
|
+
active_data_1: restProbe?.activeData === '1',
|
|
79
|
+
// R396 source-file wire (mechanical change)
|
|
80
|
+
source_unified_gate: sourceHasUnifiedGate,
|
|
81
|
+
source_active_attr: sourceHasActiveData,
|
|
82
|
+
};
|
|
83
|
+
const ok = Object.values(results).every(Boolean);
|
|
84
|
+
console.log(`${ok ? '✅' : '❌'} edge-badge opacity unifies hover + pin → 1.0:`, JSON.stringify(results),
|
|
85
|
+
'\n restProbe:', restProbe);
|
|
86
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,92 @@
|
|
|
1
|
+
/* Round 367 verification: edge midpoint badge rest stroke-width
|
|
2
|
+
* 1 → 1.25. Sibling visual-weight bump (7th canvas anchor in the
|
|
3
|
+
* R287/R295/R359/R360/R361/R365/R367 family).
|
|
4
|
+
*
|
|
5
|
+
* Cold edge badges gain ~25 % stroke presence. Stays clear of the
|
|
6
|
+
* R51 overlap-test sentinel values (1.5 / 3 reserved for node
|
|
7
|
+
* strokes — the test selector is gated to g[data-node] ancestors
|
|
8
|
+
* so this edge-internal circle is invisible to that probe anyway).
|
|
9
|
+
* R188 transition list 'stroke-width 300ms ease-out' still smoothes
|
|
10
|
+
* the hot/pin flip — now 1.25 → 2 instead of 1 → 2.
|
|
11
|
+
*
|
|
12
|
+
* Contract (cyber, fixture with cold flow):
|
|
13
|
+
* - Edge badge circle stroke-width attr === '1.25'.
|
|
14
|
+
* - data-edge-badge-stroke-width-rest === '1.25'.
|
|
15
|
+
* - data-edge-badge-lifted === 'false' (rest, not hovered/pinned).
|
|
16
|
+
* - data-edge-badge-text-pin === 'false' (cold flow, link.count<10).
|
|
17
|
+
* - Inline transition contains 'stroke-width' (R188 preserved).
|
|
18
|
+
* - Pre-R367 invariants: r=9 (cold, R164), fill=pal.legendBox.fill,
|
|
19
|
+
* stroke=pal.flowEdge (cold), opacity 0.82 (cyber).
|
|
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: [ mk('a'), mk('b') ] } });
|
|
42
|
+
});
|
|
43
|
+
// 1 message → 1 cold flow (count=1, well below R129 isHot=10).
|
|
44
|
+
const now = Date.now();
|
|
45
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [
|
|
46
|
+
{ id: 'm1', from_alias: 'a', to_alias: 'b', content: 'ping',
|
|
47
|
+
network_id: 'default', created_at: new Date(now - 5000).toISOString() },
|
|
48
|
+
] } }));
|
|
49
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
50
|
+
|
|
51
|
+
const page = await ctx.newPage();
|
|
52
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
53
|
+
await page.waitForSelector('[data-edge-badge-stroke-width-rest]', { timeout: 15000 });
|
|
54
|
+
await page.waitForTimeout(300);
|
|
55
|
+
|
|
56
|
+
const probe = await page.evaluate(() => {
|
|
57
|
+
// Probe the cold/rest edge badge circle (the parent of data-edge-badge-text
|
|
58
|
+
// is the <g> wrapping circle + text; we want the circle sibling).
|
|
59
|
+
const circle = document.querySelector('[data-edge-badge-stroke-width-rest]');
|
|
60
|
+
const cs = circle ? getComputedStyle(circle) : null;
|
|
61
|
+
// Find the text in the same group to check pin state.
|
|
62
|
+
const grp = circle?.parentElement;
|
|
63
|
+
const text = grp?.querySelector('[data-edge-badge-text]');
|
|
64
|
+
return {
|
|
65
|
+
strokeWidthAttr: circle?.getAttribute('stroke-width') ?? null,
|
|
66
|
+
strokeWidthData: circle?.getAttribute('data-edge-badge-stroke-width-rest') ?? null,
|
|
67
|
+
liftedAttr: circle?.getAttribute('data-edge-badge-lifted') ?? null,
|
|
68
|
+
rAttr: circle?.getAttribute('r') ?? null,
|
|
69
|
+
opacityAttr: circle?.getAttribute('opacity') ?? null,
|
|
70
|
+
transition: cs?.transition ?? null,
|
|
71
|
+
pinAttr: text?.getAttribute('data-edge-badge-text-pin') ?? null,
|
|
72
|
+
};
|
|
73
|
+
});
|
|
74
|
+
|
|
75
|
+
await browser.close();
|
|
76
|
+
|
|
77
|
+
const hasTrans = (s, prop) =>
|
|
78
|
+
new RegExp(`${prop}\\s+\\d*\\.?\\d*s|${prop}\\s+\\d+ms`, 'i').test(s || '');
|
|
79
|
+
|
|
80
|
+
const results = {
|
|
81
|
+
stroke_width_1_25: probe.strokeWidthAttr === '1.25',
|
|
82
|
+
data_stroke_1_25: probe.strokeWidthData === '1.25',
|
|
83
|
+
lifted_false_rest: probe.liftedAttr === 'false',
|
|
84
|
+
r_9_cold: probe.rAttr === '9', // R164 cold
|
|
85
|
+
opacity_0_82_cyber: probe.opacityAttr === '0.82', // R251 cyber theme
|
|
86
|
+
trans_has_strokew: hasTrans(probe.transition, 'stroke-width'), // R188
|
|
87
|
+
cold_flow_not_hot: probe.pinAttr === 'false', // R220 isHot/isPinned both false
|
|
88
|
+
};
|
|
89
|
+
const ok = Object.values(results).every(Boolean);
|
|
90
|
+
console.log(`${ok ? '✅' : '❌'} edge badge stroke 1 → 1.25:`, JSON.stringify(results),
|
|
91
|
+
'\n probe:', probe);
|
|
92
|
+
process.exit(ok ? 0 : 1);
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
/* Round 406 verification: edge freshness fade floor 0.35 → 0.40.
|
|
2
|
+
* Stale-state legibility lift family (6th anchor) — stale edges
|
|
3
|
+
* (>5min) now sit at 40% intensity instead of 35%. The decay
|
|
4
|
+
* curve shape (linear 1 - ageMs/300s) unchanged.
|
|
5
|
+
*
|
|
6
|
+
* Contract:
|
|
7
|
+
* - Source-file verification of `Math.max(0.40, 1 - ageMs / (5 * 60 * 1000))`
|
|
8
|
+
* - DOM probe: seed a STALE edge (ageMs > 5min) so fresh hits the
|
|
9
|
+
* floor; assert the visible flow path's opacity reflects the
|
|
10
|
+
* lifted floor (computed as 0.40 × edge_opacity_mul × base).
|
|
11
|
+
* - Compare against pre-R406 expectation (0.35 floor): the lifted
|
|
12
|
+
* floor produces ~14.3 % higher edge opacity on stale edges.
|
|
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
|
+
// 10 minutes ago — well past the 5-min decay window; freshness hits floor.
|
|
20
|
+
const stale = new Date(Date.now() - 10 * 60 * 1000).toISOString();
|
|
21
|
+
|
|
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(() => {
|
|
26
|
+
try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
|
|
27
|
+
});
|
|
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
|
+
// Seed a stale flow link 10 minutes ago — past the 5-min decay so floor binds.
|
|
40
|
+
await ctx.route('**/api/hub/messages*', (r) => r.fulfill({
|
|
41
|
+
json: { messages: [
|
|
42
|
+
{ id: 'm1', from_alias: 'alpha', to_alias: 'beta', content: 'old', created_at: stale, network_id: 'default' },
|
|
43
|
+
] },
|
|
44
|
+
}));
|
|
45
|
+
await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
|
|
46
|
+
|
|
47
|
+
const page = await ctx.newPage();
|
|
48
|
+
await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
|
|
49
|
+
await page.waitForSelector('[data-edge-visible]', { timeout: 15000 });
|
|
50
|
+
await page.waitForTimeout(600);
|
|
51
|
+
|
|
52
|
+
const probe = await page.evaluate(() => {
|
|
53
|
+
const edge = document.querySelector('[data-edge-visible]');
|
|
54
|
+
if (!edge) return null;
|
|
55
|
+
return {
|
|
56
|
+
opacityAttr: edge.getAttribute('opacity'),
|
|
57
|
+
edgeKey: edge.getAttribute('data-edge-visible'),
|
|
58
|
+
};
|
|
59
|
+
});
|
|
60
|
+
|
|
61
|
+
const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
|
|
62
|
+
const sourceHasFloor040 = /Math\.max\(0\.40, 1 - ageMs/.test(fileText);
|
|
63
|
+
const sourceLacksFloor035 = !/Math\.max\(0\.35, 1 - ageMs/.test(fileText);
|
|
64
|
+
|
|
65
|
+
await browser.close();
|
|
66
|
+
|
|
67
|
+
// Pre-R406 expected: (0.28 cyber base) × 0.35 floor × 1 mul = 0.098
|
|
68
|
+
// Post-R406 expected: (0.28 cyber base) × 0.40 floor × 1 mul = 0.112
|
|
69
|
+
// On the visible flow path the wire is:
|
|
70
|
+
// opacity = Math.min(1, (isLight ? 0.22 : 0.28) * fresh * edgeOpacityMul)
|
|
71
|
+
// For an unhovered edge in a fresh sessions context (no other hover/active):
|
|
72
|
+
// edgeOpacityMul = !hoveredAlias && !activeGroup ? (hoveredActiveLinks ? 1.5 : 1) : 0.35
|
|
73
|
+
// = 1 in this probe → opacity ≈ 0.112 post-R406
|
|
74
|
+
// Browser may serialize this as "0.112" or "0.11" — accept either with
|
|
75
|
+
// a tolerance probe.
|
|
76
|
+
const opacityNum = probe ? parseFloat(probe.opacityAttr || '0') : 0;
|
|
77
|
+
const expectedPost = 0.28 * 0.40 * 1; // 0.112
|
|
78
|
+
const expectedPre = 0.28 * 0.35 * 1; // 0.098
|
|
79
|
+
const distToPost = Math.abs(opacityNum - expectedPost);
|
|
80
|
+
const distToPre = Math.abs(opacityNum - expectedPre);
|
|
81
|
+
|
|
82
|
+
const results = {
|
|
83
|
+
// Edge mounted from seeded stale flow
|
|
84
|
+
edge_mounted: !!probe,
|
|
85
|
+
edge_key_present: !!probe?.edgeKey,
|
|
86
|
+
// Post-R406 opacity is CLOSER to 0.112 than 0.098
|
|
87
|
+
opacity_close_to_post_r406: distToPost < 0.01,
|
|
88
|
+
// Opacity is FURTHER from the pre-R406 0.098 baseline
|
|
89
|
+
opacity_not_pre_r406: distToPre > distToPost,
|
|
90
|
+
// Source-file canonical wire
|
|
91
|
+
source_floor_040: sourceHasFloor040,
|
|
92
|
+
source_no_floor_035: sourceLacksFloor035,
|
|
93
|
+
};
|
|
94
|
+
const ok = Object.values(results).every(Boolean);
|
|
95
|
+
console.log(`${ok ? '✅' : '❌'} edge freshness floor 0.35 → 0.40:`, JSON.stringify(results),
|
|
96
|
+
'\n probe:', probe,
|
|
97
|
+
'\n opacityNum:', opacityNum, '/ expectedPost:', expectedPost, '/ distToPost:', distToPost.toFixed(4),
|
|
98
|
+
'\n expectedPre:', expectedPre, '/ distToPre:', distToPre.toFixed(4));
|
|
99
|
+
process.exit(ok ? 0 : 1);
|