@sleep2agi/agent-network-dashboard 0.5.3-preview.13 → 0.5.3-preview.130

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 (280) 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/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 +12 -12
  15. package/.next/server/app/_not-found.segments/_full.segment.rsc +12 -12
  16. package/.next/server/app/_not-found.segments/_head.segment.rsc +4 -4
  17. package/.next/server/app/_not-found.segments/_index.segment.rsc +7 -7
  18. package/.next/server/app/_not-found.segments/_not-found/__PAGE__.segment.rsc +2 -2
  19. package/.next/server/app/_not-found.segments/_not-found.segment.rsc +3 -3
  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 +14 -14
  24. package/.next/server/app/admin.segments/_full.segment.rsc +14 -14
  25. package/.next/server/app/admin.segments/_head.segment.rsc +4 -4
  26. package/.next/server/app/admin.segments/_index.segment.rsc +7 -7
  27. package/.next/server/app/admin.segments/_tree.segment.rsc +2 -2
  28. package/.next/server/app/admin.segments/admin/__PAGE__.segment.rsc +4 -4
  29. package/.next/server/app/admin.segments/admin.segment.rsc +3 -3
  30. package/.next/server/app/index.html +2 -2
  31. package/.next/server/app/index.rsc +14 -14
  32. package/.next/server/app/index.segments/__PAGE__.segment.rsc +4 -4
  33. package/.next/server/app/index.segments/_full.segment.rsc +14 -14
  34. package/.next/server/app/index.segments/_head.segment.rsc +4 -4
  35. package/.next/server/app/index.segments/_index.segment.rsc +7 -7
  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 +14 -14
  40. package/.next/server/app/login.segments/_full.segment.rsc +14 -14
  41. package/.next/server/app/login.segments/_head.segment.rsc +4 -4
  42. package/.next/server/app/login.segments/_index.segment.rsc +7 -7
  43. package/.next/server/app/login.segments/_tree.segment.rsc +2 -2
  44. package/.next/server/app/login.segments/login/__PAGE__.segment.rsc +4 -4
  45. package/.next/server/app/login.segments/login.segment.rsc +3 -3
  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 +14 -14
  49. package/.next/server/app/logs.segments/_full.segment.rsc +14 -14
  50. package/.next/server/app/logs.segments/_head.segment.rsc +4 -4
  51. package/.next/server/app/logs.segments/_index.segment.rsc +7 -7
  52. package/.next/server/app/logs.segments/_tree.segment.rsc +2 -2
  53. package/.next/server/app/logs.segments/logs/__PAGE__.segment.rsc +4 -4
  54. package/.next/server/app/logs.segments/logs.segment.rsc +3 -3
  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 +14 -14
  58. package/.next/server/app/messages.segments/_full.segment.rsc +14 -14
  59. package/.next/server/app/messages.segments/_head.segment.rsc +4 -4
  60. package/.next/server/app/messages.segments/_index.segment.rsc +7 -7
  61. package/.next/server/app/messages.segments/_tree.segment.rsc +2 -2
  62. package/.next/server/app/messages.segments/messages/__PAGE__.segment.rsc +4 -4
  63. package/.next/server/app/messages.segments/messages.segment.rsc +3 -3
  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 +14 -14
  67. package/.next/server/app/node.segments/_full.segment.rsc +14 -14
  68. package/.next/server/app/node.segments/_head.segment.rsc +4 -4
  69. package/.next/server/app/node.segments/_index.segment.rsc +7 -7
  70. package/.next/server/app/node.segments/_tree.segment.rsc +2 -2
  71. package/.next/server/app/node.segments/node/__PAGE__.segment.rsc +4 -4
  72. package/.next/server/app/node.segments/node.segment.rsc +3 -3
  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 +14 -14
  76. package/.next/server/app/nodes.segments/_full.segment.rsc +14 -14
  77. package/.next/server/app/nodes.segments/_head.segment.rsc +4 -4
  78. package/.next/server/app/nodes.segments/_index.segment.rsc +7 -7
  79. package/.next/server/app/nodes.segments/_tree.segment.rsc +2 -2
  80. package/.next/server/app/nodes.segments/nodes/__PAGE__.segment.rsc +4 -4
  81. package/.next/server/app/nodes.segments/nodes.segment.rsc +3 -3
  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 +14 -14
  86. package/.next/server/app/server-logs.segments/_full.segment.rsc +14 -14
  87. package/.next/server/app/server-logs.segments/_head.segment.rsc +4 -4
  88. package/.next/server/app/server-logs.segments/_index.segment.rsc +7 -7
  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 +4 -4
  91. package/.next/server/app/server-logs.segments/server-logs.segment.rsc +3 -3
  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 +14 -14
  95. package/.next/server/app/settings/networks.segments/_full.segment.rsc +14 -14
  96. package/.next/server/app/settings/networks.segments/_head.segment.rsc +4 -4
  97. package/.next/server/app/settings/networks.segments/_index.segment.rsc +7 -7
  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 +4 -4
  100. package/.next/server/app/settings/networks.segments/settings/networks.segment.rsc +3 -3
  101. package/.next/server/app/settings/networks.segments/settings.segment.rsc +3 -3
  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 +14 -14
  106. package/.next/server/app/settings/tokens.segments/_full.segment.rsc +14 -14
  107. package/.next/server/app/settings/tokens.segments/_head.segment.rsc +4 -4
  108. package/.next/server/app/settings/tokens.segments/_index.segment.rsc +7 -7
  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 +4 -4
  111. package/.next/server/app/settings/tokens.segments/settings/tokens.segment.rsc +3 -3
  112. package/.next/server/app/settings/tokens.segments/settings.segment.rsc +3 -3
  113. package/.next/server/app/settings.html +2 -2
  114. package/.next/server/app/settings.rsc +14 -14
  115. package/.next/server/app/settings.segments/_full.segment.rsc +14 -14
  116. package/.next/server/app/settings.segments/_head.segment.rsc +4 -4
  117. package/.next/server/app/settings.segments/_index.segment.rsc +7 -7
  118. package/.next/server/app/settings.segments/_tree.segment.rsc +2 -2
  119. package/.next/server/app/settings.segments/settings/__PAGE__.segment.rsc +4 -4
  120. package/.next/server/app/settings.segments/settings.segment.rsc +3 -3
  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 +14 -14
  125. package/.next/server/app/tasks.segments/_full.segment.rsc +14 -14
  126. package/.next/server/app/tasks.segments/_head.segment.rsc +4 -4
  127. package/.next/server/app/tasks.segments/_index.segment.rsc +7 -7
  128. package/.next/server/app/tasks.segments/_tree.segment.rsc +2 -2
  129. package/.next/server/app/tasks.segments/tasks/__PAGE__.segment.rsc +4 -4
  130. package/.next/server/app/tasks.segments/tasks.segment.rsc +3 -3
  131. package/.next/server/chunks/ssr/[root-of-the-server]__0sv~g.o._.js +1 -1
  132. package/.next/server/chunks/ssr/[root-of-the-server]__0sv~g.o._.js.map +1 -1
  133. package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js +4 -4
  134. package/.next/server/chunks/ssr/agent-network-dashboard_09kk21a._.js.map +1 -1
  135. package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js +1 -1
  136. package/.next/server/chunks/ssr/agent-network-dashboard_app_01jhlxz._.js.map +1 -1
  137. package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js +1 -1
  138. package/.next/server/chunks/ssr/agent-network-dashboard_app_09d29my._.js.map +1 -1
  139. package/.next/server/chunks/ssr/agent-network-dashboard_app_components_0mvyi-4._.js +1 -1
  140. package/.next/server/chunks/ssr/agent-network-dashboard_app_components_0mvyi-4._.js.map +1 -1
  141. package/.next/server/middleware-build-manifest.js +3 -3
  142. package/.next/server/pages/404.html +2 -2
  143. package/.next/server/pages/500.html +1 -1
  144. package/.next/static/chunks/05-~ezd16iskp.css +2 -0
  145. package/.next/static/chunks/06m0zlsymhs-w.js +4 -0
  146. package/.next/static/chunks/0b67lke3ou0wm.js +1 -0
  147. package/.next/static/chunks/0ow5lxkzf479q.js +1 -0
  148. package/.next/static/chunks/{03a4--7ncekmk.js → 0v4-5tng.uh.7.js} +2 -2
  149. package/.next/static/chunks/{11iqwh145jvo5.js → 137w3utqbpw8c.js} +1 -1
  150. package/.next/trace +2 -2
  151. package/.next/trace-build +1 -1
  152. package/app/components/ServersDrawer.tsx +16 -3
  153. package/app/components/TopoGraph.tsx +3493 -148
  154. package/app/globals.css +58 -7
  155. package/package.json +4 -4
  156. package/scripts/p157-servers-copy-test.mjs +95 -0
  157. package/scripts/topo-active-chrome-hover-text-test.mjs +107 -0
  158. package/scripts/topo-alias-chat-brightness-test.mjs +79 -0
  159. package/scripts/topo-avatar-brightness-test.mjs +116 -0
  160. package/scripts/topo-avatar-drop-shadow-test.mjs +86 -0
  161. package/scripts/topo-avatar-fallback-hover-test.mjs +104 -0
  162. package/scripts/topo-avatar-fallback-rotate-test.mjs +92 -0
  163. package/scripts/topo-avatar-rotate-test.mjs +85 -0
  164. package/scripts/topo-avatar-scale-test.mjs +89 -0
  165. package/scripts/topo-brand-drop-shadow-test.mjs +71 -0
  166. package/scripts/topo-brand-logo-breath-test.mjs +102 -0
  167. package/scripts/topo-brand-logo-hover-brightness-test.mjs +105 -0
  168. package/scripts/topo-brand-logo-hover-rotate-test.mjs +93 -0
  169. package/scripts/topo-brand-logo-hover-test.mjs +85 -0
  170. package/scripts/topo-chat-ring-brightness-test.mjs +80 -0
  171. package/scripts/topo-chip-row-digit-ls-test.mjs +135 -0
  172. package/scripts/topo-chip-row-member-alias-lit-test.mjs +154 -0
  173. package/scripts/topo-chip-row-tier-glow-brightness-test.mjs +99 -0
  174. package/scripts/topo-chip-row-unit-hover-tracking-test.mjs +124 -0
  175. package/scripts/topo-click-ripple-glow-test.mjs +86 -0
  176. package/scripts/topo-cluster-count-attr-test.mjs +80 -0
  177. package/scripts/topo-crescent-breath-test.mjs +104 -0
  178. package/scripts/topo-crescent-recede-test.mjs +111 -0
  179. package/scripts/topo-edge-badge-circle-brightness-test.mjs +82 -0
  180. package/scripts/topo-edge-badge-hover-glow-test.mjs +90 -0
  181. package/scripts/topo-edge-badge-text-brightness-test.mjs +83 -0
  182. package/scripts/topo-edge-particle-brightness-test.mjs +82 -0
  183. package/scripts/topo-edge-pill-glow-test.mjs +67 -0
  184. package/scripts/topo-edge-visible-brightness-test.mjs +84 -0
  185. package/scripts/topo-endpoint-ring-brightness-test.mjs +83 -0
  186. package/scripts/topo-filter-pill-glow-test.mjs +90 -0
  187. package/scripts/topo-fleet-density-tier-test.mjs +84 -0
  188. package/scripts/topo-flow-arrow-brightness-test.mjs +82 -0
  189. package/scripts/topo-flow-rail-brightness-test.mjs +80 -0
  190. package/scripts/topo-freshness-chip-fade-test.mjs +105 -0
  191. package/scripts/topo-fullscreen-attr-test.mjs +73 -0
  192. package/scripts/topo-fullscreen-brightness-test.mjs +84 -0
  193. package/scripts/topo-fullscreen-icon-rotate-test.mjs +93 -0
  194. package/scripts/topo-grid-content-bottom-attr-test.mjs +72 -0
  195. package/scripts/topo-group-box-brightness-test.mjs +84 -0
  196. package/scripts/topo-group-label-brightness-test.mjs +84 -0
  197. package/scripts/topo-group-label-hover-glow-test.mjs +86 -0
  198. package/scripts/topo-group-label-member-alias-hover-test.mjs +125 -0
  199. package/scripts/topo-group-pill-glow-test.mjs +76 -0
  200. package/scripts/topo-group-tint-brightness-test.mjs +82 -0
  201. package/scripts/topo-hub-core-brightness-test.mjs +82 -0
  202. package/scripts/topo-hub-digit-brightness-test.mjs +79 -0
  203. package/scripts/topo-hub-digit-ls-test.mjs +119 -0
  204. package/scripts/topo-hub-halo-brightness-test.mjs +80 -0
  205. package/scripts/topo-hub-halo-glow-test.mjs +96 -0
  206. package/scripts/topo-hub-highlight-amplify-test.mjs +139 -0
  207. package/scripts/topo-hub-highlight-brightness-test.mjs +84 -0
  208. package/scripts/topo-hub-highlight-fill-transition-test.mjs +84 -0
  209. package/scripts/topo-hub-highlight-glow-test.mjs +99 -0
  210. package/scripts/topo-hub-highlight-r-test.mjs +112 -0
  211. package/scripts/topo-hub-highlight-recede-test.mjs +144 -0
  212. package/scripts/topo-hub-highlight-theme-fill-test.mjs +83 -0
  213. package/scripts/topo-hub-hover-ring-brightness-test.mjs +79 -0
  214. package/scripts/topo-hub-hover-ring-glow-test.mjs +97 -0
  215. package/scripts/topo-hub-idle-breath-test.mjs +7 -2
  216. package/scripts/topo-hub-recede-test.mjs +124 -0
  217. package/scripts/topo-hub-spoke-brightness-test.mjs +77 -0
  218. package/scripts/topo-hub-spoke-glow-test.mjs +112 -0
  219. package/scripts/topo-label-card-brightness-test.mjs +81 -0
  220. package/scripts/topo-layout-hover-fw-test.mjs +98 -0
  221. package/scripts/topo-layout-toggle-brightness-test.mjs +94 -0
  222. package/scripts/topo-legend-count-brightness-test.mjs +80 -0
  223. package/scripts/topo-legend-count-letter-spacing-test.mjs +108 -0
  224. package/scripts/topo-legend-label-fw-test.mjs +107 -0
  225. package/scripts/topo-legend-pin-ring-brightness-test.mjs +82 -0
  226. package/scripts/topo-legend-row-count-brightness-test.mjs +85 -0
  227. package/scripts/topo-legend-row-label-glow-test.mjs +102 -0
  228. package/scripts/topo-legend-swatch-glow-test.mjs +109 -0
  229. package/scripts/topo-legend-swatch-member-alias-match-test.mjs +139 -0
  230. package/scripts/topo-legend-tint-brightness-test.mjs +83 -0
  231. package/scripts/topo-minimap-hover-glow-test.mjs +109 -0
  232. package/scripts/topo-more-footer-brightness-test.mjs +94 -0
  233. package/scripts/topo-node-alias-brightness-test.mjs +84 -0
  234. package/scripts/topo-node-sub-text-brightness-test.mjs +88 -0
  235. package/scripts/topo-nodesize-brightness-test.mjs +82 -0
  236. package/scripts/topo-nodesize-hover-fw-test.mjs +99 -0
  237. package/scripts/topo-orphan-box-dash-test.mjs +89 -0
  238. package/scripts/topo-orphan-fill-opacity-test.mjs +91 -0
  239. package/scripts/topo-orphan-label-opacity-test.mjs +98 -0
  240. package/scripts/topo-panel-count-hover-ls-test.mjs +87 -0
  241. package/scripts/topo-panel-row-brightness-test.mjs +116 -0
  242. package/scripts/topo-panel-title-brightness-test.mjs +98 -0
  243. package/scripts/topo-panel-title-glow-test.mjs +111 -0
  244. package/scripts/topo-pill-x-rotate-test.mjs +96 -0
  245. package/scripts/topo-pinned-aspect-test.mjs +89 -0
  246. package/scripts/topo-pip-brightness-test.mjs +85 -0
  247. package/scripts/topo-pressure-seg-glow-test.mjs +92 -0
  248. package/scripts/topo-pressure-seg-member-alias-match-test.mjs +133 -0
  249. package/scripts/topo-pressure-seg-motion-test.mjs +101 -0
  250. package/scripts/topo-recent-count-brightness-test.mjs +84 -0
  251. package/scripts/topo-recent-more-fw-test.mjs +126 -0
  252. package/scripts/topo-recent-row-fw-test.mjs +115 -0
  253. package/scripts/topo-recent-row-text-glow-test.mjs +86 -0
  254. package/scripts/topo-recent-tint-brightness-test.mjs +80 -0
  255. package/scripts/topo-recent-ts-brightness-test.mjs +86 -0
  256. package/scripts/topo-reduced-motion-attr-test.mjs +69 -0
  257. package/scripts/topo-reset-brightness-test.mjs +83 -0
  258. package/scripts/topo-reset-icon-hover-scale-test.mjs +102 -0
  259. package/scripts/topo-runtime-badge-brightness-test.mjs +78 -0
  260. package/scripts/topo-runtime-badge-glow-test.mjs +108 -0
  261. package/scripts/topo-runtime-badge-rotate-test.mjs +85 -0
  262. package/scripts/topo-starfield-hue-test.mjs +109 -0
  263. package/scripts/topo-status-ring-brightness-test.mjs +84 -0
  264. package/scripts/topo-titleblock-h2-hover-fw-test.mjs +109 -0
  265. package/scripts/topo-titleblock-h2-hover-tracking-test.mjs +128 -0
  266. package/scripts/topo-titleblock-kicker-hover-test.mjs +134 -0
  267. package/scripts/topo-vendor-chip-glow-test.mjs +97 -0
  268. package/scripts/topo-vendor-pill-glow-test.mjs +98 -0
  269. package/scripts/topo-watermark-breath-test.mjs +100 -0
  270. package/scripts/topo-watermark-recede-test.mjs +114 -0
  271. package/scripts/topo-zoom-buttons-brightness-test.mjs +94 -0
  272. package/scripts/topo-zoom-level-brightness-test.mjs +83 -0
  273. package/scripts/topo-zoom-level-color-test.mjs +105 -0
  274. package/.next/static/chunks/0-eqn.ga3bcnl.js +0 -1
  275. package/.next/static/chunks/0m.1mvl~t.avc.css +0 -2
  276. package/.next/static/chunks/0tl3h11hxa7oe.js +0 -4
  277. package/.next/static/chunks/0zwxl-vr5q45i.js +0 -1
  278. /package/.next/static/{U6pY8Ja-T2ME4lRhun26w → 2h4soq4G-uDf06jPV2C36}/_buildManifest.js +0 -0
  279. /package/.next/static/{U6pY8Ja-T2ME4lRhun26w → 2h4soq4G-uDf06jPV2C36}/_clientMiddlewareManifest.js +0 -0
  280. /package/.next/static/{U6pY8Ja-T2ME4lRhun26w → 2h4soq4G-uDf06jPV2C36}/_ssgManifest.js +0 -0
@@ -0,0 +1,90 @@
1
+ /* Round 534 verification: edge midpoint badge gains drop-shadow glow on
2
+ * hover/pin (cyan accent), with hover precedence over isHot (amber R480).
3
+ *
4
+ * Test strategy: source-side wiring is canonical because edge hover via
5
+ * Playwright is impractical (topo-panel rect intercepts at SVG root;
6
+ * R48 hitbox is the React handler target but isn't directly hoverable
7
+ * through the panel). Rest-state DOM probe + source regex covers the
8
+ * 4-axis hover-lift parity wiring.
9
+ *
10
+ * Test phases:
11
+ * 1. rest cold: glow attr='false', filter='none', lifted attr='false'
12
+ * (regression check — pre-R534 behavior unchanged at rest)
13
+ * 2. source-side regex confirms:
14
+ * - filter ternary precedence: (hovered||pinned) > isHot > undefined
15
+ * - cyan accent hue (pal.legendAccent) at 99 hex alpha
16
+ * - data-edge-badge-glow 3-value attr ('hover' | 'hot' | 'false')
17
+ * - transition list includes filter
18
+ */
19
+ import { chromium } from 'playwright';
20
+ import { readFileSync } from 'node:fs';
21
+
22
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
23
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
24
+
25
+ const browser = await chromium.launch({ headless: true });
26
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1200 } });
27
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
28
+ await ctx.addInitScript(() => {
29
+ try {
30
+ localStorage.setItem('anet-theme', 'cyber');
31
+ localStorage.setItem('anet-topo-layout', 'ring');
32
+ sessionStorage.setItem('anet_v3_auth', '1');
33
+ } catch {}
34
+ });
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) => ({
40
+ alias, status: 'working', 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: [mk('a·1'), mk('a·2')] } });
45
+ });
46
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: {
47
+ messages: [{ id: 'm1', from_alias: 'a·1', to_alias: 'a·2', content: 'test', created_at: fresh }]
48
+ } }));
49
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
50
+ const page = await ctx.newPage();
51
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
52
+ await page.waitForSelector('[data-edge-badge-glow]', { timeout: 15000 });
53
+ await page.waitForTimeout(800);
54
+
55
+ const rest = await page.evaluate(() => {
56
+ const el = document.querySelector('[data-edge-badge-glow]');
57
+ if (!el) return null;
58
+ const cs = getComputedStyle(el);
59
+ return {
60
+ glowAttr: el.getAttribute('data-edge-badge-glow'),
61
+ liftedAttr: el.getAttribute('data-edge-badge-lifted'),
62
+ filter: cs.filter,
63
+ transition: cs.transition,
64
+ };
65
+ });
66
+
67
+ await browser.close();
68
+
69
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
70
+ const sourceFilterTernary =
71
+ /filter: \(isHoveredEdge \|\| isPinned\)\s+\? `drop-shadow\(0 0 3px \$\{pal\.legendAccent\}99\)`\s+: isHot\s+\? `drop-shadow\(0 0 3px \$\{hotStroke\}80\)`\s+: undefined,/.test(src);
72
+ const sourceAttrTernary =
73
+ /data-edge-badge-glow=\{\(isHoveredEdge \|\| isPinned\) \? 'hover' : isHot \? 'hot' : 'false'\}/.test(src);
74
+ const sourceTransitionFilter =
75
+ /transition: 'r 180ms ease-out, stroke 300ms ease-out, stroke-width 300ms ease-out, fill 200ms ease-out, opacity 200ms ease-out, filter 200ms ease-out'/.test(src);
76
+
77
+ const results = {
78
+ rest_glow_false: rest?.glowAttr === 'false',
79
+ rest_lifted_false: rest?.liftedAttr === 'false',
80
+ rest_filter_none: rest?.filter === 'none' || rest?.filter === '',
81
+ rest_transition_has_filter: /\bfilter\b/.test(rest?.transition || ''),
82
+ source_filter_ternary: sourceFilterTernary,
83
+ source_attr_ternary: sourceAttrTernary,
84
+ source_transition_filter: sourceTransitionFilter,
85
+ };
86
+ const ok = Object.values(results).every(Boolean);
87
+ console.log(`${ok ? '✅' : '❌'} R534 edge-badge hover-glow (source-canonical):`,
88
+ JSON.stringify(results, null, 2),
89
+ '\n rest:', JSON.stringify(rest));
90
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,83 @@
1
+ /* Round 570 verification: edge-badge digit gains filter brightness
2
+ * (1.15) on (isHoveredEdge || isPinned || isHot). Joins per-element
3
+ * brightness family at 7th anchor.
4
+ *
5
+ * Test phases:
6
+ * 1. mock 1 message → 1 flowLink with edge-badge rendered
7
+ * 2. rest: filter='none', brightness-attr='1'
8
+ * 3. transition-property contains 'filter'
9
+ * 4. source-side regex confirms wiring
10
+ *
11
+ * Live hover/pin on SVG edge-badge is unreliable for Playwright
12
+ * (R525 banked); use source-canonical for gate + rest data-attrs.
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: 1200 } });
22
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
23
+ await ctx.addInitScript(() => {
24
+ try {
25
+ localStorage.setItem('anet-theme', 'cyber');
26
+ localStorage.setItem('anet-topo-layout', 'ring');
27
+ sessionStorage.setItem('anet_v3_auth', '1');
28
+ } catch {}
29
+ });
30
+ await ctx.route('**/api/hub/status*', async (route) => {
31
+ const r = await route.fetch();
32
+ const b = await r.json();
33
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
34
+ const mk = (alias) => ({
35
+ alias, status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli',
36
+ network_id: nid, project_dir: null,
37
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
38
+ });
39
+ await route.fulfill({ response: r, json: { ...b, sessions: [mk('a·1'), mk('a·2')] } });
40
+ });
41
+ // 1 message so flowLinks > 0 → edge-badge renders
42
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [
43
+ { from_alias: 'a·1', to_alias: 'a·2', content: 'hi', created_at: fresh },
44
+ ] } }));
45
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
46
+ const page = await ctx.newPage();
47
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
48
+ await page.waitForSelector('[data-edge-badge-text]', { timeout: 15000 });
49
+ await page.waitForTimeout(500);
50
+
51
+ const rest = await page.evaluate(() => {
52
+ const el = document.querySelector('[data-edge-badge-text]');
53
+ if (!el) return null;
54
+ const cs = getComputedStyle(el);
55
+ return {
56
+ filter: cs.filter,
57
+ transitionProperty: cs.transitionProperty,
58
+ brightnessAttr: el.getAttribute('data-edge-badge-text-brightness'),
59
+ pinAttr: el.getAttribute('data-edge-badge-text-pin'),
60
+ };
61
+ });
62
+
63
+ await browser.close();
64
+
65
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
66
+ const sourceFilter = /filter: \(isHoveredEdge \|\| isPinned \|\| isHot\)\s*\?\s*'brightness\(1\.15\)'\s*:\s*undefined/.test(src);
67
+ const sourceAttr = /data-edge-badge-text-brightness=\{\(isHoveredEdge \|\| isPinned \|\| isHot\) \? '1\.15' : '1'\}/.test(src);
68
+ const sourceTransition = /transition: 'letter-spacing 300ms ease-out, font-weight 300ms ease-out, filter 300ms ease-out'/.test(src);
69
+
70
+ const results = {
71
+ rest_filter_none: rest?.filter === 'none',
72
+ rest_brightness_1: rest?.brightnessAttr === '1',
73
+ rest_pin_false: rest?.pinAttr === 'false',
74
+ transition_has_filter: /filter/.test(rest?.transitionProperty || ''),
75
+ source_filter: sourceFilter,
76
+ source_attr: sourceAttr,
77
+ source_transition: sourceTransition,
78
+ };
79
+ const ok = Object.values(results).every(Boolean);
80
+ console.log(`${ok ? '✅' : '❌'} R570 edge-badge text brightness(1.15) (per-element family 7th anchor):`,
81
+ JSON.stringify(results, null, 2),
82
+ `\n rest: ${JSON.stringify(rest)}`);
83
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,82 @@
1
+ /* Round 583 verification: edge flow particle gains filter
2
+ * brightness(1.15) on edge hover/endpoint hover. 22nd anchor in
3
+ * per-element brightness family.
4
+ *
5
+ * Test phases:
6
+ * 1. mock 1 message → 1 flowLink → particle renders (SMIL-animated)
7
+ * 2. rest: filter='none', brightness-attr='1'
8
+ * 3. transition-property contains 'filter'
9
+ * 4. source: filter conditional + data-attr + transition extension
10
+ */
11
+ import { chromium } from 'playwright';
12
+ import { readFileSync } from 'node:fs';
13
+
14
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
15
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
16
+
17
+ const browser = await chromium.launch({ headless: true });
18
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1200 } });
19
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
20
+ await ctx.addInitScript(() => {
21
+ try {
22
+ localStorage.setItem('anet-theme', 'cyber');
23
+ localStorage.setItem('anet-topo-layout', 'ring');
24
+ sessionStorage.setItem('anet_v3_auth', '1');
25
+ } catch {}
26
+ });
27
+ await ctx.route('**/api/hub/status*', async (route) => {
28
+ const r = await route.fetch();
29
+ const b = await r.json();
30
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
31
+ const mk = (alias) => ({
32
+ alias, status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli',
33
+ network_id: nid, project_dir: null,
34
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
35
+ });
36
+ await route.fulfill({ response: r, json: { ...b, sessions: [mk('a·1'), mk('a·2')] } });
37
+ });
38
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [
39
+ { from_alias: 'a·1', to_alias: 'a·2', content: 'hi', created_at: fresh },
40
+ ] } }));
41
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
42
+ const page = await ctx.newPage();
43
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
44
+ await page.waitForSelector('[data-edge-particle]', { timeout: 15000, state: 'attached' });
45
+ await page.waitForTimeout(500);
46
+
47
+ const rest = await page.evaluate(() => {
48
+ const el = document.querySelector('[data-edge-particle]');
49
+ if (!el) return null;
50
+ const cs = getComputedStyle(el);
51
+ return {
52
+ filter: cs.filter,
53
+ transitionProperty: cs.transitionProperty,
54
+ brightnessAttr: el.getAttribute('data-edge-particle-brightness'),
55
+ liftedAttr: el.getAttribute('data-edge-particle-lifted'),
56
+ };
57
+ });
58
+
59
+ await browser.close();
60
+
61
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
62
+ const sourceFilter = /filter: \(isHoveredEdge \|\| isEndpointHoveredEdge\)\s*\?\s*\(isLight \? 'brightness\(1\.15\)' : 'url\(#topo-glow\) brightness\(1\.15\)'\)\s*:\s*undefined/.test(src);
63
+ const sourceAttr = /data-edge-particle-brightness=\{\(isHoveredEdge \|\| isEndpointHoveredEdge\) \? '1\.15' : '1'\}/.test(src);
64
+ const sourceTransition = /transition: 'fill 200ms ease-out, opacity 200ms ease-out, r 200ms ease-out, filter 200ms ease-out'/.test(src);
65
+
66
+ const results = {
67
+ particle_present: !!rest,
68
+ // On cyber theme at rest, particle inherits url(#topo-glow) via attribute
69
+ rest_filter_has_url_or_none: /url\(/.test(rest?.filter || '') || rest?.filter === 'none',
70
+ rest_filter_no_brightness: !/brightness\(/.test(rest?.filter || ''),
71
+ rest_brightness_1: rest?.brightnessAttr === '1',
72
+ rest_lifted_false: rest?.liftedAttr === 'false',
73
+ transition_has_filter: /filter/.test(rest?.transitionProperty || ''),
74
+ source_filter: sourceFilter,
75
+ source_attr: sourceAttr,
76
+ source_transition: sourceTransition,
77
+ };
78
+ const ok = Object.values(results).every(Boolean);
79
+ console.log(`${ok ? '✅' : '❌'} R583 edge flow particle brightness (22nd anchor):`,
80
+ JSON.stringify(results, null, 2),
81
+ `\n rest: ${JSON.stringify(rest)}`);
82
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,67 @@
1
+ /* Round 546 verification: edge filter pill gains pal.flowEdge-tied drop-
2
+ * shadow when rendered. CLOSES pin-active filter-pill drop-shadow sub-
3
+ * family at the 4th pill variant. Source-canonical (banked R544 lesson —
4
+ * setPinnedEdgeKey is bound to SVG edge hitbox/badge, Playwright can't
5
+ * reach through topo-panel rect's pointer interception).
6
+ *
7
+ * Test phases:
8
+ * 1. unpinned: no [data-active-filter="edge"] element
9
+ * 2. source-side regex confirms filter wired with pal.flowEdge via
10
+ * color-mix 60% syntax, scoped to edge-pill block (setPinnedEdgeKey
11
+ * handler proximity)
12
+ */
13
+ import { chromium } from 'playwright';
14
+ import { readFileSync } from 'node:fs';
15
+
16
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
17
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
18
+
19
+ const browser = await chromium.launch({ headless: true });
20
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1200 } });
21
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
22
+ await ctx.addInitScript(() => {
23
+ try {
24
+ localStorage.setItem('anet-theme', 'cyber');
25
+ localStorage.setItem('anet-topo-layout', 'ring');
26
+ sessionStorage.setItem('anet_v3_auth', '1');
27
+ } catch {}
28
+ });
29
+ await ctx.route('**/api/hub/status*', async (route) => {
30
+ const r = await route.fetch();
31
+ const b = await r.json();
32
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
33
+ const mk = (alias) => ({
34
+ alias, status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli',
35
+ network_id: nid, project_dir: null,
36
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
37
+ });
38
+ await route.fulfill({ response: r, json: { ...b, sessions: [mk('a·1'), mk('a·2')] } });
39
+ });
40
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
41
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
42
+ const page = await ctx.newPage();
43
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
44
+ await page.waitForSelector('svg[data-topo-layout]', { timeout: 15000 });
45
+ await page.waitForTimeout(500);
46
+
47
+ const unpinned = await page.evaluate(() =>
48
+ document.querySelector('[data-active-filter="edge"]') !== null
49
+ );
50
+
51
+ await browser.close();
52
+
53
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
54
+ // Scope: setPinnedEdgeKey(null) handler vicinity (edge pill block)
55
+ const edgePillScope = src.match(/onClick=\{\(\) => setPinnedEdgeKey\(null\)\}[\s\S]{0,2500}/)?.[0] || '';
56
+ const sourceFilterScoped =
57
+ /filter: `drop-shadow\(0 0 3px color-mix\(in srgb, \$\{pal\.flowEdge\} 60%, transparent\)\)`,/.test(edgePillScope);
58
+
59
+ const results = {
60
+ unpinned_pill_absent: unpinned === false,
61
+ source_filter_scoped: sourceFilterScoped,
62
+ };
63
+ const ok = Object.values(results).every(Boolean);
64
+ console.log(`${ok ? '✅' : '❌'} R546 edge filter pill glow (source-canonical, CLOSES sub-family):`,
65
+ JSON.stringify(results, null, 2),
66
+ '\n unpinned absent:', unpinned);
67
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,84 @@
1
+ /* Round 582 verification: edge visible flow path gains filter
2
+ * brightness(1.15) stacked with url(#topo-glow) on cyber theme,
3
+ * plain brightness on light. 21st anchor in per-element
4
+ * brightness family.
5
+ *
6
+ * Test phases:
7
+ * 1. mock 1 message → 1 flowLink → visible path renders
8
+ * 2. rest (cyber): filter contains 'url("#topo-glow")'
9
+ * (no brightness stacked, no hover state)
10
+ * 3. brightness-attr='1' at rest
11
+ * 4. source: filter conditional + data-attr + transition extension
12
+ */
13
+ import { chromium } from 'playwright';
14
+ import { readFileSync } from 'node:fs';
15
+
16
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
17
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
18
+
19
+ const browser = await chromium.launch({ headless: true });
20
+ const ctx = await browser.newContext({ viewport: { width: 1500, height: 1200 } });
21
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
22
+ await ctx.addInitScript(() => {
23
+ try {
24
+ localStorage.setItem('anet-theme', 'cyber');
25
+ localStorage.setItem('anet-topo-layout', 'ring');
26
+ sessionStorage.setItem('anet_v3_auth', '1');
27
+ } catch {}
28
+ });
29
+ await ctx.route('**/api/hub/status*', async (route) => {
30
+ const r = await route.fetch();
31
+ const b = await r.json();
32
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
33
+ const mk = (alias) => ({
34
+ alias, status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli',
35
+ network_id: nid, project_dir: null,
36
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
37
+ });
38
+ await route.fulfill({ response: r, json: { ...b, sessions: [mk('a·1'), mk('a·2')] } });
39
+ });
40
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [
41
+ { from_alias: 'a·1', to_alias: 'a·2', content: 'hi', created_at: fresh },
42
+ ] } }));
43
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
44
+ const page = await ctx.newPage();
45
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
46
+ await page.waitForSelector('[data-edge-visible]', { timeout: 15000, state: 'attached' });
47
+ await page.waitForTimeout(500);
48
+
49
+ const rest = await page.evaluate(() => {
50
+ const el = document.querySelector('[data-edge-visible]');
51
+ if (!el) return null;
52
+ const cs = getComputedStyle(el);
53
+ return {
54
+ filter: cs.filter,
55
+ transitionProperty: cs.transitionProperty,
56
+ brightnessAttr: el.getAttribute('data-edge-visible-brightness'),
57
+ endpointHoveredAttr: el.getAttribute('data-edge-visible-endpoint-hovered'),
58
+ };
59
+ });
60
+
61
+ await browser.close();
62
+
63
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
64
+ const sourceFilter = /filter: \(isHoveredEdge \|\| isEndpointHoveredEdge\)\s*\?\s*\(isLight \? 'brightness\(1\.15\)' : 'url\(#topo-glow\) brightness\(1\.15\)'\)\s*:\s*\(isLight \? undefined : 'url\(#topo-glow\)'\)/.test(src);
65
+ const sourceAttr = /data-edge-visible-brightness=\{\(isHoveredEdge \|\| isEndpointHoveredEdge\) \? '1\.15' : '1'\}/.test(src);
66
+ const sourceTransition = /transition: 'opacity 300ms ease-out, stroke-width 300ms ease-out, stroke 300ms ease-out, filter 300ms ease-out'/.test(src);
67
+
68
+ const results = {
69
+ path_present: !!rest,
70
+ // On cyber theme (default in mock) at rest: filter = url(#topo-glow)
71
+ rest_filter_has_url: /url\(/.test(rest?.filter || ''),
72
+ rest_filter_no_brightness: !/brightness\(/.test(rest?.filter || ''),
73
+ rest_brightness_1: rest?.brightnessAttr === '1',
74
+ rest_endpoint_false: rest?.endpointHoveredAttr === 'false',
75
+ transition_has_filter: /filter/.test(rest?.transitionProperty || ''),
76
+ source_filter: sourceFilter,
77
+ source_attr: sourceAttr,
78
+ source_transition: sourceTransition,
79
+ };
80
+ const ok = Object.values(results).every(Boolean);
81
+ console.log(`${ok ? '✅' : '❌'} R582 edge visible path stacked brightness (21st anchor):`,
82
+ JSON.stringify(results, null, 2),
83
+ `\n rest: ${JSON.stringify(rest)}`);
84
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,83 @@
1
+ /* Round 585 verification: edge endpoint emphasis ring gains
2
+ * filter brightness(1.15) when isEndpoint activates. 24th
3
+ * anchor in per-element brightness family, fourth edge-tier
4
+ * paint layer (rail R581 + curve R582 + particle R583 +
5
+ * endpoint-ring R585).
6
+ *
7
+ * Test phases:
8
+ * 1. mock 2 idle nodes, 0 messages → endpoint rings render
9
+ * (opacity=0 at rest, isEndpoint=false)
10
+ * 2. rest: filter='none', brightness-attr='1', active='false'
11
+ * 3. transition-property contains 'filter'
12
+ * 4. source: filter conditional + data-attr + transition extension
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: 1200 } });
22
+ await ctx.addCookies([{ name: 'anet_dashboard_session', value: `v3:${TOKEN}`, domain: '127.0.0.1', path: '/' }]);
23
+ await ctx.addInitScript(() => {
24
+ try {
25
+ localStorage.setItem('anet-theme', 'cyber');
26
+ localStorage.setItem('anet-topo-layout', 'ring');
27
+ sessionStorage.setItem('anet_v3_auth', '1');
28
+ } catch {}
29
+ });
30
+ await ctx.route('**/api/hub/status*', async (route) => {
31
+ const r = await route.fetch();
32
+ const b = await r.json();
33
+ const nid = (b.sessions || [])[0]?.network_id || 'default';
34
+ const mk = (alias) => ({
35
+ alias, status: 'idle', model: 'claude-opus-4', runtime: 'claude-code-cli',
36
+ network_id: nid, project_dir: null,
37
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
38
+ });
39
+ await route.fulfill({ response: r, json: { ...b, sessions: [mk('a·1'), mk('a·2')] } });
40
+ });
41
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
42
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
43
+ const page = await ctx.newPage();
44
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
45
+ await page.waitForSelector('[data-edge-endpoint-ring]', { timeout: 15000, state: 'attached' });
46
+ await page.waitForTimeout(500);
47
+
48
+ const rest = await page.evaluate(() => {
49
+ const el = document.querySelector('[data-edge-endpoint-ring]');
50
+ if (!el) return null;
51
+ const cs = getComputedStyle(el);
52
+ return {
53
+ filter: cs.filter,
54
+ transitionProperty: cs.transitionProperty,
55
+ brightnessAttr: el.getAttribute('data-edge-endpoint-ring-brightness'),
56
+ activeAttr: el.getAttribute('data-edge-endpoint-active'),
57
+ opacity: cs.opacity,
58
+ };
59
+ });
60
+
61
+ await browser.close();
62
+
63
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
64
+ const sourceFilter = /filter: isEndpoint \? 'brightness\(1\.15\)' : undefined/.test(src);
65
+ const sourceAttr = /data-edge-endpoint-ring-brightness=\{isEndpoint \? '1\.15' : '1'\}/.test(src);
66
+ const sourceTransition = /transition: 'opacity 180ms ease-out, stroke-width 180ms ease-out, r 180ms ease-out, filter 180ms ease-out'/.test(src);
67
+
68
+ const results = {
69
+ ring_present: !!rest,
70
+ rest_filter_none: rest?.filter === 'none',
71
+ rest_brightness_1: rest?.brightnessAttr === '1',
72
+ rest_active_false: rest?.activeAttr === 'false',
73
+ rest_opacity_zero: parseFloat(rest?.opacity || '1') === 0,
74
+ transition_has_filter: /filter/.test(rest?.transitionProperty || ''),
75
+ source_filter: sourceFilter,
76
+ source_attr: sourceAttr,
77
+ source_transition: sourceTransition,
78
+ };
79
+ const ok = Object.values(results).every(Boolean);
80
+ console.log(`${ok ? '✅' : '❌'} R585 endpoint-ring brightness (24th anchor, edge-tier 4/4):`,
81
+ JSON.stringify(results, null, 2),
82
+ `\n rest: ${JSON.stringify(rest)}`);
83
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,90 @@
1
+ /* Round 543 verification: status filter pill gains always-on tier-color
2
+ * drop-shadow when rendered (pin-gated visual). Sibling to R477 legend
3
+ * pin-ring pin-gated drop-shadow at the chip-row scope.
4
+ *
5
+ * Test phases:
6
+ * 1. unpinned: no [data-active-filter="status"] element (pill hidden)
7
+ * 2. click working pressure-bar segment to pin → pill renders
8
+ * 3. pill has filter='drop-shadow(...)' with cyber working tier
9
+ * color rgba(134, 239, 172, ...) [#86efac]
10
+ * 4. source-side regex confirms filter wired with tier-color ternary
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', 'ring');
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
+ mk('a·1', 'working'), mk('a·2', 'idle'), mk('a·3', 'offline'),
39
+ ] } });
40
+ });
41
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
42
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
43
+ const page = await ctx.newPage();
44
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
45
+ await page.waitForSelector('[data-pressure-seg="working"]', { timeout: 15000 });
46
+ await page.waitForTimeout(500);
47
+
48
+ // Phase 1: unpinned — verify pill absent
49
+ const unpinned = await page.evaluate(() => {
50
+ return document.querySelector('[data-active-filter="status"]') !== null;
51
+ });
52
+
53
+ // Phase 2: click working pressure-seg to pin status='working'
54
+ await page.click('[data-pressure-seg="working"]');
55
+ await page.waitForTimeout(400);
56
+
57
+ // Phase 3: pill should now exist
58
+ await page.waitForSelector('[data-active-filter="status"]', { timeout: 5000 });
59
+ const pinned = await page.evaluate(() => {
60
+ const el = document.querySelector('[data-active-filter="status"]');
61
+ if (!el) return null;
62
+ const cs = getComputedStyle(el);
63
+ return {
64
+ inlineFilter: el.style.filter,
65
+ filter: cs.filter,
66
+ };
67
+ });
68
+
69
+ await browser.close();
70
+
71
+ // Source regex
72
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
73
+ const sourceFilterWired =
74
+ /filter: `drop-shadow\(0 0 3px \$\{[\s\S]*?pinnedStatus === 'working'[\s\S]*?'#047857'[\s\S]*?'#86efac'[\s\S]*?\}99\)`,/.test(src);
75
+
76
+ const results = {
77
+ unpinned_pill_absent: unpinned === false,
78
+ pinned_pill_present: pinned !== null,
79
+ pinned_inline_filter: /drop-shadow/.test(pinned?.inlineFilter || ''),
80
+ pinned_computed_filter: /drop-shadow/.test(pinned?.filter || ''),
81
+ // Cyber working text color #86efac → rgb(134, 239, 172); +0x99 alpha → rgba(134, 239, 172, 0.6)
82
+ pinned_working_color: /rgba?\(134,?\s*239,?\s*172/.test(pinned?.filter || ''),
83
+ source_filter_wired: sourceFilterWired,
84
+ };
85
+ const ok = Object.values(results).every(Boolean);
86
+ console.log(`${ok ? '✅' : '❌'} R543 status filter pill glow:`,
87
+ JSON.stringify(results, null, 2),
88
+ '\n unpinned absent:', unpinned,
89
+ '\n pinned:', JSON.stringify(pinned));
90
+ process.exit(ok ? 0 : 1);
@@ -0,0 +1,84 @@
1
+ /* Round 502 verification: root svg surfaces `data-topo-fleet-density-tier`
2
+ * categorical attribute (12th in canvas state surface set). Paired with
3
+ * R469 numeric counts; classifies onlineNodes.length into 5 buckets:
4
+ * empty (0) / sparse (1-3) / normal (4-15) / dense (16-30) / very-dense (31+)
5
+ *
6
+ * Boundaries align with R109 denseLayout = >16 gate so the tier name
7
+ * is semantically aligned with the canvas's existing visual-mode switch.
8
+ *
9
+ * Verification across 5 fixture scenarios + source-side regex.
10
+ */
11
+ import { chromium } from 'playwright';
12
+ import { readFileSync } from 'node:fs';
13
+
14
+ const TOKEN = JSON.parse(readFileSync('/home/vansin/.anet/config.json', 'utf8')).token;
15
+ const fresh = new Date(Date.now() - 60 * 1000).toISOString();
16
+
17
+ async function probe({ nodeCount, label }) {
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', 'ring');
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 sessions = Array.from({ length: nodeCount }, (_, i) => ({
33
+ alias: `a·${i}`, status: 'idle',
34
+ model: 'claude-opus-4', runtime: 'claude-code-cli',
35
+ network_id: nid, project_dir: null,
36
+ created_at: fresh, updated_at: fresh, last_seen_at: fresh,
37
+ }));
38
+ await route.fulfill({ response: r, json: { ...b, sessions } });
39
+ });
40
+ await ctx.route('**/api/hub/messages*', (r) => r.fulfill({ json: { messages: [] } }));
41
+ await ctx.route('**/api/hub/tasks*', (r) => r.fulfill({ json: { tasks: [] } }));
42
+ const page = await ctx.newPage();
43
+ await page.goto('http://127.0.0.1:3000/', { waitUntil: 'networkidle' });
44
+ await page.waitForTimeout(nodeCount === 0 ? 1500 : 2500); // empty fixture has no SVG canvas; sparse+ needs render time
45
+ const tier = await page.evaluate(() => {
46
+ const svg = document.querySelector('svg[viewBox="0 0 1000 680"]');
47
+ return svg?.getAttribute('data-topo-fleet-density-tier');
48
+ });
49
+ const onlineCount = await page.evaluate(() => {
50
+ const svg = document.querySelector('svg[viewBox="0 0 1000 680"]');
51
+ return svg?.getAttribute('data-topo-online-count');
52
+ });
53
+ await browser.close();
54
+ return { label, nodeCount, tier, onlineCount };
55
+ }
56
+
57
+ // 5 fixtures across all tier boundaries. Empty case can't probe (no SVG canvas).
58
+ // Use 1, 4, 16, 31 to hit each non-empty tier boundary directly.
59
+ const sparse = await probe({ nodeCount: 1, label: 'sparse boundary @ 1' });
60
+ const sparseTop = await probe({ nodeCount: 3, label: 'sparse boundary @ 3' });
61
+ const normal = await probe({ nodeCount: 4, label: 'normal boundary @ 4' });
62
+ const normalTop = await probe({ nodeCount: 15, label: 'normal boundary @ 15' });
63
+ const dense = await probe({ nodeCount: 16, label: 'dense boundary @ 16' });
64
+ // Skip dense=30 + very-dense=31 to keep test fast — boundary coverage already there
65
+
66
+ const src = readFileSync('/home/vansin/agent-network-dashboard/app/components/TopoGraph.tsx', 'utf8');
67
+ const sourceWired = /data-topo-fleet-density-tier=\{[\s\S]*?onlineNodes\.length === 0 \? 'empty' :[\s\S]*?onlineNodes\.length <= 3 \? 'sparse' :[\s\S]*?onlineNodes\.length <= 15 \? 'normal' :[\s\S]*?onlineNodes\.length <= 30 \? 'dense' :[\s\S]*?'very-dense'/.test(src);
68
+
69
+ const results = {
70
+ sparse_1: sparse.tier === 'sparse',
71
+ sparse_top_3: sparseTop.tier === 'sparse',
72
+ normal_4: normal.tier === 'normal',
73
+ normal_top_15: normalTop.tier === 'normal',
74
+ dense_16: dense.tier === 'dense',
75
+ source_wired: sourceWired,
76
+ };
77
+ const ok = Object.values(results).every(Boolean);
78
+ console.log(`${ok ? '✅' : '❌'} R502 fleet-density-tier:`, JSON.stringify(results),
79
+ '\n ', sparse,
80
+ '\n ', sparseTop,
81
+ '\n ', normal,
82
+ '\n ', normalTop,
83
+ '\n ', dense);
84
+ process.exit(ok ? 0 : 1);