@sleep2agi/agent-network-dashboard 0.5.1-preview.9 → 0.5.1-preview.90

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (235) 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 +6 -6
  4. package/.next/fallback-build-manifest.json +3 -3
  5. package/.next/server/app/_global-error.html +1 -1
  6. package/.next/server/app/_global-error.rsc +1 -1
  7. package/.next/server/app/_global-error.segments/__PAGE__.segment.rsc +1 -1
  8. package/.next/server/app/_global-error.segments/_full.segment.rsc +1 -1
  9. package/.next/server/app/_global-error.segments/_head.segment.rsc +1 -1
  10. package/.next/server/app/_global-error.segments/_index.segment.rsc +1 -1
  11. package/.next/server/app/_global-error.segments/_tree.segment.rsc +1 -1
  12. package/.next/server/app/_not-found/page_client-reference-manifest.js +1 -1
  13. package/.next/server/app/_not-found.html +2 -2
  14. package/.next/server/app/_not-found.rsc +2 -2
  15. package/.next/server/app/_not-found.segments/_full.segment.rsc +2 -2
  16. package/.next/server/app/_not-found.segments/_head.segment.rsc +1 -1
  17. package/.next/server/app/_not-found.segments/_index.segment.rsc +2 -2
  18. package/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +1 -1
  19. package/.next/server/app/_not-found.segments/_not-found.segment.rsc +1 -1
  20. package/.next/server/app/_not-found.segments/_tree.segment.rsc +2 -2
  21. package/.next/server/app/admin/page_client-reference-manifest.js +1 -1
  22. package/.next/server/app/admin.html +2 -2
  23. package/.next/server/app/admin.rsc +2 -2
  24. package/.next/server/app/admin.segments/_full.segment.rsc +2 -2
  25. package/.next/server/app/admin.segments/_head.segment.rsc +1 -1
  26. package/.next/server/app/admin.segments/_index.segment.rsc +2 -2
  27. package/.next/server/app/admin.segments/_tree.segment.rsc +2 -2
  28. package/.next/server/app/admin.segments/admin/__PAGE__.segment.rsc +1 -1
  29. package/.next/server/app/admin.segments/admin.segment.rsc +1 -1
  30. package/.next/server/app/index.html +2 -2
  31. package/.next/server/app/index.rsc +3 -3
  32. package/.next/server/app/index.segments/__PAGE__.segment.rsc +2 -2
  33. package/.next/server/app/index.segments/_full.segment.rsc +3 -3
  34. package/.next/server/app/index.segments/_head.segment.rsc +1 -1
  35. package/.next/server/app/index.segments/_index.segment.rsc +2 -2
  36. package/.next/server/app/index.segments/_tree.segment.rsc +2 -2
  37. package/.next/server/app/login/page_client-reference-manifest.js +1 -1
  38. package/.next/server/app/login.html +2 -2
  39. package/.next/server/app/login.rsc +3 -3
  40. package/.next/server/app/login.segments/_full.segment.rsc +3 -3
  41. package/.next/server/app/login.segments/_head.segment.rsc +1 -1
  42. package/.next/server/app/login.segments/_index.segment.rsc +2 -2
  43. package/.next/server/app/login.segments/_tree.segment.rsc +2 -2
  44. package/.next/server/app/login.segments/login/__PAGE__.segment.rsc +2 -2
  45. package/.next/server/app/login.segments/login.segment.rsc +1 -1
  46. package/.next/server/app/logs/page_client-reference-manifest.js +1 -1
  47. package/.next/server/app/logs.html +2 -2
  48. package/.next/server/app/logs.rsc +2 -2
  49. package/.next/server/app/logs.segments/_full.segment.rsc +2 -2
  50. package/.next/server/app/logs.segments/_head.segment.rsc +1 -1
  51. package/.next/server/app/logs.segments/_index.segment.rsc +2 -2
  52. package/.next/server/app/logs.segments/_tree.segment.rsc +2 -2
  53. package/.next/server/app/logs.segments/logs/__PAGE__.segment.rsc +1 -1
  54. package/.next/server/app/logs.segments/logs.segment.rsc +1 -1
  55. package/.next/server/app/messages/page_client-reference-manifest.js +1 -1
  56. package/.next/server/app/messages.html +2 -2
  57. package/.next/server/app/messages.rsc +2 -2
  58. package/.next/server/app/messages.segments/_full.segment.rsc +2 -2
  59. package/.next/server/app/messages.segments/_head.segment.rsc +1 -1
  60. package/.next/server/app/messages.segments/_index.segment.rsc +2 -2
  61. package/.next/server/app/messages.segments/_tree.segment.rsc +2 -2
  62. package/.next/server/app/messages.segments/messages/__PAGE__.segment.rsc +1 -1
  63. package/.next/server/app/messages.segments/messages.segment.rsc +1 -1
  64. package/.next/server/app/node/page_client-reference-manifest.js +1 -1
  65. package/.next/server/app/node.html +2 -2
  66. package/.next/server/app/node.rsc +2 -2
  67. package/.next/server/app/node.segments/_full.segment.rsc +2 -2
  68. package/.next/server/app/node.segments/_head.segment.rsc +1 -1
  69. package/.next/server/app/node.segments/_index.segment.rsc +2 -2
  70. package/.next/server/app/node.segments/_tree.segment.rsc +2 -2
  71. package/.next/server/app/node.segments/node/__PAGE__.segment.rsc +1 -1
  72. package/.next/server/app/node.segments/node.segment.rsc +1 -1
  73. package/.next/server/app/nodes/page_client-reference-manifest.js +1 -1
  74. package/.next/server/app/nodes.html +2 -2
  75. package/.next/server/app/nodes.rsc +2 -2
  76. package/.next/server/app/nodes.segments/_full.segment.rsc +2 -2
  77. package/.next/server/app/nodes.segments/_head.segment.rsc +1 -1
  78. package/.next/server/app/nodes.segments/_index.segment.rsc +2 -2
  79. package/.next/server/app/nodes.segments/_tree.segment.rsc +2 -2
  80. package/.next/server/app/nodes.segments/nodes/__PAGE__.segment.rsc +1 -1
  81. package/.next/server/app/nodes.segments/nodes.segment.rsc +1 -1
  82. package/.next/server/app/page_client-reference-manifest.js +1 -1
  83. package/.next/server/app/server-logs/page_client-reference-manifest.js +1 -1
  84. package/.next/server/app/server-logs.html +2 -2
  85. package/.next/server/app/server-logs.rsc +2 -2
  86. package/.next/server/app/server-logs.segments/_full.segment.rsc +2 -2
  87. package/.next/server/app/server-logs.segments/_head.segment.rsc +1 -1
  88. package/.next/server/app/server-logs.segments/_index.segment.rsc +2 -2
  89. package/.next/server/app/server-logs.segments/_tree.segment.rsc +2 -2
  90. package/.next/server/app/server-logs.segments/server-logs/__PAGE__.segment.rsc +1 -1
  91. package/.next/server/app/server-logs.segments/server-logs.segment.rsc +1 -1
  92. package/.next/server/app/settings/networks/page_client-reference-manifest.js +1 -1
  93. package/.next/server/app/settings/networks.html +2 -2
  94. package/.next/server/app/settings/networks.rsc +2 -2
  95. package/.next/server/app/settings/networks.segments/_full.segment.rsc +2 -2
  96. package/.next/server/app/settings/networks.segments/_head.segment.rsc +1 -1
  97. package/.next/server/app/settings/networks.segments/_index.segment.rsc +2 -2
  98. package/.next/server/app/settings/networks.segments/_tree.segment.rsc +2 -2
  99. package/.next/server/app/settings/networks.segments/settings/networks/__PAGE__.segment.rsc +1 -1
  100. package/.next/server/app/settings/networks.segments/settings/networks.segment.rsc +1 -1
  101. package/.next/server/app/settings/networks.segments/settings.segment.rsc +1 -1
  102. package/.next/server/app/settings/page_client-reference-manifest.js +1 -1
  103. package/.next/server/app/settings/tokens/page_client-reference-manifest.js +1 -1
  104. package/.next/server/app/settings/tokens.html +2 -2
  105. package/.next/server/app/settings/tokens.rsc +2 -2
  106. package/.next/server/app/settings/tokens.segments/_full.segment.rsc +2 -2
  107. package/.next/server/app/settings/tokens.segments/_head.segment.rsc +1 -1
  108. package/.next/server/app/settings/tokens.segments/_index.segment.rsc +2 -2
  109. package/.next/server/app/settings/tokens.segments/_tree.segment.rsc +2 -2
  110. package/.next/server/app/settings/tokens.segments/settings/tokens/__PAGE__.segment.rsc +1 -1
  111. package/.next/server/app/settings/tokens.segments/settings/tokens.segment.rsc +1 -1
  112. package/.next/server/app/settings/tokens.segments/settings.segment.rsc +1 -1
  113. package/.next/server/app/settings.html +2 -2
  114. package/.next/server/app/settings.rsc +3 -3
  115. package/.next/server/app/settings.segments/_full.segment.rsc +3 -3
  116. package/.next/server/app/settings.segments/_head.segment.rsc +1 -1
  117. package/.next/server/app/settings.segments/_index.segment.rsc +2 -2
  118. package/.next/server/app/settings.segments/_tree.segment.rsc +2 -2
  119. package/.next/server/app/settings.segments/settings/__PAGE__.segment.rsc +2 -2
  120. package/.next/server/app/settings.segments/settings.segment.rsc +1 -1
  121. package/.next/server/app/tasks/[id]/page_client-reference-manifest.js +1 -1
  122. package/.next/server/app/tasks/page_client-reference-manifest.js +1 -1
  123. package/.next/server/app/tasks.html +2 -2
  124. package/.next/server/app/tasks.rsc +2 -2
  125. package/.next/server/app/tasks.segments/_full.segment.rsc +2 -2
  126. package/.next/server/app/tasks.segments/_head.segment.rsc +1 -1
  127. package/.next/server/app/tasks.segments/_index.segment.rsc +2 -2
  128. package/.next/server/app/tasks.segments/_tree.segment.rsc +2 -2
  129. package/.next/server/app/tasks.segments/tasks/__PAGE__.segment.rsc +1 -1
  130. package/.next/server/app/tasks.segments/tasks.segment.rsc +1 -1
  131. package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js +3 -3
  132. package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js.map +1 -1
  133. package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js +1 -1
  134. package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js.map +1 -1
  135. package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js +1 -1
  136. package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js.map +1 -1
  137. package/.next/server/middleware-build-manifest.js +3 -3
  138. package/.next/server/pages/404.html +2 -2
  139. package/.next/server/pages/500.html +1 -1
  140. package/.next/static/chunks/017hq2-5l~_98.css +2 -0
  141. package/.next/static/chunks/09el212cmtr70.js +1 -0
  142. package/.next/static/chunks/0apaxz3c96keo.js +4 -0
  143. package/.next/static/chunks/188mrp1elgpea.js +1 -0
  144. package/.next/trace +2 -2
  145. package/.next/trace-build +1 -1
  146. package/app/components/TopoGraph.tsx +1745 -111
  147. package/package.json +1 -1
  148. package/scripts/topo-active-links-chip-hover-lift-test.mjs +93 -0
  149. package/scripts/topo-chip-digit-fontweight-test.mjs +105 -0
  150. package/scripts/topo-chip-digit-hover-bold-test.mjs +94 -0
  151. package/scripts/topo-chip-row-group-hover-brighten-test.mjs +107 -0
  152. package/scripts/topo-chip-row-hover-lift-test.mjs +95 -0
  153. package/scripts/topo-chrome-button-hover-lift-test.mjs +94 -0
  154. package/scripts/topo-chrome-segmented-radius-test.mjs +100 -0
  155. package/scripts/topo-click-ripple-opacity-test.mjs +99 -0
  156. package/scripts/topo-edge-badge-digit-fw-test.mjs +103 -0
  157. package/scripts/topo-edge-badge-fontsize-test.mjs +90 -0
  158. package/scripts/topo-edge-badge-hover-opacity-test.mjs +94 -0
  159. package/scripts/topo-edge-badge-hover-stroke-test.mjs +92 -0
  160. package/scripts/topo-edge-badge-opacity-test.mjs +80 -0
  161. package/scripts/topo-edge-badge-pin-opacity-test.mjs +86 -0
  162. package/scripts/topo-edge-badge-stroke-test.mjs +92 -0
  163. package/scripts/topo-edge-freshness-floor-test.mjs +99 -0
  164. package/scripts/topo-edge-particle-radius-test.mjs +76 -0
  165. package/scripts/topo-edge-visible-linecap-test.mjs +89 -0
  166. package/scripts/topo-filter-pill-hover-lift-test.mjs +101 -0
  167. package/scripts/topo-filter-pill-hover-opacity-test.mjs +110 -0
  168. package/scripts/topo-filter-pill-value-fw-test.mjs +88 -0
  169. package/scripts/topo-filter-pill-x-hover-scale-test.mjs +99 -0
  170. package/scripts/topo-flow-rail-linecap-test.mjs +79 -0
  171. package/scripts/topo-freshness-chip-hierarchy-test.mjs +93 -0
  172. package/scripts/topo-freshness-chip-tabular-test.mjs +41 -0
  173. package/scripts/topo-freshness-floor-lift-test.mjs +92 -0
  174. package/scripts/topo-freshness-suffix-tabular-test.mjs +88 -0
  175. package/scripts/topo-fullscreen-icon-hover-scale-test.mjs +91 -0
  176. package/scripts/topo-group-box-stroke-test.mjs +105 -0
  177. package/scripts/topo-group-label-count-fontweight-test.mjs +108 -0
  178. package/scripts/topo-hover-detail-body-fw-test.mjs +101 -0
  179. package/scripts/topo-hover-detail-model-fw-test.mjs +98 -0
  180. package/scripts/topo-hover-detail-opacity-test.mjs +98 -0
  181. package/scripts/topo-hover-detail-rx-test.mjs +81 -0
  182. package/scripts/topo-hub-digit-fontsize-test.mjs +86 -0
  183. package/scripts/topo-hub-digit-fw-hover-test.mjs +102 -0
  184. package/scripts/topo-hub-halo-light-trough-test.mjs +88 -0
  185. package/scripts/topo-hub-halo-radius-test.mjs +86 -0
  186. package/scripts/topo-hub-halo-trough-test.mjs +83 -0
  187. package/scripts/topo-hub-highlight-opacity-test.mjs +88 -0
  188. package/scripts/topo-hub-highlight-radius-test.mjs +90 -0
  189. package/scripts/topo-hub-hover-ring-opacity-test.mjs +96 -0
  190. package/scripts/topo-hub-hover-ring-stroke-test.mjs +86 -0
  191. package/scripts/topo-hub-spoke-linecap-test.mjs +80 -0
  192. package/scripts/topo-label-card-opacity-hover-test.mjs +99 -0
  193. package/scripts/topo-layout-toggle-hover-tracking-test.mjs +109 -0
  194. package/scripts/topo-layout-toggle-radius-test.mjs +87 -0
  195. package/scripts/topo-legend-label-fontweight-test.mjs +94 -0
  196. package/scripts/topo-legend-pin-ring-stroke-test.mjs +101 -0
  197. package/scripts/topo-minimap-offline-opacity-test.mjs +90 -0
  198. package/scripts/topo-minimap-online-hover-opacity-test.mjs +92 -0
  199. package/scripts/topo-minimap-online-opacity-test.mjs +93 -0
  200. package/scripts/topo-minimap-online-radius-test.mjs +85 -0
  201. package/scripts/topo-minimap-viewport-linejoin-test.mjs +75 -0
  202. package/scripts/topo-minimap-viewport-rx-test.mjs +85 -0
  203. package/scripts/topo-more-flows-fontweight-test.mjs +103 -0
  204. package/scripts/topo-node-alias-letter-spacing-test.mjs +112 -0
  205. package/scripts/topo-node-halo-offline-opacity-test.mjs +87 -0
  206. package/scripts/topo-node-label-card-rx-test.mjs +85 -0
  207. package/scripts/topo-node-pulse-peak-test.mjs +89 -0
  208. package/scripts/topo-node-pulse-trough-test.mjs +91 -0
  209. package/scripts/topo-node-sub-text-letter-spacing-test.mjs +115 -0
  210. package/scripts/topo-panel-count-fw-hover-test.mjs +105 -0
  211. package/scripts/topo-panel-count-letterspacing-test.mjs +89 -0
  212. package/scripts/topo-panel-stroke-hover-test.mjs +110 -0
  213. package/scripts/topo-pressure-bar-height-test.mjs +92 -0
  214. package/scripts/topo-pressure-kicker-fontweight-test.mjs +76 -0
  215. package/scripts/topo-recent-pip-radius-2-test.mjs +72 -0
  216. package/scripts/topo-recent-pip-radius-test.mjs +76 -0
  217. package/scripts/topo-recent-row-content-opacity-test.mjs +81 -0
  218. package/scripts/topo-recent-row-text-fontweight-test.mjs +90 -0
  219. package/scripts/topo-reset-hover-rotate-test.mjs +102 -0
  220. package/scripts/topo-spoke-active-opacity-test.mjs +104 -0
  221. package/scripts/topo-spoke-active-stroke-test.mjs +95 -0
  222. package/scripts/topo-spoke-idle-opacity-test.mjs +91 -0
  223. package/scripts/topo-vendor-chip-hover-lift-test.mjs +87 -0
  224. package/scripts/topo-vendor-glyph-fontweight-test.mjs +102 -0
  225. package/scripts/topo-vendor-letter-hover-scale-test.mjs +129 -0
  226. package/scripts/topo-vendor-suffix-hover-brighten-test.mjs +87 -0
  227. package/scripts/topo-zoom-icon-hover-scale-test.mjs +114 -0
  228. package/scripts/topo-zoom-level-hover-fw-test.mjs +95 -0
  229. package/.next/static/chunks/08fc_cz1nk7b9.js +0 -1
  230. package/.next/static/chunks/0aauz~36q5n2a.css +0 -2
  231. package/.next/static/chunks/0e0okm.affulg.js +0 -1
  232. package/.next/static/chunks/0s3vtwfo26_t6.js +0 -4
  233. /package/.next/static/{egukPz1ctU--4WnT2FpEU → mQHthzMGmjydHu598yl-Z}/_buildManifest.js +0 -0
  234. /package/.next/static/{egukPz1ctU--4WnT2FpEU → mQHthzMGmjydHu598yl-Z}/_clientMiddlewareManifest.js +0 -0
  235. /package/.next/static/{egukPz1ctU--4WnT2FpEU → mQHthzMGmjydHu598yl-Z}/_ssgManifest.js +0 -0
@@ -0,0 +1,115 @@
1
+ /* Round 428 verification: node sub-text (status label line beneath
2
+ * the alias) adopts hover letter-spacing tween 0 → 0.2px on
3
+ * hoveredAlias. Sibling to R427 alias-text 0 → 0.3 — the alias is
4
+ * primary identity (0.3 kerning), sub-text is secondary status (0.2).
5
+ *
6
+ * Contract:
7
+ * - rest: every sub-text has letter-spacing '0px' and hover='false'
8
+ * - hover one node: that sub-text has letter-spacing '0.2px' and
9
+ * hover='true'; siblings stay at rest
10
+ * - alias-text and sub-text tween together on the same hovered node
11
+ * (R427 0.3 + R428 0.2)
12
+ * - source-file probe confirms conditional + transition extended
13
+ */
14
+ import { chromium } from 'playwright';
15
+ import { readFileSync } from 'node:fs';
16
+
17
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
18
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
19
+
20
+ const browser = await chromium.launch({ headless: true });
21
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
22
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
23
+ await ctx.addInitScript(() => {
24
+ try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
25
+ });
26
+ await ctx.route('**/api/hub/status*', async (route) => {
27
+ const r = await route.fetch();
28
+ const b = await r.json();
29
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
30
+ const mk = (alias, status) => ({
31
+ alias, status, model: 'claude-opus-4', runtime: 'claude-code-cli',
32
+ network_id: nid, project_dir: null,
33
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
34
+ });
35
+ // small fleet — card-mode labels
36
+ await route.fulfill({ response: r, json: { ...b, sessions: [
37
+ mk('alpha', 'working'),
38
+ mk('beta', 'idle'),
39
+ mk('gamma', 'working'),
40
+ ] } });
41
+ });
42
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
43
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
44
+
45
+ const page = await ctx.newPage();
46
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
47
+ await page.waitForSelector('[data-node-sub-text]', { timeout: 15000 });
48
+ await page.waitForTimeout(400);
49
+
50
+ const readAll = () => page.evaluate(() => {
51
+ const subs = [...document.querySelectorAll('[data-node-sub-text]')];
52
+ const alis = [...document.querySelectorAll('[data-node-alias-text]')];
53
+ return {
54
+ sub: subs.map(t => ({
55
+ alias: t.getAttribute('data-node-sub-text'),
56
+ ls: t.style.letterSpacing,
57
+ hover: t.getAttribute('data-node-sub-text-hovered'),
58
+ })),
59
+ alias: alis.map(t => ({
60
+ alias: t.getAttribute('data-node-alias-text'),
61
+ ls: t.style.letterSpacing,
62
+ hover: t.getAttribute('data-node-alias-hovered'),
63
+ })),
64
+ };
65
+ });
66
+
67
+ const rest = await readAll();
68
+ const firstAlias = rest.sub[0]?.alias;
69
+ let hover = null;
70
+ if (firstAlias) {
71
+ const box = await page.evaluate((alias) => {
72
+ const t = document.querySelector(`[data-node-alias-text="${alias}"]`);
73
+ if (!t) return null;
74
+ const node = t.closest('[data-node]');
75
+ const target = node || t;
76
+ const b = target.getBoundingClientRect();
77
+ return { x: b.x + b.width / 2, y: b.y + b.height / 2 };
78
+ }, firstAlias);
79
+ if (box) {
80
+ await page.mouse.move(box.x, box.y);
81
+ await page.waitForTimeout(300);
82
+ hover = await readAll();
83
+ await page.mouse.move(0, 0);
84
+ }
85
+ }
86
+
87
+ // Source-file probe
88
+ const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
89
+ const sourceWired = /data-node-sub-text-hovered.*\n.*letterSpacing:\s*hoveredAlias\s*===\s*session\.alias\s*\?\s*'0\.2px'\s*:\s*'0px'/s.test(fileText);
90
+ const sourceTransition = /transition: 'fill 300ms ease-out, letter-spacing 200ms ease-out'/.test(fileText);
91
+
92
+ await browser.close();
93
+
94
+ const isRest = (ls) => ls === '0px' || ls === '' || ls === 'normal';
95
+ const hoveredSubEntry = hover?.sub.find(r => r.alias === firstAlias);
96
+ const hoveredAliasEntry = hover?.alias.find(r => r.alias === firstAlias);
97
+ const othersSubRest = hover ? hover.sub.filter(r => r.alias !== firstAlias).every(r => isRest(r.ls)) : false;
98
+
99
+ const results = {
100
+ rest_sub_count_gte_3: rest.sub.length >= 3,
101
+ rest_sub_all_zero: rest.sub.every(r => isRest(r.ls)),
102
+ rest_sub_all_hover_false: rest.sub.every(r => r.hover === 'false'),
103
+ hover_sub_target_ls_0_2: hoveredSubEntry?.ls === '0.2px',
104
+ hover_sub_target_attr_true: hoveredSubEntry?.hover === 'true',
105
+ hover_sub_others_stay_rest: othersSubRest,
106
+ // R427/R428 sibling parity — alias and sub-text tween together on same hovered node
107
+ hover_alias_target_ls_0_3: hoveredAliasEntry?.ls === '0.3px',
108
+ source_conditional_wired: sourceWired,
109
+ source_transition_wired: sourceTransition,
110
+ };
111
+ const ok = Object.values(results).every(Boolean);
112
+ console.log(`${ok ? '✅' : '❌'} node sub-text letter-spacing hover:`, JSON.stringify(results),
113
+ '\n hover sub:', JSON.stringify(hoveredSubEntry),
114
+ '\n hover alias:', JSON.stringify(hoveredAliasEntry));
115
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,105 @@
1
+ /* Round 424 verification: panel header count digit fontWeight 600 →
2
+ * 700 on panel hover. Closes the 5-layer panel hover cue stack with
3
+ * a typographic-weight axis at the panel-header data scope:
4
+ * R135 drop-shadow (depth)
5
+ * R348 fill opacity (solidity)
6
+ * R345 title letter-spacing (spacing)
7
+ * R423 stroke→legendAccent (color/edge)
8
+ * R424 count digit fw 600→700 (weight)
9
+ *
10
+ * Contract:
11
+ * - rest state: data-recent-panel-count + data-legend-panel-count
12
+ * tspans report computed font-weight matching rest branch (600)
13
+ * - geometry / R311 / R310 invariants preserved (digit text reads,
14
+ * R348 rest opacity '0.92' still applies to outer rect)
15
+ * - source-file probe verifies both panels are wired with
16
+ * `hoveredPanel === '<key>' ? '700' : '600'` conditional
17
+ * - panel hover dispatch on the panel <g> is unreliable
18
+ * (React's onMouseEnter non-bubbling synthetic — same R423
19
+ * finding); source-file verification is the canonical contract
20
+ * for the hover branch, same pattern R348/R394/R412/R423 used.
21
+ */
22
+ import { chromium } from 'playwright';
23
+ import { readFileSync } from 'node:fs';
24
+
25
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
26
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
27
+
28
+ const browser = await chromium.launch({ headless: true });
29
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
30
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
31
+ await ctx.addInitScript(() => {
32
+ try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
33
+ });
34
+ await ctx.route('**/api/hub/status*', async (route) => {
35
+ const r = await route.fetch();
36
+ const b = await r.json();
37
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
38
+ const mk = (alias, status) => ({
39
+ alias, status, model: 'claude-opus-4', runtime: 'claude-code-cli',
40
+ network_id: nid, project_dir: null,
41
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
42
+ });
43
+ await route.fulfill({ response: r, json: { ...b, sessions: [
44
+ mk('alpha', 'working'),
45
+ mk('beta', 'idle'),
46
+ ] } });
47
+ });
48
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({
49
+ json: { messages: [
50
+ { id: 'm1', from_alias: 'alpha', to_alias: 'beta', content: 'ping', created_at: fresh, network_id: 'default' },
51
+ ] },
52
+ }));
53
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
54
+
55
+ const page = await ctx.newPage();
56
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
57
+ await page.waitForSelector('[data-recent-panel-count], [data-legend-panel-count]', { timeout: 15000 });
58
+ await page.waitForTimeout(400);
59
+
60
+ const rest = await page.evaluate(() => {
61
+ const r = document.querySelector('[data-recent-panel-count]');
62
+ const l = document.querySelector('[data-legend-panel-count]');
63
+ return {
64
+ recent: r ? {
65
+ fw_attr: r.getAttribute('font-weight'),
66
+ fw_computed: getComputedStyle(r).fontWeight,
67
+ text: r.textContent,
68
+ } : null,
69
+ legend: l ? {
70
+ fw_attr: l.getAttribute('font-weight'),
71
+ fw_computed: getComputedStyle(l).fontWeight,
72
+ text: l.textContent,
73
+ } : null,
74
+ };
75
+ });
76
+
77
+ // Source-file verification — runtime hover dispatch on the panel <g>
78
+ // is unreliable (R423 finding). Same source-file probe pattern.
79
+ const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
80
+ const recentSourceWired = /fontWeight=\{hoveredPanel === 'recent' \? '700' : '600'\}/.test(fileText);
81
+ const legendSourceWired = /fontWeight=\{hoveredPanel === 'legend' \? '700' : '600'\}/.test(fileText);
82
+ // R247 transition list extended to include font-weight on both
83
+ const recentTransitionWired = /transition: 'fill 200ms ease-out, font-weight 200ms ease-out'/.test(fileText);
84
+
85
+ await browser.close();
86
+
87
+ const results = {
88
+ // mount / textContent invariants
89
+ recent_mounted: !!rest.recent,
90
+ legend_mounted: !!rest.legend,
91
+ recent_has_flows: /flows/.test(rest.recent?.text || ''),
92
+ legend_has_nodes: /node/.test(rest.legend?.text || ''),
93
+ // rest font-weight resolves to base 600 (string '600' from attr or computed)
94
+ recent_rest_fw_600: rest.recent?.fw_attr === '600' || rest.recent?.fw_computed === '600',
95
+ legend_rest_fw_600: rest.legend?.fw_attr === '600' || rest.legend?.fw_computed === '600',
96
+ // source-file: both panels wired with conditional fw branch
97
+ source_recent_hover_wired: recentSourceWired,
98
+ source_legend_hover_wired: legendSourceWired,
99
+ // transition list extended to font-weight
100
+ source_transition_wired: recentTransitionWired,
101
+ };
102
+ const ok = Object.values(results).every(Boolean);
103
+ console.log(`${ok ? '✅' : '❌'} panel count digit fw hover:`, JSON.stringify(results),
104
+ '\n rest:', rest);
105
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,89 @@
1
+ /* Round 349 verification: panel header counts pick up editorial
2
+ * letter-spacing 0.2, one tier below the R301 panel title 0.3.
3
+ * Closes the panel-pair editorial symmetry — both top-corner panels
4
+ * now have a 2-step header hierarchy:
5
+ * title fontSize=12 fontWeight=700 letterSpacing=0.3 (R301)
6
+ * count fontSize=10 fontWeight=600 letterSpacing=0.2 (R349, this round)
7
+ * Joins the R285 / R289 / R301 / R302 / R304 / R325 editorial-
8
+ * letterspacing tier at the panel-summary scope. The R162 freshness
9
+ * fill, R225 tabular-nums, R311 fontWeight, R336 unit-tspan opacity-0.7
10
+ * split all preserved (SVG inheritance propagates the tier to descendant
11
+ * tspans).
12
+ *
13
+ * Contract:
14
+ * - Recent-signal panel header count <text> letter-spacing="0.2".
15
+ * - Legend panel header count <text> letter-spacing="0.2".
16
+ * - data-{recent,legend}-panel-count-letter-spacing="0.2" attrs present.
17
+ * - Pre-R349 invariants: R311 fontWeight=600, R225 tabular-nums,
18
+ * R336 unit-tspan opacity=0.7 all preserved.
19
+ */
20
+ import { chromium } from 'playwright';
21
+ import { readFileSync } from 'node:fs';
22
+
23
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
24
+ const browser = await chromium.launch({ headless: true });
25
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
26
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
27
+ await ctx.addInitScript(() => {
28
+ try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
29
+ });
30
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
31
+ await ctx.route('**/api/hub/status*', async (route) => {
32
+ const r = await route.fetch();
33
+ const b = await r.json();
34
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
35
+ const mk = (alias) => ({
36
+ alias, status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli',
37
+ network_id: nid, project_dir: null,
38
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
39
+ });
40
+ await route.fulfill({ response: r, json: { ...b, sessions: [ mk('a'), mk('b'), mk('c') ] } });
41
+ });
42
+ // One message → recent panel mounts.
43
+ const now = Date.now();
44
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [
45
+ { id: 'm1', from_alias: 'a', to_alias: 'b', content: 'ping',
46
+ network_id: 'default', created_at: new Date(now - 5000).toISOString() },
47
+ ] } }));
48
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
49
+
50
+ const page = await ctx.newPage();
51
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
52
+ await page.waitForSelector('[data-recent-panel-count]', { timeout: 15000 });
53
+ await page.waitForSelector('[data-legend-panel-count]', { timeout: 5000 });
54
+ await page.waitForTimeout(300);
55
+
56
+ const probe = await page.evaluate(() => {
57
+ const rTspan = document.querySelector('[data-recent-panel-count]');
58
+ const rText = rTspan?.closest('text');
59
+ const rUnit = document.querySelector('[data-recent-panel-count-unit]');
60
+ const lText = document.querySelector('[data-legend-panel-count]');
61
+ const lUnit = document.querySelector('[data-legend-panel-count-unit]');
62
+ return {
63
+ recentTextLs: rText?.getAttribute('letter-spacing') ?? null,
64
+ recentLsAttr: rText?.getAttribute('data-recent-panel-count-letter-spacing') ?? null,
65
+ recentFw: rTspan?.getAttribute('font-weight') ?? null,
66
+ recentUnitOp: rUnit?.getAttribute('opacity') ?? null,
67
+ legendTextLs: lText?.getAttribute('letter-spacing') ?? null,
68
+ legendLsAttr: lText?.getAttribute('data-legend-panel-count-letter-spacing') ?? null,
69
+ legendFw: lText?.getAttribute('font-weight') ?? null,
70
+ legendUnitOp: lUnit?.getAttribute('opacity') ?? null,
71
+ };
72
+ });
73
+
74
+ await browser.close();
75
+
76
+ const results = {
77
+ recent_text_ls_0_2: probe.recentTextLs === '0.2',
78
+ recent_attr_0_2: probe.recentLsAttr === '0.2',
79
+ recent_fw_600: probe.recentFw === '600', // R311
80
+ recent_unit_op_0_7: probe.recentUnitOp === '0.7', // R336
81
+ legend_text_ls_0_2: probe.legendTextLs === '0.2',
82
+ legend_attr_0_2: probe.legendLsAttr === '0.2',
83
+ legend_fw_600: probe.legendFw === '600', // R310
84
+ legend_unit_op_0_7: probe.legendUnitOp === '0.7', // R336
85
+ };
86
+ const ok = Object.values(results).every(Boolean);
87
+ console.log(`${ok ? '✅' : '❌'} panel count letter-spacing 0.2:`, JSON.stringify(results),
88
+ '\n probe:', probe);
89
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,110 @@
1
+ /* Round 423 verification: panel rect stroke tints to legendAccent
2
+ * on hover. Sibling to R217 label-card stroke hover-tint. 4-layer
3
+ * hover cue stack (drop-shadow + opacity + title letter-spacing +
4
+ * stroke-tint) now symmetric across both side panels.
5
+ *
6
+ * Contract:
7
+ * - Probe rest state for both panels:
8
+ * * stroke attr === pal.legendBox.stroke (theme-driven)
9
+ * - Hover the recent-signal panel:
10
+ * * recent panel stroke shifts away from rest value
11
+ * * legend panel stroke stays at rest value
12
+ * - Hover the legend panel:
13
+ * * legend panel stroke shifts
14
+ * * recent panel stroke returns to rest
15
+ * - Pre-R423 invariants preserved:
16
+ * * data-topo-panel-elevation attr present
17
+ * * rect rx '10', width 230 / 224
18
+ * * R348 hover opacity '0.97' lift on hovered panel
19
+ */
20
+ import { chromium } from 'playwright';
21
+ import { readFileSync } from 'node:fs';
22
+
23
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
24
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
25
+
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
+ 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, status) => ({
37
+ alias, status, model: 'claude-opus-4', runtime: 'claude-code-cli',
38
+ network_id: nid, project_dir: null,
39
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
40
+ });
41
+ await route.fulfill({ response: r, json: { ...b, sessions: [
42
+ mk('alpha', 'working'),
43
+ mk('beta', 'idle'),
44
+ ] } });
45
+ });
46
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({
47
+ json: { messages: [
48
+ { id: 'm1', from_alias: 'alpha', to_alias: 'beta', content: 'ping', created_at: fresh, network_id: 'default' },
49
+ ] },
50
+ }));
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-topo-panel-elevation]', { timeout: 15000 });
56
+ await page.waitForTimeout(400);
57
+
58
+ const readState = () => page.evaluate(() => {
59
+ const recent = document.querySelector('[data-topo-panel-elevation="recent"]');
60
+ const legend = document.querySelector('[data-topo-panel-elevation="legend"]');
61
+ return {
62
+ recent: recent ? {
63
+ stroke: recent.getAttribute('stroke'),
64
+ opacity: recent.getAttribute('opacity'),
65
+ rx: recent.getAttribute('rx'),
66
+ width: recent.getAttribute('width'),
67
+ } : null,
68
+ legend: legend ? {
69
+ stroke: legend.getAttribute('stroke'),
70
+ opacity: legend.getAttribute('opacity'),
71
+ rx: legend.getAttribute('rx'),
72
+ width: legend.getAttribute('width'),
73
+ } : null,
74
+ };
75
+ });
76
+
77
+ const rest = await readState();
78
+
79
+ // Source-file verification — runtime hover-state on the panel <g>
80
+ // is hard to trigger via page.hover (inner row hitboxes intercept)
81
+ // and via DOM dispatch (React's onMouseEnter doesn't bubble + relies
82
+ // on its synthetic delegation system). The wire change is mechanical
83
+ // + the rest-state DOM probe confirms the default branch resolves
84
+ // correctly, so source-file verification suffices for the hover
85
+ // branch (same pattern R348 / R394 / R412 etc tests used).
86
+ const fileText = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
87
+ const recentSourceWired = /stroke=\{hoveredPanel === 'recent' \? pal\.legendAccent : pal\.legendBox\.stroke\}/.test(fileText);
88
+ const legendSourceWired = /stroke=\{hoveredPanel === 'legend' \? pal\.legendAccent : pal\.legendBox\.stroke\}/.test(fileText);
89
+
90
+ await browser.close();
91
+
92
+ const results = {
93
+ // Setup invariants
94
+ recent_rx_10: rest.recent?.rx === '10',
95
+ recent_width_230: rest.recent?.width === '230',
96
+ legend_rx_10: rest.legend?.rx === '10',
97
+ legend_width_224: rest.legend?.width === '224',
98
+ // Rest stroke matches across panels (default branch resolves)
99
+ rest_strokes_match: rest.recent?.stroke === rest.legend?.stroke,
100
+ // R348 rest opacity invariant
101
+ rest_recent_opacity_0_92: rest.recent?.opacity === '0.92',
102
+ rest_legend_opacity_0_92: rest.legend?.opacity === '0.92',
103
+ // R423 source-file: both panels wired with hover-tint conditional
104
+ source_recent_hover_wired: recentSourceWired,
105
+ source_legend_hover_wired: legendSourceWired,
106
+ };
107
+ const ok = Object.values(results).every(Boolean);
108
+ console.log(`${ok ? '✅' : '❌'} panel rect stroke hover-tint:`, JSON.stringify(results),
109
+ '\n rest:', rest);
110
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,92 @@
1
+ /* Round 374 verification: pressure-bar height h-1.5 → h-2 (6 → 8 px).
2
+ * Sibling visual-weight bump (8th canvas/HTML anchor in the family).
3
+ * +33 % bar height gives working/idle/offline segments more visibility.
4
+ *
5
+ * Geometry-safe: items-center flex centers the bar inside the chip's
6
+ * py-1 — bar at 8 px stays comfortably inside the 10-px text-row.
7
+ *
8
+ * Contract (cyber, fixture with mix):
9
+ * - Pressure bar wrapper computed height === '8px' (h-2).
10
+ * - data-fleet-pressure-bar-height === 'h-2'.
11
+ * - Pre-R374 invariants:
12
+ * * width still w-16 (64 px)
13
+ * * inline-flex display + rounded-full + overflow-hidden
14
+ * * background rgb(75 85 99 / 0.25)
15
+ * * Parent [data-fleet-pressure] chip className unchanged
16
+ * (font-mono + chip structure)
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 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
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
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, status = 'working') => ({
34
+ alias, status, 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
+ // Mix so all 3 segments render.
39
+ await route.fulfill({ response: r, json: { ...b, sessions: [
40
+ mk('w1'), mk('w2'),
41
+ mk('i1', 'idle'), mk('i2', 'idle'),
42
+ mk('o1', 'offline'),
43
+ ] } });
44
+ });
45
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
46
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
47
+
48
+ const page = await ctx.newPage();
49
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
50
+ await page.waitForSelector('[data-fleet-pressure-bar-height]', { timeout: 15000 });
51
+ await page.waitForTimeout(300);
52
+
53
+ const probe = await page.evaluate(() => {
54
+ const bar = document.querySelector('[data-fleet-pressure-bar-height]');
55
+ const cs = bar ? getComputedStyle(bar) : null;
56
+ const parent = document.querySelector('[data-fleet-pressure]');
57
+ // Count visible segments inside the bar.
58
+ const segs = bar ? Array.from(bar.querySelectorAll('[data-pressure-seg]')) : [];
59
+ return {
60
+ height: cs?.height ?? null,
61
+ width: cs?.width ?? null,
62
+ barHeightData: bar?.getAttribute('data-fleet-pressure-bar-height') ?? null,
63
+ display: cs?.display ?? null,
64
+ borderRadius: cs?.borderRadius ?? null,
65
+ overflow: cs?.overflow ?? null,
66
+ bgColor: cs?.backgroundColor ?? null,
67
+ parentMono: /font-mono/.test(parent?.getAttribute('class') || ''),
68
+ segCount: segs.length,
69
+ };
70
+ });
71
+
72
+ await browser.close();
73
+
74
+ const results = {
75
+ height_8px: probe.height === '8px', // h-2
76
+ width_64px: probe.width === '64px', // w-16
77
+ data_height_h_2: probe.barHeightData === 'h-2',
78
+ // Flex items get display blockified: inline-flex → flex when nested
79
+ // in flex container. Accept either.
80
+ display_inline_flex: probe.display === 'inline-flex' || probe.display === 'flex',
81
+ // Tailwind rounded-full is border-radius: 9999px, which browsers
82
+ // sometimes render as 3.35544e+07px (huge approximation). Accept any
83
+ // value indicating fully-rounded.
84
+ has_rounded_full: /9999|3\.355|e\+/.test(probe.borderRadius || ''),
85
+ overflow_hidden: probe.overflow === 'hidden',
86
+ parent_font_mono: probe.parentMono,
87
+ three_segments: probe.segCount === 3,
88
+ };
89
+ const ok = Object.values(results).every(Boolean);
90
+ console.log(`${ok ? '✅' : '❌'} pressure-bar height 1.5 → 2:`, JSON.stringify(results),
91
+ '\n probe:', probe);
92
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,76 @@
1
+ /* Round 373 verification: pressure-bar kicker label fontWeight 400 →
2
+ * 500. Sibling small-text fw lift family with R363/R364/R366/R368 —
3
+ * 5th surface in the family.
4
+ *
5
+ * Family snapshot post-R373:
6
+ * recent-row alias fw 500 (R363)
7
+ * legend-row label fw 500 (R364)
8
+ * group-label count fw 500 (R366)
9
+ * +N more flows footer fw 500 (R368)
10
+ * pressure-bar kicker fw 500 (R373, this round)
11
+ *
12
+ * Contract (cyber, fixture with mix so pressure bar renders):
13
+ * - [data-fleet-pressure-kicker] computed fontWeight === '500'.
14
+ * - Pre-R373 invariants:
15
+ * * text-[10px] (fontSize 10) preserved
16
+ * * tracking-wide (letter-spacing 0.025em) preserved
17
+ * * Parent [data-fleet-pressure] still has font-mono + chip
18
+ * structure
19
+ */
20
+ import { chromium } from 'playwright';
21
+ import { readFileSync } from 'node:fs';
22
+
23
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
24
+ const browser = await chromium.launch({ headless: true });
25
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
26
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
27
+ await ctx.addInitScript(() => {
28
+ try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
29
+ });
30
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
31
+ await ctx.route('**/api/hub/status*', async (route) => {
32
+ const r = await route.fetch();
33
+ const b = await r.json();
34
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
35
+ const mk = (alias) => ({
36
+ alias, status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli',
37
+ network_id: nid, project_dir: null,
38
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
39
+ });
40
+ await route.fulfill({ response: r, json: { ...b, sessions: [ mk('a'), mk('b'), mk('c') ] } });
41
+ });
42
+ 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-fleet-pressure-kicker]', { timeout: 15000 });
48
+ await page.waitForTimeout(300);
49
+
50
+ const probe = await page.evaluate(() => {
51
+ const kicker = document.querySelector('[data-fleet-pressure-kicker]');
52
+ const parent = document.querySelector('[data-fleet-pressure]');
53
+ const cs = kicker ? getComputedStyle(kicker) : null;
54
+ return {
55
+ fontWeight: cs?.fontWeight ?? null,
56
+ fontSize: cs?.fontSize ?? null,
57
+ letterSpacing: cs?.letterSpacing ?? null,
58
+ text: kicker?.textContent ?? null,
59
+ parentCls: parent?.getAttribute('class') ?? null,
60
+ };
61
+ });
62
+
63
+ await browser.close();
64
+
65
+ const results = {
66
+ font_weight_500: probe.fontWeight === '500',
67
+ font_size_10px: probe.fontSize === '10px',
68
+ // tracking-wide = 0.025em → at 10px font that's 0.25px computed.
69
+ letter_spacing_tracking_wide: /0\.25px|0\.025em/.test(probe.letterSpacing || ''),
70
+ text_is_pressure: probe.text === 'pressure',
71
+ parent_font_mono: /font-mono/.test(probe.parentCls || ''),
72
+ };
73
+ const ok = Object.values(results).every(Boolean);
74
+ console.log(`${ok ? '✅' : '❌'} pressure-bar kicker fw 400 → 500:`, JSON.stringify(results),
75
+ '\n probe:', probe);
76
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,72 @@
1
+ /* Round 383 verification: recency pip base radius 1.8 → 2.0.
2
+ * Continues R359 lift trajectory; sibling visual-weight bump
3
+ * family (9th anchor).
4
+ *
5
+ * Contract:
6
+ * - Pip element computed r === '2.0' (rendered) or attribute equals 2.
7
+ * - data-recent-row-freshness-radius === '2.0'.
8
+ * - Pre-R383 invariants:
9
+ * * cx=10 (R160 left margin position unchanged)
10
+ * * fill=pal.legendAccent
11
+ * * opacity transition (R358) preserved
12
+ * * R358 stale-alpha floor 0.30 still applies
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 browser = await chromium.launch({ headless: true });
19
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1500 } });
20
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
21
+ await ctx.addInitScript(() => {
22
+ try { localStorage.setItem('anet-theme', 'cyber'); sessionStorage.setItem('anet_v3_auth', '1'); } catch {}
23
+ });
24
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
25
+ await ctx.route('**/api/hub/status*', async (route) => {
26
+ const r = await route.fetch();
27
+ const b = await r.json();
28
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
29
+ const mk = (alias) => ({
30
+ alias, status: 'working', model: 'claude-opus-4', runtime: 'claude-code-cli',
31
+ network_id: nid, project_dir: null,
32
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
33
+ });
34
+ await route.fulfill({ response: r, json: { ...b, sessions: [ mk('a'), mk('b') ] } });
35
+ });
36
+ const now = Date.now();
37
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [
38
+ { id: 'm1', from_alias: 'a', to_alias: 'b', content: 'ping',
39
+ network_id: 'default', created_at: new Date(now - 5000).toISOString(),
40
+ last_at: new Date(now - 5000).toISOString() },
41
+ ] } }));
42
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
43
+
44
+ const page = await ctx.newPage();
45
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'domcontentloaded' });
46
+ await page.waitForSelector('[data-recent-row-freshness]', { timeout: 15000 });
47
+ await page.waitForTimeout(300);
48
+
49
+ const probe = await page.evaluate(() => {
50
+ const pip = document.querySelector('[data-recent-row-freshness]');
51
+ return {
52
+ rAttr: pip?.getAttribute('r') ?? null,
53
+ radiusAttr: pip?.getAttribute('data-recent-row-freshness-radius') ?? null,
54
+ cxAttr: pip?.getAttribute('cx') ?? null,
55
+ linkKey: pip?.getAttribute('data-recent-row-freshness') ?? null,
56
+ opacityAttr: pip?.getAttribute('opacity') ?? null,
57
+ };
58
+ });
59
+
60
+ await browser.close();
61
+
62
+ const results = {
63
+ r_attr_2: probe.rAttr === '2' || probe.rAttr === '2.0',
64
+ radius_attr_2_0: probe.radiusAttr === '2.0',
65
+ cx_unchanged_10: probe.cxAttr === '10', // R160 invariant
66
+ link_key_present: (probe.linkKey || '').length > 0,
67
+ opacity_above_zero: parseFloat(probe.opacityAttr || '0') > 0, // fresh, alpha > 0
68
+ };
69
+ const ok = Object.values(results).every(Boolean);
70
+ console.log(`${ok ? '✅' : '❌'} recent pip radius 1.8 → 2.0:`, JSON.stringify(results),
71
+ '\n probe:', probe);
72
+ process.exit(ok ? 0 : 1);