@sleep2agi/agent-network-dashboard 0.5.1-preview.99 → 0.5.2-preview.0

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.
Files changed (203) hide show
  1. package/.next/BUILD_ID +1 -1
  2. package/.next/build-manifest.json +3 -3
  3. package/.next/diagnostics/route-bundle-stats.json +32 -32
  4. package/.next/fallback-build-manifest.json +3 -3
  5. package/.next/prerender-manifest.json +3 -3
  6. package/.next/server/app/_global-error.html +1 -1
  7. package/.next/server/app/_global-error.rsc +1 -1
  8. package/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +1 -1
  9. package/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  10. package/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  11. package/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  12. package/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  13. package/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  14. package/.next/server/app/_not-found.html +2 -2
  15. package/.next/server/app/_not-found.rsc +12 -12
  16. package/.next/server/app/_not-found.segments/_full.segment.rsc +12 -12
  17. package/.next/server/app/_not-found.segments/_head.segment.rsc +4 -4
  18. package/.next/server/app/_not-found.segments/_index.segment.rsc +7 -7
  19. package/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +2 -2
  20. package/.next/server/app/_not-found.segments/_not-found.segment.rsc +3 -3
  21. package/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
  22. package/.next/server/app/admin/page_client-reference-manifest.js +1 -1
  23. package/.next/server/app/admin.html +2 -2
  24. package/.next/server/app/admin.rsc +14 -14
  25. package/.next/server/app/admin.segments/_full.segment.rsc +14 -14
  26. package/.next/server/app/admin.segments/_head.segment.rsc +4 -4
  27. package/.next/server/app/admin.segments/_index.segment.rsc +7 -7
  28. package/.next/server/app/admin.segments/_tree.segment.rsc +2 -2
  29. package/.next/server/app/admin.segments/admin/__PAGE__.segment.rsc +4 -4
  30. package/.next/server/app/admin.segments/admin.segment.rsc +3 -3
  31. package/.next/server/app/index.html +2 -2
  32. package/.next/server/app/index.rsc +14 -14
  33. package/.next/server/app/index.segments/__PAGE__.segment.rsc +4 -4
  34. package/.next/server/app/index.segments/_full.segment.rsc +14 -14
  35. package/.next/server/app/index.segments/_head.segment.rsc +4 -4
  36. package/.next/server/app/index.segments/_index.segment.rsc +7 -7
  37. package/.next/server/app/index.segments/_tree.segment.rsc +2 -2
  38. package/.next/server/app/login/page_client-reference-manifest.js +1 -1
  39. package/.next/server/app/login.html +2 -2
  40. package/.next/server/app/login.rsc +14 -14
  41. package/.next/server/app/login.segments/_full.segment.rsc +14 -14
  42. package/.next/server/app/login.segments/_head.segment.rsc +4 -4
  43. package/.next/server/app/login.segments/_index.segment.rsc +7 -7
  44. package/.next/server/app/login.segments/_tree.segment.rsc +2 -2
  45. package/.next/server/app/login.segments/login/__PAGE__.segment.rsc +4 -4
  46. package/.next/server/app/login.segments/login.segment.rsc +3 -3
  47. package/.next/server/app/logs/page_client-reference-manifest.js +1 -1
  48. package/.next/server/app/logs.html +2 -2
  49. package/.next/server/app/logs.rsc +14 -14
  50. package/.next/server/app/logs.segments/_full.segment.rsc +14 -14
  51. package/.next/server/app/logs.segments/_head.segment.rsc +4 -4
  52. package/.next/server/app/logs.segments/_index.segment.rsc +7 -7
  53. package/.next/server/app/logs.segments/_tree.segment.rsc +2 -2
  54. package/.next/server/app/logs.segments/logs/__PAGE__.segment.rsc +4 -4
  55. package/.next/server/app/logs.segments/logs.segment.rsc +3 -3
  56. package/.next/server/app/messages/page_client-reference-manifest.js +1 -1
  57. package/.next/server/app/messages.html +2 -2
  58. package/.next/server/app/messages.rsc +14 -14
  59. package/.next/server/app/messages.segments/_full.segment.rsc +14 -14
  60. package/.next/server/app/messages.segments/_head.segment.rsc +4 -4
  61. package/.next/server/app/messages.segments/_index.segment.rsc +7 -7
  62. package/.next/server/app/messages.segments/_tree.segment.rsc +2 -2
  63. package/.next/server/app/messages.segments/messages/__PAGE__.segment.rsc +4 -4
  64. package/.next/server/app/messages.segments/messages.segment.rsc +3 -3
  65. package/.next/server/app/node/page_client-reference-manifest.js +1 -1
  66. package/.next/server/app/node.html +2 -2
  67. package/.next/server/app/node.rsc +14 -14
  68. package/.next/server/app/node.segments/_full.segment.rsc +14 -14
  69. package/.next/server/app/node.segments/_head.segment.rsc +4 -4
  70. package/.next/server/app/node.segments/_index.segment.rsc +7 -7
  71. package/.next/server/app/node.segments/_tree.segment.rsc +2 -2
  72. package/.next/server/app/node.segments/node/__PAGE__.segment.rsc +4 -4
  73. package/.next/server/app/node.segments/node.segment.rsc +3 -3
  74. package/.next/server/app/nodes/page_client-reference-manifest.js +1 -1
  75. package/.next/server/app/nodes.html +2 -2
  76. package/.next/server/app/nodes.rsc +14 -14
  77. package/.next/server/app/nodes.segments/_full.segment.rsc +14 -14
  78. package/.next/server/app/nodes.segments/_head.segment.rsc +4 -4
  79. package/.next/server/app/nodes.segments/_index.segment.rsc +7 -7
  80. package/.next/server/app/nodes.segments/_tree.segment.rsc +2 -2
  81. package/.next/server/app/nodes.segments/nodes/__PAGE__.segment.rsc +4 -4
  82. package/.next/server/app/nodes.segments/nodes.segment.rsc +3 -3
  83. package/.next/server/app/page.js.nft.json +1 -1
  84. package/.next/server/app/page_client-reference-manifest.js +1 -1
  85. package/.next/server/app/server-logs/page_client-reference-manifest.js +1 -1
  86. package/.next/server/app/server-logs.html +2 -2
  87. package/.next/server/app/server-logs.rsc +14 -14
  88. package/.next/server/app/server-logs.segments/_full.segment.rsc +14 -14
  89. package/.next/server/app/server-logs.segments/_head.segment.rsc +4 -4
  90. package/.next/server/app/server-logs.segments/_index.segment.rsc +7 -7
  91. package/.next/server/app/server-logs.segments/_tree.segment.rsc +2 -2
  92. package/.next/server/app/server-logs.segments/server-logs/__PAGE__.segment.rsc +4 -4
  93. package/.next/server/app/server-logs.segments/server-logs.segment.rsc +3 -3
  94. package/.next/server/app/settings/networks/page_client-reference-manifest.js +1 -1
  95. package/.next/server/app/settings/networks.html +2 -2
  96. package/.next/server/app/settings/networks.rsc +14 -14
  97. package/.next/server/app/settings/networks.segments/_full.segment.rsc +14 -14
  98. package/.next/server/app/settings/networks.segments/_head.segment.rsc +4 -4
  99. package/.next/server/app/settings/networks.segments/_index.segment.rsc +7 -7
  100. package/.next/server/app/settings/networks.segments/_tree.segment.rsc +2 -2
  101. package/.next/server/app/settings/networks.segments/settings/networks/__PAGE__.segment.rsc +4 -4
  102. package/.next/server/app/settings/networks.segments/settings/networks.segment.rsc +3 -3
  103. package/.next/server/app/settings/networks.segments/settings.segment.rsc +3 -3
  104. package/.next/server/app/settings/page_client-reference-manifest.js +1 -1
  105. package/.next/server/app/settings/tokens/page_client-reference-manifest.js +1 -1
  106. package/.next/server/app/settings/tokens.html +2 -2
  107. package/.next/server/app/settings/tokens.rsc +14 -14
  108. package/.next/server/app/settings/tokens.segments/_full.segment.rsc +14 -14
  109. package/.next/server/app/settings/tokens.segments/_head.segment.rsc +4 -4
  110. package/.next/server/app/settings/tokens.segments/_index.segment.rsc +7 -7
  111. package/.next/server/app/settings/tokens.segments/_tree.segment.rsc +2 -2
  112. package/.next/server/app/settings/tokens.segments/settings/tokens/__PAGE__.segment.rsc +4 -4
  113. package/.next/server/app/settings/tokens.segments/settings/tokens.segment.rsc +3 -3
  114. package/.next/server/app/settings/tokens.segments/settings.segment.rsc +3 -3
  115. package/.next/server/app/settings.html +2 -2
  116. package/.next/server/app/settings.rsc +14 -14
  117. package/.next/server/app/settings.segments/_full.segment.rsc +14 -14
  118. package/.next/server/app/settings.segments/_head.segment.rsc +4 -4
  119. package/.next/server/app/settings.segments/_index.segment.rsc +7 -7
  120. package/.next/server/app/settings.segments/_tree.segment.rsc +2 -2
  121. package/.next/server/app/settings.segments/settings/__PAGE__.segment.rsc +4 -4
  122. package/.next/server/app/settings.segments/settings.segment.rsc +3 -3
  123. package/.next/server/app/tasks/[id]/page_client-reference-manifest.js +1 -1
  124. package/.next/server/app/tasks/page_client-reference-manifest.js +1 -1
  125. package/.next/server/app/tasks.html +2 -2
  126. package/.next/server/app/tasks.rsc +14 -14
  127. package/.next/server/app/tasks.segments/_full.segment.rsc +14 -14
  128. package/.next/server/app/tasks.segments/_head.segment.rsc +4 -4
  129. package/.next/server/app/tasks.segments/_index.segment.rsc +7 -7
  130. package/.next/server/app/tasks.segments/_tree.segment.rsc +2 -2
  131. package/.next/server/app/tasks.segments/tasks/__PAGE__.segment.rsc +4 -4
  132. package/.next/server/app/tasks.segments/tasks.segment.rsc +3 -3
  133. package/.next/server/chunks/ssr/{[root-of-the-server]__03b.f76._.js → [root-of-the-server]__0sv~g.o._.js} +2 -2
  134. package/.next/server/chunks/ssr/[root-of-the-server]__0sv~g.o._.js.map +1 -0
  135. package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js +3 -3
  136. package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js.map +1 -1
  137. package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js +1 -1
  138. package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js.map +1 -1
  139. package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js +1 -1
  140. package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js.map +1 -1
  141. package/.next/server/chunks/ssr/agent-network-dashboard_app_components_0mvyi-4._.js +1 -1
  142. package/.next/server/chunks/ssr/agent-network-dashboard_app_components_0mvyi-4._.js.map +1 -1
  143. package/.next/server/middleware-build-manifest.js +3 -3
  144. package/.next/server/pages/404.html +2 -2
  145. package/.next/server/pages/500.html +1 -1
  146. package/.next/server/server-reference-manifest.js +1 -1
  147. package/.next/server/server-reference-manifest.json +1 -1
  148. package/.next/static/chunks/{0l_~q07bhpkcx.js → 03a4--7ncekmk.js} +1 -1
  149. package/.next/static/chunks/0a4hmfvj-81x5.css +2 -0
  150. package/.next/static/chunks/0caxil0dw-oe9.js +4 -0
  151. package/.next/static/chunks/{03~~oirxz7~vc.js → 0p8xwrfjtykvn.js} +1 -1
  152. package/.next/static/chunks/0x.m3vy8e5iit.js +1 -0
  153. package/.next/static/chunks/12gq1w9k_7v06.js +1 -0
  154. package/.next/trace +2 -2
  155. package/.next/trace-build +1 -1
  156. package/app/components/ServersDrawer.tsx +17 -0
  157. package/app/components/TopoGraph.tsx +770 -76
  158. package/package.json +1 -1
  159. package/screenshots/v0.10.2-disk-verify/disk-render-full.png +0 -0
  160. package/screenshots/v0.10.2-disk-verify/disk-render.png +0 -0
  161. package/screenshots/v0.11.0-147/after.png +0 -0
  162. package/scripts/p0-147-screenshot.mjs +53 -0
  163. package/scripts/p0-servers-drawer-screenshot.mjs +2 -0
  164. package/scripts/topo-any-hover-attr-test.mjs +83 -0
  165. package/scripts/topo-any-pinned-attr-test.mjs +86 -0
  166. package/scripts/topo-chrome-fullscreen-icon-sw-test.mjs +92 -0
  167. package/scripts/topo-chrome-reset-icon-sw-test.mjs +80 -0
  168. package/scripts/topo-chrome-zoom-icon-sw-test.mjs +90 -0
  169. package/scripts/topo-dashboard-version-attr-test.mjs +69 -0
  170. package/scripts/topo-dense-alias-opacity-test.mjs +68 -0
  171. package/scripts/topo-edge-particle-hover-r-test.mjs +113 -0
  172. package/scripts/topo-endpoint-ring-r-hover-test.mjs +89 -0
  173. package/scripts/topo-fleet-count-attrs-test.mjs +87 -0
  174. package/scripts/topo-group-box-geom-transition-test.mjs +110 -0
  175. package/scripts/topo-group-box-rx-pin-test.mjs +103 -0
  176. package/scripts/topo-group-label-count-fw-test.mjs +100 -0
  177. package/scripts/topo-group-label-fw-pin-test.mjs +99 -0
  178. package/scripts/topo-group-label-tint-geom-test.mjs +94 -0
  179. package/scripts/topo-group-label-tint-transition-test.mjs +97 -0
  180. package/scripts/topo-group-pip-fontsize-test.mjs +106 -0
  181. package/scripts/topo-group-tier-attr-test.mjs +84 -0
  182. package/scripts/topo-group-tint-rx-pin-test.mjs +107 -0
  183. package/scripts/topo-hub-core-fill-hover-test.mjs +85 -0
  184. package/scripts/topo-hub-halo-r-hover-test.mjs +82 -0
  185. package/scripts/topo-legend-count-active-opacity-test.mjs +102 -0
  186. package/scripts/topo-legend-count-pin-fw-test.mjs +90 -0
  187. package/scripts/topo-minimap-viewport-opacity-test.mjs +96 -0
  188. package/scripts/topo-node-halo-hover-opacity-test.mjs +104 -0
  189. package/scripts/topo-node-halo-light-offline-test.mjs +80 -0
  190. package/scripts/topo-node-sub-text-fw-test.mjs +75 -0
  191. package/scripts/topo-overlap-stale-build-guard-test.mjs +66 -0
  192. package/scripts/topo-overlap-test.mjs +42 -1
  193. package/scripts/topo-recent-row-count-pin-fw-test.mjs +106 -0
  194. package/scripts/topo-recent-row-pip-hover-r-test.mjs +104 -0
  195. package/scripts/topo-runtime-icon-hover-test.mjs +96 -0
  196. package/.next/server/chunks/ssr/[root-of-the-server]__03b.f76._.js.map +0 -1
  197. package/.next/static/chunks/0.sf46gnv4wwm.js +0 -1
  198. package/.next/static/chunks/017hq2-5l~_98.css +0 -2
  199. package/.next/static/chunks/0_igeywsok2_-.js +0 -1
  200. package/.next/static/chunks/0igz0bww16uvc.js +0 -4
  201. /package/.next/static/{4rHxTzLwe2XC9M1rN-MpJ → 5IMzNtH5S5Oqe-FCw1CX4}/_buildManifest.js +0 -0
  202. /package/.next/static/{4rHxTzLwe2XC9M1rN-MpJ → 5IMzNtH5S5Oqe-FCw1CX4}/_clientMiddlewareManifest.js +0 -0
  203. /package/.next/static/{4rHxTzLwe2XC9M1rN-MpJ → 5IMzNtH5S5Oqe-FCw1CX4}/_ssgManifest.js +0 -0
@@ -0,0 +1,99 @@
1
+ /* Round 457 verification: group label parent text fontWeight 700 → 800
2
+ * on isPinned. Adds typographic weight axis at parent-text scope.
3
+ *
4
+ * Contract:
5
+ * - rest: every group-label parent <text> reports font-weight '700' +
6
+ * pinned='false'
7
+ * - click a group-label-hit: that group's <text> flips to '800' +
8
+ * pinned='true'; siblings stay rest
9
+ */
10
+ import { chromium } from 'playwright';
11
+ import { readFileSync } from 'node:fs';
12
+
13
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
14
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
15
+
16
+ const browser = await chromium.launch({ headless: true });
17
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
18
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
19
+ await ctx.addInitScript(() => {
20
+ try {
21
+ localStorage.setItem('anet-theme', 'cyber');
22
+ localStorage.setItem('anet-topo-layout', 'grid');
23
+ sessionStorage.setItem('anet_v3_auth', '1');
24
+ } 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
+ await route.fulfill({ response: r, json: { ...b, sessions: [
36
+ mk('alpha·1', 'working'),
37
+ mk('alpha·2', 'idle'),
38
+ mk('beta·1', 'working'),
39
+ mk('beta·2', 'idle'),
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-group-label]', { timeout: 15000 });
48
+ await page.waitForTimeout(400);
49
+
50
+ const readAll = () => page.evaluate(() => {
51
+ const ts = [...document.querySelectorAll('[data-group-label]')];
52
+ return ts.map(t => ({
53
+ key: t.getAttribute('data-group-label'),
54
+ fw: t.getAttribute('font-weight'),
55
+ fw_d: t.getAttribute('data-group-label-font-weight'),
56
+ pinned: t.getAttribute('data-group-label-pinned'),
57
+ }));
58
+ });
59
+
60
+ const rest = await readAll();
61
+ const firstKey = rest[0]?.key;
62
+
63
+ // Pin first group by clicking its hit-area
64
+ let pinned = null;
65
+ if (firstKey) {
66
+ const hit = await page.$(`[data-group-label-hit="${firstKey}"]`);
67
+ if (hit) {
68
+ await hit.click();
69
+ await page.waitForTimeout(250);
70
+ pinned = await readAll();
71
+ }
72
+ }
73
+
74
+ const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
75
+ const sourceWired = /fontWeight=\{isPinned \? '800' : '700'\}/.test(fileText);
76
+
77
+ await browser.close();
78
+
79
+ const restAll700 = rest.every(r => r.fw === '700');
80
+ const restNoPin = rest.every(r => r.pinned === 'false');
81
+ const pinnedEntry = pinned?.find(r => r.key === firstKey);
82
+ const pinFw800 = pinnedEntry?.fw === '800';
83
+ const pinFlag = pinnedEntry?.pinned === 'true';
84
+ const othersRest = pinned ? pinned.filter(r => r.key !== firstKey).every(r => r.fw === '700' && r.pinned === 'false') : false;
85
+
86
+ const results = {
87
+ rest_count_ge_2: rest.length >= 2,
88
+ rest_all_fw_700: restAll700,
89
+ rest_no_pin: restNoPin,
90
+ pinned_target_fw_800: pinFw800,
91
+ pinned_target_flag: pinFlag,
92
+ pinned_others_stay: othersRest,
93
+ source_wired: sourceWired,
94
+ };
95
+ const ok = Object.values(results).every(Boolean);
96
+ console.log(`${ok ? '✅' : '❌'} group label fw pin:`, JSON.stringify(results),
97
+ '\n rest:', JSON.stringify(rest),
98
+ '\n pinned target:', JSON.stringify(pinnedEntry));
99
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,94 @@
1
+ /* Round 460 verification: group-label tint rect transition list
2
+ * extends to include `x 200ms, width 200ms`. Pre-R460 the hitbox
3
+ * snap-jumped when a cluster grew/shrunk (member joins or leaves
4
+ * re-priced box.x / box.w); R460 tweens both geometry axes so
5
+ * the rect slides into the new bounds matching codex preview.125
6
+ * + R459 200ms motion vocabulary.
7
+ *
8
+ * Contract:
9
+ * - every <rect data-group-label-tint-geom-transition="x,width">
10
+ * renders inside the group cluster boundary
11
+ * - inline style includes 'x 200ms ease-out' AND 'width 200ms
12
+ * ease-out' alongside the R459 fill+opacity transitions
13
+ * - source-file conditional wired
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
+ const browser = await chromium.launch({ headless: true });
22
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1200 } });
23
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
24
+ await ctx.addInitScript(() => {
25
+ try {
26
+ localStorage.setItem('anet-theme', 'cyber');
27
+ localStorage.setItem('anet-topo-layout', 'grid');
28
+ sessionStorage.setItem('anet_v3_auth', '1');
29
+ } catch {}
30
+ });
31
+ await ctx.route('**/api/hub/status*', async (route) => {
32
+ const r = await route.fetch();
33
+ const b = await r.json();
34
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
35
+ const mk = (alias, status) => ({
36
+ alias, status, model: 'claude-opus-4', runtime: 'claude-code-cli',
37
+ network_id: nid, project_dir: null,
38
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
39
+ });
40
+ await route.fulfill({ response: r, json: { ...b, sessions: [
41
+ mk('alpha·1', 'working'),
42
+ mk('alpha·2', 'idle'),
43
+ mk('beta·1', 'working'),
44
+ mk('beta·2', 'idle'),
45
+ ] } });
46
+ });
47
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
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-group-label]', { timeout: 15000 });
53
+ await page.waitForTimeout(400);
54
+
55
+ const probe = await page.evaluate(() => {
56
+ const tints = [...document.querySelectorAll('[data-group-label-tint-geom-transition]')];
57
+ return {
58
+ count: tints.length,
59
+ nodes: tints.map(t => ({
60
+ geom: t.getAttribute('data-group-label-tint-geom-transition'),
61
+ style: t.getAttribute('style') || '',
62
+ })),
63
+ };
64
+ });
65
+
66
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
67
+ const sourceHasGeom = /data-group-label-tint-geom-transition="x,width"/.test(src);
68
+ const sourceHasXTween = /x 200ms ease-out/.test(src);
69
+ const sourceHasWidthTween = /width 200ms ease-out/.test(src);
70
+
71
+ await browser.close();
72
+
73
+ const countGe2 = probe.count >= 2;
74
+ const allDataAttrXW = probe.nodes.every(n => n.geom === 'x,width');
75
+ const allStyleHasX = probe.nodes.every(n => /x 200ms ease-out/.test(n.style));
76
+ const allStyleHasW = probe.nodes.every(n => /width 200ms ease-out/.test(n.style));
77
+ const allStylePreservesFillOpacity = probe.nodes.every(n =>
78
+ /fill 200ms ease-out/.test(n.style) && /opacity 200ms ease-out/.test(n.style));
79
+
80
+ const results = {
81
+ tint_count_ge_2: countGe2,
82
+ all_data_attr_xw: allDataAttrXW,
83
+ all_style_has_x: allStyleHasX,
84
+ all_style_has_width: allStyleHasW,
85
+ all_style_preserves_fill_opacity: allStylePreservesFillOpacity,
86
+ source_geom_attr_wired: sourceHasGeom,
87
+ source_x_tween_wired: sourceHasXTween,
88
+ source_w_tween_wired: sourceHasWidthTween,
89
+ };
90
+ const ok = Object.values(results).every(Boolean);
91
+ console.log(`${ok ? '✅' : '❌'} group label tint geom transition x+width:`, JSON.stringify(results),
92
+ '\n count:', probe.count,
93
+ '\n first style:', probe.nodes[0]?.style);
94
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,97 @@
1
+ /* Round 459 verification: group-label tint hitbox rect transition
2
+ * cadence sync 150ms → 200ms to match codex preview.125 parent
3
+ * <text> transition list. The R107 hitbox rect carried legacy 150ms
4
+ * cadence; codex's #147 P1 lifted the parent label to 200ms across
5
+ * every axis. R459 closes the 50ms desync so tint + label flip as
6
+ * one motion-coherent unit on hover / pin / unpin.
7
+ *
8
+ * Contract:
9
+ * - every <rect data-group-label-tint-transition="200ms">
10
+ * renders inside the group cluster boundary (i.e. there's at
11
+ * least one when fixtures have prefix-group clusters)
12
+ * - source-file style string is 'fill 200ms ease-out, opacity
13
+ * 200ms ease-out' on the rect just below #147 Hero D pivot
14
+ * comment, NOT 'fill 150ms, opacity 150ms'
15
+ * - source-file the data attr `data-group-label-tint-transition`
16
+ * is wired
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: 1200 } });
26
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
27
+ await ctx.addInitScript(() => {
28
+ try {
29
+ localStorage.setItem('anet-theme', 'cyber');
30
+ localStorage.setItem('anet-topo-layout', 'grid');
31
+ sessionStorage.setItem('anet_v3_auth', '1');
32
+ } catch {}
33
+ });
34
+ await ctx.route('**/api/hub/status*', async (route) => {
35
+ const r = await route.fetch();
36
+ const b = await r.json();
37
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
38
+ const mk = (alias, status) => ({
39
+ alias, status, model: 'claude-opus-4', runtime: 'claude-code-cli',
40
+ network_id: nid, project_dir: null,
41
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
42
+ });
43
+ await route.fulfill({ response: r, json: { ...b, sessions: [
44
+ mk('alpha·1', 'working'),
45
+ mk('alpha·2', 'idle'),
46
+ mk('beta·1', 'working'),
47
+ mk('beta·2', 'idle'),
48
+ ] } });
49
+ });
50
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
51
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
52
+
53
+ const page = await ctx.newPage();
54
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
55
+ await page.waitForSelector('[data-group-label]', { timeout: 15000 });
56
+ await page.waitForTimeout(400);
57
+
58
+ const probe = await page.evaluate(() => {
59
+ const tints = [...document.querySelectorAll('[data-group-label-tint-transition]')];
60
+ return {
61
+ count: tints.length,
62
+ transitions: tints.map(t => ({
63
+ value: t.getAttribute('data-group-label-tint-transition'),
64
+ style: t.getAttribute('style') || '',
65
+ computedTransition: getComputedStyle(t).transition,
66
+ })),
67
+ };
68
+ });
69
+
70
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
71
+ const sourceWired200 = /data-group-label-tint-transition="200ms"[\s\S]{0,300}?fill 200ms ease-out, opacity 200ms ease-out/.test(src);
72
+ // negative: there must NOT be a legacy 150ms transition WITHIN the
73
+ // same JSX element as data-group-label-tinted (the hitbox rect).
74
+ // other elements may still legitimately use 150ms (e.g. recent-row,
75
+ // legend-row at lines 8597 / 9458 — those are NOT the hitbox).
76
+ const tintRectBlockMatch = src.match(/data-group-label-tinted=[\s\S]{0,2000}?\/>/);
77
+ const tintRectBlock = tintRectBlockMatch ? tintRectBlockMatch[0] : '';
78
+ const sourceNoLegacy150 = !!tintRectBlock && !tintRectBlock.includes('fill 150ms');
79
+
80
+ await browser.close();
81
+
82
+ const tintCountGe2 = probe.count >= 2;
83
+ const allDataAttr200 = probe.transitions.every(t => t.value === '200ms');
84
+ const allStyle200 = probe.transitions.every(t => /fill 200ms ease-out, opacity 200ms ease-out/.test(t.style));
85
+
86
+ const results = {
87
+ tint_rect_count_ge_2: tintCountGe2,
88
+ all_data_attr_200: allDataAttr200,
89
+ all_style_200ms: allStyle200,
90
+ source_wired_200: sourceWired200,
91
+ source_no_legacy_150: sourceNoLegacy150,
92
+ };
93
+ const ok = Object.values(results).every(Boolean);
94
+ console.log(`${ok ? '✅' : '❌'} group label tint transition 150→200:`, JSON.stringify(results),
95
+ '\n count:', probe.count,
96
+ '\n first tint style:', probe.transitions[0]?.style);
97
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,106 @@
1
+ /* Round 458 verification: status pip strip fontSize 11 → 8 + dx
2
+ * 8/4/4 → 6/3/3, to scale-match the new 9px label that N站牛/codex
3
+ * preview.125 shipped (Hero D #147 Option C: label fontSize 13→9,
4
+ * opacity 0.55 rest / 1 hover+pin, count tspan 11→8). R458 closes
5
+ * the pip-strip leg of the same family.
6
+ *
7
+ * Contract:
8
+ * - every <tspan data-group-pip="working|idle|offline">
9
+ * reports font-size '8' (down from '11')
10
+ * - source-file dx values for the 3 pips are '6'/'3'/'3'
11
+ * (down from '8'/'4'/'4')
12
+ * - parent <text data-group-label> font-size stays '9' (codex
13
+ * preview.125 baseline preserved — R458 is a follow-on, not a
14
+ * replacement)
15
+ * - count tspan font-size stays '8' (codex preview.125 baseline)
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: 1200 } });
25
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
26
+ await ctx.addInitScript(() => {
27
+ try {
28
+ localStorage.setItem('anet-theme', 'cyber');
29
+ localStorage.setItem('anet-topo-layout', 'grid');
30
+ sessionStorage.setItem('anet_v3_auth', '1');
31
+ } catch {}
32
+ });
33
+ // Fixture: 2 prefix groups, each MIXED (so at least one pip
34
+ // renders — R319 drops pips when count===box.count).
35
+ await ctx.route('**/api/hub/status*', async (route) => {
36
+ const r = await route.fetch();
37
+ const b = await r.json();
38
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
39
+ const mk = (alias, status) => ({
40
+ alias, status, model: 'claude-opus-4', runtime: 'claude-code-cli',
41
+ network_id: nid, project_dir: null,
42
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
43
+ });
44
+ await route.fulfill({ response: r, json: { ...b, sessions: [
45
+ mk('alpha·1', 'working'),
46
+ mk('alpha·2', 'idle'),
47
+ mk('alpha·3', 'offline'),
48
+ mk('beta·1', 'working'),
49
+ mk('beta·2', 'idle'),
50
+ ] } });
51
+ });
52
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
53
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
54
+
55
+ const page = await ctx.newPage();
56
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
57
+ await page.waitForSelector('[data-group-label]', { timeout: 15000 });
58
+ await page.waitForTimeout(500);
59
+
60
+ const probe = await page.evaluate(() => {
61
+ const pips = [...document.querySelectorAll('[data-group-pip]')].map(t => ({
62
+ tier: t.getAttribute('data-group-pip'),
63
+ fontSize: t.getAttribute('font-size'),
64
+ text: t.textContent,
65
+ }));
66
+ const labels = [...document.querySelectorAll('[data-group-label]')].map(t => ({
67
+ key: t.getAttribute('data-group-label'),
68
+ fontSize: t.getAttribute('font-size'),
69
+ }));
70
+ const counts = [...document.querySelectorAll('[data-group-label-count]')].map(t => ({
71
+ key: t.getAttribute('data-group-label-count'),
72
+ fontSize: t.getAttribute('font-size'),
73
+ }));
74
+ return { pips, labels, counts };
75
+ });
76
+
77
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
78
+ // dx values for the 3 pips — the source file uses dx="6" then dx="3" twice
79
+ const dxWorking = /data-group-pip="working"[\s\S]{0,400}?dx="6"/.test(src) ||
80
+ /dx="6"[\s\S]{0,400}?data-group-pip="working"/.test(src);
81
+ const dxIdle = /data-group-pip="idle"[\s\S]{0,400}?dx="3"/.test(src) ||
82
+ /dx="3"[\s\S]{0,400}?data-group-pip="idle"/.test(src);
83
+ const dxOffline = /data-group-pip="offline"[\s\S]{0,400}?dx="3"/.test(src) ||
84
+ /dx="3"[\s\S]{0,400}?data-group-pip="offline"/.test(src);
85
+
86
+ await browser.close();
87
+
88
+ const allPipsAre8 = probe.pips.length >= 2 && probe.pips.every(p => p.fontSize === '8');
89
+ const labelStays9 = probe.labels.length >= 2 && probe.labels.every(l => l.fontSize === '9');
90
+ const countStays8 = probe.counts.length >= 2 && probe.counts.every(c => c.fontSize === '8');
91
+
92
+ const results = {
93
+ pips_count_ge_2: probe.pips.length >= 2,
94
+ pips_all_fontsize_8: allPipsAre8,
95
+ labels_stay_9: labelStays9,
96
+ counts_stay_8: countStays8,
97
+ dx_working_is_6: dxWorking,
98
+ dx_idle_is_3: dxIdle,
99
+ dx_offline_is_3: dxOffline,
100
+ };
101
+ const ok = Object.values(results).every(Boolean);
102
+ console.log(`${ok ? '✅' : '❌'} group pip fontSize scale-match:`, JSON.stringify(results),
103
+ '\n pips:', JSON.stringify(probe.pips),
104
+ '\n labels:', JSON.stringify(probe.labels),
105
+ '\n counts:', JSON.stringify(probe.counts));
106
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,84 @@
1
+ /* Round 468 verification: each group <g> carries `data-group-tier`
2
+ * classifier ('all-working' / 'all-idle' / 'all-offline' / 'mixed').
3
+ * Surfaces the semantic the R319 pip-strip implicitly encodes
4
+ * (single-tier groups render `name · count` only) as a queryable
5
+ * DOM attr.
6
+ *
7
+ * Contract:
8
+ * - fixture mixes 2 working + 1 idle in alpha cluster → 'mixed'
9
+ * - fixture has 3 idle in beta cluster → 'all-idle'
10
+ * - source-file conditional wired
11
+ */
12
+ import { chromium } from 'playwright';
13
+ import { readFileSync } from 'node:fs';
14
+
15
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
16
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
17
+
18
+ const browser = await chromium.launch({ headless: true });
19
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1200 } });
20
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
21
+ await ctx.addInitScript(() => {
22
+ try {
23
+ localStorage.setItem('anet-theme', 'cyber');
24
+ localStorage.setItem('anet-topo-layout', 'grid');
25
+ sessionStorage.setItem('anet_v3_auth', '1');
26
+ } 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, status) => ({
33
+ alias, status, 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: [
38
+ // alpha: 2 working + 1 idle → 'mixed'
39
+ mk('alpha·1', 'working'),
40
+ mk('alpha·2', 'working'),
41
+ mk('alpha·3', 'idle'),
42
+ // beta: 3 idle → 'all-idle'
43
+ mk('beta·1', 'idle'),
44
+ mk('beta·2', 'idle'),
45
+ mk('beta·3', 'idle'),
46
+ ] } });
47
+ });
48
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
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-group][data-group-tier]', { timeout: 15000 });
54
+ await page.waitForTimeout(400);
55
+
56
+ const groups = await page.evaluate(() =>
57
+ [...document.querySelectorAll('[data-group][data-group-tier]')].map(g => ({
58
+ key: g.getAttribute('data-group'),
59
+ tier: g.getAttribute('data-group-tier'),
60
+ }))
61
+ );
62
+
63
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
64
+ const sourceClassifier = /groupTier =/.test(src) && /'all-working' :/.test(src) && /'mixed'/.test(src);
65
+ const sourceAttr = /data-group-tier=\{groupTier\}/.test(src);
66
+
67
+ await browser.close();
68
+
69
+ const alpha = groups.find(g => g.key === 'alpha·');
70
+ const beta = groups.find(g => g.key === 'beta·');
71
+ const tierIsValid = (t) => ['all-working', 'all-idle', 'all-offline', 'mixed'].includes(t);
72
+
73
+ const results = {
74
+ groups_count_ge_2: groups.length >= 2,
75
+ alpha_is_mixed: alpha?.tier === 'mixed',
76
+ beta_is_all_idle: beta?.tier === 'all-idle',
77
+ all_tiers_valid: groups.every(g => tierIsValid(g.tier)),
78
+ source_classifier_wired: sourceClassifier,
79
+ source_attr_wired: sourceAttr,
80
+ };
81
+ const ok = Object.values(results).every(Boolean);
82
+ console.log(`${ok ? '✅' : '❌'} group-tier classifier attr:`, JSON.stringify(results),
83
+ '\n groups:', JSON.stringify(groups));
84
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,107 @@
1
+ /* Round 465 verification: hitbox tint rect rx 4 → 5 on
2
+ * pinnedGroup match — mirror R464 (parent group-box rx 14 → 16
3
+ * on isPinned) at the hitbox tier. Pre-R465 the inner hitbox
4
+ * stayed at fixed rx=4 (codex p.125) while the outer container
5
+ * gained pin-tier rx softening; R465 echoes the locked posture
6
+ * at the smaller rect scale.
7
+ *
8
+ * Contract:
9
+ * - every <rect data-group-label-tint-rx> reports rx='4' at rest
10
+ * - clicking the hitbox flips that group's rx to '5'
11
+ * - sibling stays '4'
12
+ * - transition list (R460) extends to include rx 200ms
13
+ * - data-group-label-tint-geom-transition='x,width,rx'
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
+ const browser = await chromium.launch({ headless: true });
22
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1200 } });
23
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
24
+ await ctx.addInitScript(() => {
25
+ try {
26
+ localStorage.setItem('anet-theme', 'cyber');
27
+ localStorage.setItem('anet-topo-layout', 'grid');
28
+ sessionStorage.setItem('anet_v3_auth', '1');
29
+ } catch {}
30
+ });
31
+ await ctx.route('**/api/hub/status*', async (route) => {
32
+ const r = await route.fetch();
33
+ const b = await r.json();
34
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
35
+ const mk = (alias, status) => ({
36
+ alias, status, model: 'claude-opus-4', runtime: 'claude-code-cli',
37
+ network_id: nid, project_dir: null,
38
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
39
+ });
40
+ await route.fulfill({ response: r, json: { ...b, sessions: [
41
+ mk('alpha·1', 'working'),
42
+ mk('alpha·2', 'idle'),
43
+ mk('beta·1', 'working'),
44
+ mk('beta·2', 'idle'),
45
+ ] } });
46
+ });
47
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
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-group-label-tint-rx]', { timeout: 15000 });
53
+ await page.waitForTimeout(400);
54
+
55
+ const readAll = () => page.evaluate(() => {
56
+ const tints = [...document.querySelectorAll('[data-group-label-tint-rx]')];
57
+ return tints.map(t => {
58
+ const parentG = t.closest('g[data-group]');
59
+ return {
60
+ key: parentG ? parentG.getAttribute('data-group') : null,
61
+ rx_attr: t.getAttribute('data-group-label-tint-rx'),
62
+ rx_live: t.getAttribute('rx'),
63
+ geom: t.getAttribute('data-group-label-tint-geom-transition'),
64
+ };
65
+ });
66
+ });
67
+
68
+ const rest = await readAll();
69
+ const firstKey = rest[0]?.key;
70
+
71
+ let pinned = null;
72
+ if (firstKey) {
73
+ const hit = await page.$(`[data-group-label-hit="${firstKey}"]`);
74
+ if (hit) {
75
+ await hit.click();
76
+ await page.waitForTimeout(300);
77
+ pinned = await readAll();
78
+ }
79
+ }
80
+
81
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
82
+ const sourceRxConditional = /rx=\{pinnedGroup === box\.key \? '5' : '4'\}/.test(src);
83
+ const sourceGeomXWidthRx = /data-group-label-tint-geom-transition="x,width,rx"/.test(src);
84
+
85
+ await browser.close();
86
+
87
+ const restCountGe2 = rest.length >= 2;
88
+ const restAllRx4 = rest.every(r => r.rx_attr === '4' && r.rx_live === '4');
89
+ const restGeomXWR = rest.every(r => r.geom === 'x,width,rx');
90
+ const pinnedTarget = pinned?.find(r => r.key === firstKey);
91
+ const pinTargetRx5 = pinnedTarget?.rx_attr === '5' && pinnedTarget?.rx_live === '5';
92
+ const pinSiblings = pinned ? pinned.filter(r => r.key !== firstKey).every(r => r.rx_attr === '4') : false;
93
+
94
+ const results = {
95
+ rest_count_ge_2: restCountGe2,
96
+ rest_all_rx_4: restAllRx4,
97
+ rest_geom_attr_xwrx: restGeomXWR,
98
+ pinned_target_rx_5: pinTargetRx5,
99
+ pinned_siblings_rest: pinSiblings,
100
+ source_rx_conditional: sourceRxConditional,
101
+ source_geom_attr_xwrx: sourceGeomXWidthRx,
102
+ };
103
+ const ok = Object.values(results).every(Boolean);
104
+ console.log(`${ok ? '✅' : '❌'} hitbox tint rx pin 4→5:`, JSON.stringify(results),
105
+ '\n rest:', JSON.stringify(rest),
106
+ '\n pinned target:', JSON.stringify(pinnedTarget));
107
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,85 @@
1
+ /* Round 441 verification: hub center core fill brighten on hoveredHub.
2
+ * Cyber emerald-500 #10b981 → emerald-400 #34d399.
3
+ *
4
+ * Contract:
5
+ * - rest: core fill = '#10b981' + hovered='false'
6
+ * - hover the hub <g>: core fill = '#34d399' + hovered='true'
7
+ */
8
+ import { chromium } from 'playwright';
9
+ import { readFileSync } from 'node:fs';
10
+
11
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
12
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
13
+
14
+ const browser = await chromium.launch({ headless: true });
15
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
16
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
17
+ await ctx.addInitScript(() => {
18
+ try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
19
+ });
20
+ await ctx.route('**/api/hub/status*', async (route) => {
21
+ const r = await route.fetch();
22
+ const b = await r.json();
23
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
24
+ const mk = (alias, status) => ({
25
+ alias, status, model: 'claude-opus-4', runtime: 'claude-code-cli',
26
+ network_id: nid, project_dir: null,
27
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
28
+ });
29
+ await route.fulfill({ response: r, json: { ...b, sessions: [
30
+ mk('alpha', 'working'),
31
+ mk('beta', 'idle'),
32
+ ] } });
33
+ });
34
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
35
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
36
+
37
+ const page = await ctx.newPage();
38
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
39
+ await page.waitForSelector('[data-topo-hub-core]', { timeout: 15000 });
40
+ await page.waitForTimeout(400);
41
+
42
+ const readCore = () => page.evaluate(() => {
43
+ const c = document.querySelector('[data-topo-hub-core]');
44
+ return c ? {
45
+ fill_attr: c.getAttribute('fill'),
46
+ fill_data: c.getAttribute('data-topo-hub-core-fill'),
47
+ hovered: c.getAttribute('data-topo-hub-core-hovered'),
48
+ r: c.getAttribute('r'),
49
+ } : null;
50
+ });
51
+
52
+ const rest = await readCore();
53
+
54
+ const hubBox = await page.evaluate(() => {
55
+ const hub = document.querySelector('[data-topo-hub]');
56
+ if (!hub) return null;
57
+ const b = hub.getBoundingClientRect();
58
+ return { x: b.x + b.width / 2, y: b.y + b.height / 2 };
59
+ });
60
+ let hover = null;
61
+ if (hubBox) {
62
+ await page.mouse.move(hubBox.x, hubBox.y);
63
+ await page.waitForTimeout(300);
64
+ hover = await readCore();
65
+ await page.mouse.move(0, 0);
66
+ }
67
+
68
+ const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
69
+ const sourceWired = /const coreFill = isLight\s*\n\s*\?\s*\(isCoreHovered \? '#10b981' : '#059669'\)\s*\n\s*:\s*\(isCoreHovered \? '#34d399' : '#10b981'\)/.test(fileText);
70
+
71
+ await browser.close();
72
+
73
+ const results = {
74
+ rest_mounted: !!rest,
75
+ rest_fill_emerald_500: rest?.fill_data === '#10b981',
76
+ rest_hovered_false: rest?.hovered === 'false',
77
+ rest_r_10: rest?.r === '10',
78
+ hover_fill_emerald_400: hover?.fill_data === '#34d399',
79
+ hover_hovered_true: hover?.hovered === 'true',
80
+ source_wired: sourceWired,
81
+ };
82
+ const ok = Object.values(results).every(Boolean);
83
+ console.log(`${ok ? '✅' : '❌'} hub core fill hover:`, JSON.stringify(results),
84
+ '\n rest:', rest, '\n hover:', hover);
85
+ process.exit(ok ? 0 : 1);