@stigmer/react 0.3.4 → 0.4.1

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 (450) hide show
  1. package/billing/AutoRechargeCard.d.ts +38 -0
  2. package/billing/AutoRechargeCard.d.ts.map +1 -0
  3. package/billing/AutoRechargeCard.js +90 -0
  4. package/billing/AutoRechargeCard.js.map +1 -0
  5. package/billing/BillingSection.d.ts +32 -0
  6. package/billing/BillingSection.d.ts.map +1 -0
  7. package/billing/BillingSection.js +81 -0
  8. package/billing/BillingSection.js.map +1 -0
  9. package/billing/CreditBalanceCard.d.ts +25 -0
  10. package/billing/CreditBalanceCard.d.ts.map +1 -0
  11. package/billing/CreditBalanceCard.js +28 -0
  12. package/billing/CreditBalanceCard.js.map +1 -0
  13. package/billing/CreditLedgerTable.d.ts +22 -0
  14. package/billing/CreditLedgerTable.d.ts.map +1 -0
  15. package/billing/CreditLedgerTable.js +75 -0
  16. package/billing/CreditLedgerTable.js.map +1 -0
  17. package/billing/CreditPackGrid.d.ts +31 -0
  18. package/billing/CreditPackGrid.d.ts.map +1 -0
  19. package/billing/CreditPackGrid.js +35 -0
  20. package/billing/CreditPackGrid.js.map +1 -0
  21. package/billing/LowBalanceBanner.d.ts +26 -0
  22. package/billing/LowBalanceBanner.d.ts.map +1 -0
  23. package/billing/LowBalanceBanner.js +33 -0
  24. package/billing/LowBalanceBanner.js.map +1 -0
  25. package/billing/PaymentMethodCard.d.ts +35 -0
  26. package/billing/PaymentMethodCard.d.ts.map +1 -0
  27. package/billing/PaymentMethodCard.js +48 -0
  28. package/billing/PaymentMethodCard.js.map +1 -0
  29. package/billing/credit-packs.d.ts +25 -0
  30. package/billing/credit-packs.d.ts.map +1 -0
  31. package/billing/credit-packs.js +39 -0
  32. package/billing/credit-packs.js.map +1 -0
  33. package/billing/format.d.ts +39 -0
  34. package/billing/format.d.ts.map +1 -0
  35. package/billing/format.js +90 -0
  36. package/billing/format.js.map +1 -0
  37. package/billing/index.d.ts +32 -0
  38. package/billing/index.d.ts.map +1 -0
  39. package/billing/index.js +21 -0
  40. package/billing/index.js.map +1 -0
  41. package/billing/useBillingAccount.d.ts +40 -0
  42. package/billing/useBillingAccount.d.ts.map +1 -0
  43. package/billing/useBillingAccount.js +35 -0
  44. package/billing/useBillingAccount.js.map +1 -0
  45. package/billing/useBillingUsageReport.d.ts +42 -0
  46. package/billing/useBillingUsageReport.d.ts.map +1 -0
  47. package/billing/useBillingUsageReport.js +43 -0
  48. package/billing/useBillingUsageReport.js.map +1 -0
  49. package/billing/useCreateBillingPortalSession.d.ts +35 -0
  50. package/billing/useCreateBillingPortalSession.d.ts.map +1 -0
  51. package/billing/useCreateBillingPortalSession.js +50 -0
  52. package/billing/useCreateBillingPortalSession.js.map +1 -0
  53. package/billing/useCreateCheckoutSession.d.ts +54 -0
  54. package/billing/useCreateCheckoutSession.d.ts.map +1 -0
  55. package/billing/useCreateCheckoutSession.js +58 -0
  56. package/billing/useCreateCheckoutSession.js.map +1 -0
  57. package/billing/useCreditLedger.d.ts +48 -0
  58. package/billing/useCreditLedger.d.ts.map +1 -0
  59. package/billing/useCreditLedger.js +39 -0
  60. package/billing/useCreditLedger.js.map +1 -0
  61. package/billing/useCustomerModelPricing.d.ts +41 -0
  62. package/billing/useCustomerModelPricing.d.ts.map +1 -0
  63. package/billing/useCustomerModelPricing.js +37 -0
  64. package/billing/useCustomerModelPricing.js.map +1 -0
  65. package/billing/useSetAutoRechargeConfig.d.ts +50 -0
  66. package/billing/useSetAutoRechargeConfig.d.ts.map +1 -0
  67. package/billing/useSetAutoRechargeConfig.js +53 -0
  68. package/billing/useSetAutoRechargeConfig.js.map +1 -0
  69. package/composer/ComposerToolbar.js +1 -1
  70. package/composer/ComposerToolbar.js.map +1 -1
  71. package/composer/SessionComposer.d.ts +1 -1
  72. package/composer/SessionComposer.d.ts.map +1 -1
  73. package/composer/SessionComposer.js +19 -4
  74. package/composer/SessionComposer.js.map +1 -1
  75. package/composer/__tests__/SessionComposer-memo.test.d.ts +2 -0
  76. package/composer/__tests__/SessionComposer-memo.test.d.ts.map +1 -0
  77. package/composer/__tests__/SessionComposer-memo.test.js +23 -0
  78. package/composer/__tests__/SessionComposer-memo.test.js.map +1 -0
  79. package/execution/ApprovalCard.d.ts +5 -1
  80. package/execution/ApprovalCard.d.ts.map +1 -1
  81. package/execution/ApprovalCard.js +7 -3
  82. package/execution/ApprovalCard.js.map +1 -1
  83. package/execution/ExecutionPhaseBadge.d.ts +1 -1
  84. package/execution/ExecutionPhaseBadge.d.ts.map +1 -1
  85. package/execution/ExecutionPhaseBadge.js +3 -2
  86. package/execution/ExecutionPhaseBadge.js.map +1 -1
  87. package/execution/MessageEntry.d.ts +7 -3
  88. package/execution/MessageEntry.d.ts.map +1 -1
  89. package/execution/MessageEntry.js +19 -8
  90. package/execution/MessageEntry.js.map +1 -1
  91. package/execution/MessageThread.d.ts +84 -3
  92. package/execution/MessageThread.d.ts.map +1 -1
  93. package/execution/MessageThread.js +113 -65
  94. package/execution/MessageThread.js.map +1 -1
  95. package/execution/SetupProgress.d.ts +1 -1
  96. package/execution/SetupProgress.d.ts.map +1 -1
  97. package/execution/SetupProgress.js +3 -3
  98. package/execution/SetupProgress.js.map +1 -1
  99. package/execution/SubAgentSection.d.ts +5 -1
  100. package/execution/SubAgentSection.d.ts.map +1 -1
  101. package/execution/SubAgentSection.js +13 -7
  102. package/execution/SubAgentSection.js.map +1 -1
  103. package/execution/ThreadSkeleton.d.ts +22 -0
  104. package/execution/ThreadSkeleton.d.ts.map +1 -0
  105. package/execution/ThreadSkeleton.js +26 -0
  106. package/execution/ThreadSkeleton.js.map +1 -0
  107. package/execution/ToolCallGroup.d.ts +16 -1
  108. package/execution/ToolCallGroup.d.ts.map +1 -1
  109. package/execution/ToolCallGroup.js +31 -3
  110. package/execution/ToolCallGroup.js.map +1 -1
  111. package/execution/UsageWidget.d.ts +1 -1
  112. package/execution/__tests__/message-entry.test.d.ts +2 -0
  113. package/execution/__tests__/message-entry.test.d.ts.map +1 -0
  114. package/execution/__tests__/message-entry.test.js +178 -0
  115. package/execution/__tests__/message-entry.test.js.map +1 -0
  116. package/execution/__tests__/thread-keys.test.d.ts +2 -0
  117. package/execution/__tests__/thread-keys.test.d.ts.map +1 -0
  118. package/execution/__tests__/thread-keys.test.js +289 -0
  119. package/execution/__tests__/thread-keys.test.js.map +1 -0
  120. package/execution/__tests__/thread-memoization.test.d.ts +2 -0
  121. package/execution/__tests__/thread-memoization.test.d.ts.map +1 -0
  122. package/execution/__tests__/thread-memoization.test.js +262 -0
  123. package/execution/__tests__/thread-memoization.test.js.map +1 -0
  124. package/execution/__tests__/thread-skeleton.test.d.ts +2 -0
  125. package/execution/__tests__/thread-skeleton.test.d.ts.map +1 -0
  126. package/execution/__tests__/thread-skeleton.test.js +35 -0
  127. package/execution/__tests__/thread-skeleton.test.js.map +1 -0
  128. package/execution/__tests__/useExecutionStream.test.js +73 -10
  129. package/execution/__tests__/useExecutionStream.test.js.map +1 -1
  130. package/execution/__tests__/useSessionVariables-stability.test.d.ts +2 -0
  131. package/execution/__tests__/useSessionVariables-stability.test.d.ts.map +1 -0
  132. package/execution/__tests__/useSessionVariables-stability.test.js +69 -0
  133. package/execution/__tests__/useSessionVariables-stability.test.js.map +1 -0
  134. package/execution/__tests__/virtualized-thread.test.d.ts +2 -0
  135. package/execution/__tests__/virtualized-thread.test.d.ts.map +1 -0
  136. package/execution/__tests__/virtualized-thread.test.js +274 -0
  137. package/execution/__tests__/virtualized-thread.test.js.map +1 -0
  138. package/execution/index.d.ts +2 -0
  139. package/execution/index.d.ts.map +1 -1
  140. package/execution/index.js +1 -0
  141. package/execution/index.js.map +1 -1
  142. package/execution/useExecutionStream.d.ts +35 -10
  143. package/execution/useExecutionStream.d.ts.map +1 -1
  144. package/execution/useExecutionStream.js +79 -40
  145. package/execution/useExecutionStream.js.map +1 -1
  146. package/execution/useSessionVariables.d.ts.map +1 -1
  147. package/execution/useSessionVariables.js +4 -3
  148. package/execution/useSessionVariables.js.map +1 -1
  149. package/github/useGitHubConnection.d.ts.map +1 -1
  150. package/github/useGitHubConnection.js +5 -4
  151. package/github/useGitHubConnection.js.map +1 -1
  152. package/identity-account/index.d.ts +2 -0
  153. package/identity-account/index.d.ts.map +1 -0
  154. package/identity-account/index.js +2 -0
  155. package/identity-account/index.js.map +1 -0
  156. package/identity-account/useIdentityAccountGate.d.ts +81 -0
  157. package/identity-account/useIdentityAccountGate.d.ts.map +1 -0
  158. package/identity-account/useIdentityAccountGate.js +100 -0
  159. package/identity-account/useIdentityAccountGate.js.map +1 -0
  160. package/index.d.ts +10 -4
  161. package/index.d.ts.map +1 -1
  162. package/index.js +8 -2
  163. package/index.js.map +1 -1
  164. package/internal/FetchCacheProvider.d.ts +44 -0
  165. package/internal/FetchCacheProvider.d.ts.map +1 -0
  166. package/internal/FetchCacheProvider.js +61 -0
  167. package/internal/FetchCacheProvider.js.map +1 -0
  168. package/internal/JumpToLatestButton.d.ts +14 -0
  169. package/internal/JumpToLatestButton.d.ts.map +1 -0
  170. package/internal/JumpToLatestButton.js +19 -0
  171. package/internal/JumpToLatestButton.js.map +1 -0
  172. package/internal/ThreadItemWrapper.d.ts +20 -0
  173. package/internal/ThreadItemWrapper.d.ts.map +1 -0
  174. package/internal/ThreadItemWrapper.js +44 -0
  175. package/internal/ThreadItemWrapper.js.map +1 -0
  176. package/internal/VirtualizedThread.d.ts +25 -0
  177. package/internal/VirtualizedThread.d.ts.map +1 -0
  178. package/internal/VirtualizedThread.js +58 -0
  179. package/internal/VirtualizedThread.js.map +1 -0
  180. package/internal/__tests__/fetch-cache.test.d.ts +2 -0
  181. package/internal/__tests__/fetch-cache.test.d.ts.map +1 -0
  182. package/internal/__tests__/fetch-cache.test.js +182 -0
  183. package/internal/__tests__/fetch-cache.test.js.map +1 -0
  184. package/internal/__tests__/stream-controller.test.d.ts +2 -0
  185. package/internal/__tests__/stream-controller.test.d.ts.map +1 -0
  186. package/internal/__tests__/stream-controller.test.js +294 -0
  187. package/internal/__tests__/stream-controller.test.js.map +1 -0
  188. package/internal/__tests__/thread-animation.test.d.ts +2 -0
  189. package/internal/__tests__/thread-animation.test.d.ts.map +1 -0
  190. package/internal/__tests__/thread-animation.test.js +79 -0
  191. package/internal/__tests__/thread-animation.test.js.map +1 -0
  192. package/internal/__tests__/useAutoScroll.test.d.ts +2 -0
  193. package/internal/__tests__/useAutoScroll.test.d.ts.map +1 -0
  194. package/internal/__tests__/useAutoScroll.test.js +188 -0
  195. package/internal/__tests__/useAutoScroll.test.js.map +1 -0
  196. package/internal/__tests__/useFetch-cache.test.d.ts +2 -0
  197. package/internal/__tests__/useFetch-cache.test.d.ts.map +1 -0
  198. package/internal/__tests__/useFetch-cache.test.js +137 -0
  199. package/internal/__tests__/useFetch-cache.test.js.map +1 -0
  200. package/internal/dev/__tests__/use-key-stability.test.d.ts +2 -0
  201. package/internal/dev/__tests__/use-key-stability.test.d.ts.map +1 -0
  202. package/internal/dev/__tests__/use-key-stability.test.js +72 -0
  203. package/internal/dev/__tests__/use-key-stability.test.js.map +1 -0
  204. package/internal/dev/__tests__/use-render-tracer.test.d.ts +2 -0
  205. package/internal/dev/__tests__/use-render-tracer.test.d.ts.map +1 -0
  206. package/internal/dev/__tests__/use-render-tracer.test.js +55 -0
  207. package/internal/dev/__tests__/use-render-tracer.test.js.map +1 -0
  208. package/internal/dev/dom-counter.d.ts +14 -0
  209. package/internal/dev/dom-counter.d.ts.map +1 -0
  210. package/internal/dev/dom-counter.js +39 -0
  211. package/internal/dev/dom-counter.js.map +1 -0
  212. package/internal/dev/index.d.ts +6 -0
  213. package/internal/dev/index.d.ts.map +1 -0
  214. package/internal/dev/index.js +6 -0
  215. package/internal/dev/index.js.map +1 -0
  216. package/internal/dev/profiler-wrapper.d.ts +16 -0
  217. package/internal/dev/profiler-wrapper.d.ts.map +1 -0
  218. package/internal/dev/profiler-wrapper.js +31 -0
  219. package/internal/dev/profiler-wrapper.js.map +1 -0
  220. package/internal/dev/use-key-stability.d.ts +22 -0
  221. package/internal/dev/use-key-stability.d.ts.map +1 -0
  222. package/internal/dev/use-key-stability.js +67 -0
  223. package/internal/dev/use-key-stability.js.map +1 -0
  224. package/internal/dev/use-render-tracer.d.ts +13 -0
  225. package/internal/dev/use-render-tracer.d.ts.map +1 -0
  226. package/internal/dev/use-render-tracer.js +57 -0
  227. package/internal/dev/use-render-tracer.js.map +1 -0
  228. package/internal/dev/use-stream-rate.d.ts +23 -0
  229. package/internal/dev/use-stream-rate.d.ts.map +1 -0
  230. package/internal/dev/use-stream-rate.js +94 -0
  231. package/internal/dev/use-stream-rate.js.map +1 -0
  232. package/internal/fetch-cache.d.ts +72 -0
  233. package/internal/fetch-cache.d.ts.map +1 -0
  234. package/internal/fetch-cache.js +118 -0
  235. package/internal/fetch-cache.js.map +1 -0
  236. package/internal/store/__tests__/conversation-store.test.d.ts +2 -0
  237. package/internal/store/__tests__/conversation-store.test.d.ts.map +1 -0
  238. package/internal/store/__tests__/conversation-store.test.js +200 -0
  239. package/internal/store/__tests__/conversation-store.test.js.map +1 -0
  240. package/internal/store/__tests__/structural-share.test.d.ts +2 -0
  241. package/internal/store/__tests__/structural-share.test.d.ts.map +1 -0
  242. package/internal/store/__tests__/structural-share.test.js +368 -0
  243. package/internal/store/__tests__/structural-share.test.js.map +1 -0
  244. package/internal/store/conversation-store.d.ts +62 -0
  245. package/internal/store/conversation-store.d.ts.map +1 -0
  246. package/internal/store/conversation-store.js +95 -0
  247. package/internal/store/conversation-store.js.map +1 -0
  248. package/internal/store/index.d.ts +31 -0
  249. package/internal/store/index.d.ts.map +1 -0
  250. package/internal/store/index.js +54 -0
  251. package/internal/store/index.js.map +1 -0
  252. package/internal/store/structural-share.d.ts +13 -0
  253. package/internal/store/structural-share.d.ts.map +1 -0
  254. package/internal/store/structural-share.js +240 -0
  255. package/internal/store/structural-share.js.map +1 -0
  256. package/internal/stream-controller.d.ts +85 -0
  257. package/internal/stream-controller.d.ts.map +1 -0
  258. package/internal/stream-controller.js +146 -0
  259. package/internal/stream-controller.js.map +1 -0
  260. package/internal/useAutoScroll.d.ts +32 -0
  261. package/internal/useAutoScroll.d.ts.map +1 -0
  262. package/internal/useAutoScroll.js +97 -0
  263. package/internal/useAutoScroll.js.map +1 -0
  264. package/internal/useFetch.d.ts +14 -0
  265. package/internal/useFetch.d.ts.map +1 -1
  266. package/internal/useFetch.js +32 -2
  267. package/internal/useFetch.js.map +1 -1
  268. package/mcp-server/McpServerDetailView.d.ts.map +1 -1
  269. package/mcp-server/McpServerDetailView.js +3 -3
  270. package/mcp-server/McpServerDetailView.js.map +1 -1
  271. package/mcp-server/useMcpServerOAuthConnect.d.ts.map +1 -1
  272. package/mcp-server/useMcpServerOAuthConnect.js +37 -9
  273. package/mcp-server/useMcpServerOAuthConnect.js.map +1 -1
  274. package/package.json +7 -5
  275. package/session/__tests__/useNewSessionFlow.test.js +16 -0
  276. package/session/__tests__/useNewSessionFlow.test.js.map +1 -1
  277. package/session/__tests__/usePersistedModel.test.d.ts +2 -0
  278. package/session/__tests__/usePersistedModel.test.d.ts.map +1 -0
  279. package/session/__tests__/usePersistedModel.test.js +82 -0
  280. package/session/__tests__/usePersistedModel.test.js.map +1 -0
  281. package/session/__tests__/useSession.test.d.ts +2 -0
  282. package/session/__tests__/useSession.test.d.ts.map +1 -0
  283. package/session/__tests__/useSession.test.js +130 -0
  284. package/session/__tests__/useSession.test.js.map +1 -0
  285. package/session/useNewSessionFlow.d.ts.map +1 -1
  286. package/session/useNewSessionFlow.js +12 -6
  287. package/session/useNewSessionFlow.js.map +1 -1
  288. package/session/usePersistedModel.d.ts +3 -0
  289. package/session/usePersistedModel.d.ts.map +1 -1
  290. package/session/usePersistedModel.js +27 -2
  291. package/session/usePersistedModel.js.map +1 -1
  292. package/session/useSession.d.ts.map +1 -1
  293. package/session/useSession.js +1 -1
  294. package/session/useSession.js.map +1 -1
  295. package/session/useSessionConversation.d.ts.map +1 -1
  296. package/session/useSessionConversation.js +9 -1
  297. package/session/useSessionConversation.js.map +1 -1
  298. package/session/useSessionExecutions.d.ts.map +1 -1
  299. package/session/useSessionExecutions.js +1 -1
  300. package/session/useSessionExecutions.js.map +1 -1
  301. package/session/useSessionPageFlow.js +1 -1
  302. package/session/useSessionPageFlow.js.map +1 -1
  303. package/session/useSessionUsage.d.ts +24 -40
  304. package/session/useSessionUsage.d.ts.map +1 -1
  305. package/session/useSessionUsage.js +64 -97
  306. package/session/useSessionUsage.js.map +1 -1
  307. package/settings/BillingSection.d.ts +3 -0
  308. package/settings/BillingSection.d.ts.map +1 -0
  309. package/settings/BillingSection.js +3 -0
  310. package/settings/BillingSection.js.map +1 -0
  311. package/settings/index.d.ts +2 -0
  312. package/settings/index.d.ts.map +1 -1
  313. package/settings/index.js +1 -0
  314. package/settings/index.js.map +1 -1
  315. package/settings/settings-nav.js +1 -1
  316. package/settings/settings-nav.js.map +1 -1
  317. package/src/billing/AutoRechargeCard.tsx +274 -0
  318. package/src/billing/BillingSection.tsx +255 -0
  319. package/src/billing/CreditBalanceCard.tsx +81 -0
  320. package/src/billing/CreditLedgerTable.tsx +281 -0
  321. package/src/billing/CreditPackGrid.tsx +132 -0
  322. package/src/billing/LowBalanceBanner.tsx +67 -0
  323. package/src/billing/PaymentMethodCard.tsx +133 -0
  324. package/src/billing/credit-packs.ts +54 -0
  325. package/src/billing/format.ts +97 -0
  326. package/src/billing/index.ts +51 -0
  327. package/src/billing/useBillingAccount.ts +64 -0
  328. package/src/billing/useBillingUsageReport.ts +73 -0
  329. package/src/billing/useCreateBillingPortalSession.ts +76 -0
  330. package/src/billing/useCreateCheckoutSession.ts +101 -0
  331. package/src/billing/useCreditLedger.ts +79 -0
  332. package/src/billing/useCustomerModelPricing.ts +67 -0
  333. package/src/billing/useSetAutoRechargeConfig.ts +90 -0
  334. package/src/composer/ComposerToolbar.tsx +1 -1
  335. package/src/composer/SessionComposer.tsx +22 -4
  336. package/src/composer/__tests__/SessionComposer-memo.test.ts +26 -0
  337. package/src/execution/ApprovalCard.tsx +7 -3
  338. package/src/execution/ExecutionPhaseBadge.tsx +3 -2
  339. package/src/execution/MessageEntry.tsx +27 -16
  340. package/src/execution/MessageThread.tsx +308 -131
  341. package/src/execution/SetupProgress.tsx +3 -3
  342. package/src/execution/SubAgentSection.tsx +14 -6
  343. package/src/execution/ThreadSkeleton.tsx +73 -0
  344. package/src/execution/ToolCallGroup.tsx +36 -3
  345. package/src/execution/UsageWidget.tsx +1 -1
  346. package/src/execution/__tests__/message-entry.test.tsx +236 -0
  347. package/src/execution/__tests__/thread-keys.test.ts +409 -0
  348. package/src/execution/__tests__/thread-memoization.test.ts +320 -0
  349. package/src/execution/__tests__/thread-skeleton.test.tsx +44 -0
  350. package/src/execution/__tests__/useExecutionStream.test.tsx +109 -12
  351. package/src/execution/__tests__/useSessionVariables-stability.test.ts +95 -0
  352. package/src/execution/__tests__/virtualized-thread.test.tsx +401 -0
  353. package/src/execution/index.ts +3 -0
  354. package/src/execution/useExecutionStream.ts +123 -48
  355. package/src/execution/useSessionVariables.ts +17 -12
  356. package/src/github/useGitHubConnection.ts +18 -13
  357. package/src/identity-account/index.ts +5 -0
  358. package/src/identity-account/useIdentityAccountGate.ts +163 -0
  359. package/src/index.ts +73 -0
  360. package/src/internal/FetchCacheProvider.tsx +74 -0
  361. package/src/internal/JumpToLatestButton.tsx +61 -0
  362. package/src/internal/ThreadItemWrapper.tsx +65 -0
  363. package/src/internal/VirtualizedThread.tsx +162 -0
  364. package/src/internal/__tests__/fetch-cache.test.ts +230 -0
  365. package/src/internal/__tests__/stream-controller.test.ts +395 -0
  366. package/src/internal/__tests__/thread-animation.test.tsx +121 -0
  367. package/src/internal/__tests__/useAutoScroll.test.tsx +261 -0
  368. package/src/internal/__tests__/useFetch-cache.test.ts +214 -0
  369. package/src/internal/dev/__tests__/use-key-stability.test.ts +124 -0
  370. package/src/internal/dev/__tests__/use-render-tracer.test.ts +78 -0
  371. package/src/internal/dev/dom-counter.ts +47 -0
  372. package/src/internal/dev/index.ts +5 -0
  373. package/src/internal/dev/profiler-wrapper.tsx +52 -0
  374. package/src/internal/dev/use-key-stability.ts +86 -0
  375. package/src/internal/dev/use-render-tracer.ts +70 -0
  376. package/src/internal/dev/use-stream-rate.ts +138 -0
  377. package/src/internal/fetch-cache.ts +155 -0
  378. package/src/internal/store/__tests__/conversation-store.test.ts +257 -0
  379. package/src/internal/store/__tests__/structural-share.test.ts +454 -0
  380. package/src/internal/store/conversation-store.ts +128 -0
  381. package/src/internal/store/index.ts +68 -0
  382. package/src/internal/store/structural-share.ts +318 -0
  383. package/src/internal/stream-controller.ts +201 -0
  384. package/src/internal/useAutoScroll.ts +121 -0
  385. package/src/internal/useFetch.ts +51 -2
  386. package/src/mcp-server/McpServerDetailView.tsx +15 -0
  387. package/src/mcp-server/useMcpServerOAuthConnect.ts +37 -9
  388. package/src/session/__tests__/useNewSessionFlow.test.tsx +22 -0
  389. package/src/session/__tests__/usePersistedModel.test.tsx +117 -0
  390. package/src/session/__tests__/useSession.test.tsx +187 -0
  391. package/src/session/useNewSessionFlow.ts +12 -6
  392. package/src/session/usePersistedModel.ts +28 -2
  393. package/src/session/useSession.ts +1 -0
  394. package/src/session/useSessionConversation.ts +11 -2
  395. package/src/session/useSessionExecutions.ts +1 -0
  396. package/src/session/useSessionPageFlow.ts +1 -1
  397. package/src/session/useSessionUsage.ts +102 -123
  398. package/src/settings/BillingSection.tsx +4 -0
  399. package/src/settings/index.ts +2 -0
  400. package/src/settings/settings-nav.ts +1 -1
  401. package/src/styles.css +31 -0
  402. package/src/usage/AgentBreakdownList.tsx +147 -0
  403. package/src/usage/CreditRunwayIndicator.tsx +71 -0
  404. package/src/usage/ExportButton.tsx +115 -0
  405. package/src/usage/HarnessSplitCard.tsx +103 -0
  406. package/src/usage/OrgUsagePanel.tsx +109 -45
  407. package/src/usage/index.ts +15 -0
  408. package/src/usage/useExportCSV.ts +115 -0
  409. package/src/usage/useOrgUsageReport.ts +2 -1
  410. package/src/workspace/__tests__/useWorkspaceEntries-stability.test.ts +76 -0
  411. package/src/workspace/useWorkspaceEntries.ts +16 -11
  412. package/styles.css +1 -1
  413. package/usage/AgentBreakdownList.d.ts +21 -0
  414. package/usage/AgentBreakdownList.d.ts.map +1 -0
  415. package/usage/AgentBreakdownList.js +44 -0
  416. package/usage/AgentBreakdownList.js.map +1 -0
  417. package/usage/CreditRunwayIndicator.d.ts +21 -0
  418. package/usage/CreditRunwayIndicator.d.ts.map +1 -0
  419. package/usage/CreditRunwayIndicator.js +38 -0
  420. package/usage/CreditRunwayIndicator.js.map +1 -0
  421. package/usage/ExportButton.d.ts +20 -0
  422. package/usage/ExportButton.d.ts.map +1 -0
  423. package/usage/ExportButton.js +36 -0
  424. package/usage/ExportButton.js.map +1 -0
  425. package/usage/HarnessSplitCard.d.ts +17 -0
  426. package/usage/HarnessSplitCard.d.ts.map +1 -0
  427. package/usage/HarnessSplitCard.js +38 -0
  428. package/usage/HarnessSplitCard.js.map +1 -0
  429. package/usage/OrgUsagePanel.d.ts.map +1 -1
  430. package/usage/OrgUsagePanel.js +30 -22
  431. package/usage/OrgUsagePanel.js.map +1 -1
  432. package/usage/index.d.ts +10 -0
  433. package/usage/index.d.ts.map +1 -1
  434. package/usage/index.js +5 -0
  435. package/usage/index.js.map +1 -1
  436. package/usage/useExportCSV.d.ts +23 -0
  437. package/usage/useExportCSV.d.ts.map +1 -0
  438. package/usage/useExportCSV.js +81 -0
  439. package/usage/useExportCSV.js.map +1 -0
  440. package/usage/useOrgUsageReport.d.ts +2 -1
  441. package/usage/useOrgUsageReport.d.ts.map +1 -1
  442. package/usage/useOrgUsageReport.js +2 -1
  443. package/usage/useOrgUsageReport.js.map +1 -1
  444. package/workspace/__tests__/useWorkspaceEntries-stability.test.d.ts +2 -0
  445. package/workspace/__tests__/useWorkspaceEntries-stability.test.d.ts.map +1 -0
  446. package/workspace/__tests__/useWorkspaceEntries-stability.test.js +57 -0
  447. package/workspace/__tests__/useWorkspaceEntries-stability.test.js.map +1 -0
  448. package/workspace/useWorkspaceEntries.d.ts.map +1 -1
  449. package/workspace/useWorkspaceEntries.js +5 -4
  450. package/workspace/useWorkspaceEntries.js.map +1 -1
@@ -0,0 +1,137 @@
1
+ import { describe, it, expect, vi } from "vitest";
2
+ import { renderHook, act } from "@testing-library/react";
3
+ import { createElement } from "react";
4
+ import { FetchCacheContext } from "../FetchCacheProvider";
5
+ import { FetchCache } from "../fetch-cache";
6
+ import { useFetch } from "../useFetch";
7
+ // ---------------------------------------------------------------------------
8
+ // Helpers
9
+ // ---------------------------------------------------------------------------
10
+ async function flush() {
11
+ await act(async () => {
12
+ await Promise.resolve();
13
+ });
14
+ }
15
+ /**
16
+ * Build a wrapper that provides a specific FetchCache instance via
17
+ * context. Tests that need cross-mount caching share the same cache
18
+ * object, mirroring the production layout where FetchCacheProvider
19
+ * sits above the key-based remount boundary.
20
+ */
21
+ function createCacheWrapper(cache) {
22
+ return function Wrapper({ children }) {
23
+ return createElement(FetchCacheContext.Provider, { value: cache }, children);
24
+ };
25
+ }
26
+ // ---------------------------------------------------------------------------
27
+ // Tests: useFetch + FetchCacheProvider
28
+ // ---------------------------------------------------------------------------
29
+ describe("useFetch — cache integration", () => {
30
+ it("serves cached data on mount and skips isLoading", async () => {
31
+ const cache = new FetchCache();
32
+ const wrapper = createCacheWrapper(cache);
33
+ // First mount: populate the cache via a normal fetch.
34
+ const fetchFn = vi.fn(async () => "fetched-value");
35
+ const { result, unmount } = renderHook(() => useFetch(fetchFn, [], "init", { cacheKey: "test:1" }), { wrapper });
36
+ expect(result.current.isLoading).toBe(true);
37
+ await flush();
38
+ expect(result.current.data).toBe("fetched-value");
39
+ expect(result.current.isLoading).toBe(false);
40
+ unmount();
41
+ // Second mount (simulating remount after key change): should
42
+ // seed state from cache and never show isLoading.
43
+ const fetchFn2 = vi.fn(async () => "fresh-value");
44
+ const { result: result2 } = renderHook(() => useFetch(fetchFn2, [], "init", { cacheKey: "test:1" }), { wrapper });
45
+ // On mount, cached data is available immediately.
46
+ expect(result2.current.data).toBe("fetched-value");
47
+ expect(result2.current.isLoading).toBe(false);
48
+ expect(result2.current.isRefetching).toBe(true);
49
+ // Background fetch completes and updates data.
50
+ await flush();
51
+ expect(result2.current.data).toBe("fresh-value");
52
+ expect(result2.current.isRefetching).toBe(false);
53
+ });
54
+ it("falls back to initialData on cache miss", async () => {
55
+ const cache = new FetchCache();
56
+ const wrapper = createCacheWrapper(cache);
57
+ const fetchFn = vi.fn(async () => "fetched");
58
+ const { result } = renderHook(() => useFetch(fetchFn, [], "init", { cacheKey: "miss:1" }), { wrapper });
59
+ expect(result.current.data).toBe("init");
60
+ expect(result.current.isLoading).toBe(true);
61
+ await flush();
62
+ expect(result.current.data).toBe("fetched");
63
+ });
64
+ it("writes to cache on fetch success", async () => {
65
+ const cache = new FetchCache();
66
+ const wrapper = createCacheWrapper(cache);
67
+ const fetchFn = vi.fn(async () => "data-1");
68
+ const { result } = renderHook(() => useFetch(fetchFn, [], "init", { cacheKey: "write:1" }), { wrapper });
69
+ expect(cache.has("write:1")).toBe(false);
70
+ await flush();
71
+ expect(result.current.data).toBe("data-1");
72
+ expect(cache.get("write:1")).toBe("data-1");
73
+ });
74
+ it("does not interact with cache when cacheKey is undefined", async () => {
75
+ const cache = new FetchCache();
76
+ const wrapper = createCacheWrapper(cache);
77
+ const fetchFn = vi.fn(async () => "val");
78
+ const { result } = renderHook(() => useFetch(fetchFn, [], "init"), { wrapper });
79
+ await flush();
80
+ expect(result.current.data).toBe("val");
81
+ expect(cache.size).toBe(0);
82
+ });
83
+ it("works without FetchCacheProvider (graceful degradation)", async () => {
84
+ const fetchFn = vi.fn(async () => "value");
85
+ // No wrapper — no provider.
86
+ const { result } = renderHook(() => useFetch(fetchFn, [], "init", { cacheKey: "no-provider:1" }));
87
+ expect(result.current.data).toBe("init");
88
+ expect(result.current.isLoading).toBe(true);
89
+ await flush();
90
+ expect(result.current.data).toBe("value");
91
+ expect(result.current.isLoading).toBe(false);
92
+ });
93
+ it("does not read cache when fetchFn is null", async () => {
94
+ const cache = new FetchCache();
95
+ const wrapper = createCacheWrapper(cache);
96
+ const fetchFn1 = vi.fn(async () => "cached-val");
97
+ // First: populate cache.
98
+ const { unmount } = renderHook(() => useFetch(fetchFn1, [], "init", { cacheKey: "null-fn:1" }), { wrapper });
99
+ await flush();
100
+ unmount();
101
+ // Second: mount with fetchFn=null — should reset to initialData
102
+ // even though cache has data for this key.
103
+ const { result } = renderHook(() => useFetch(null, [], "fallback", {
104
+ cacheKey: "null-fn:1",
105
+ }), { wrapper });
106
+ expect(result.current.data).toBe("fallback");
107
+ expect(result.current.isLoading).toBe(false);
108
+ });
109
+ it("does not write to cache on fetch error", async () => {
110
+ const cache = new FetchCache();
111
+ const wrapper = createCacheWrapper(cache);
112
+ const fetchFn = vi.fn(async () => {
113
+ throw new Error("fail");
114
+ });
115
+ const { result } = renderHook(() => useFetch(fetchFn, [], "init", { cacheKey: "err:1" }), { wrapper });
116
+ await flush();
117
+ expect(result.current.error?.message).toBe("fail");
118
+ expect(cache.has("err:1")).toBe(false);
119
+ });
120
+ it("independent cache keys do not interfere", async () => {
121
+ const cache = new FetchCache();
122
+ const wrapper = createCacheWrapper(cache);
123
+ const fetchA = vi.fn(async () => "session-A");
124
+ const fetchB = vi.fn(async () => "session-B");
125
+ const { unmount: unmountA } = renderHook(() => useFetch(fetchA, [], null, { cacheKey: "session:A" }), { wrapper });
126
+ await flush();
127
+ unmountA();
128
+ const { unmount: unmountB } = renderHook(() => useFetch(fetchB, [], null, { cacheKey: "session:B" }), { wrapper });
129
+ await flush();
130
+ unmountB();
131
+ // Remount A — should get A's cached data, not B's.
132
+ const { result } = renderHook(() => useFetch(vi.fn(async () => "fresh-A"), [], null, { cacheKey: "session:A" }), { wrapper });
133
+ expect(result.current.data).toBe("session-A");
134
+ expect(result.current.isLoading).toBe(false);
135
+ });
136
+ });
137
+ //# sourceMappingURL=useFetch-cache.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"useFetch-cache.test.js","sourceRoot":"","sources":["../../../src/internal/__tests__/useFetch-cache.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=use-key-stability.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-key-stability.test.d.ts","sourceRoot":"","sources":["../../../../src/internal/dev/__tests__/use-key-stability.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,72 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
2
+ import { renderHook } from "@testing-library/react";
3
+ import { useKeyStability } from "../use-key-stability";
4
+ function makeItems(...specs) {
5
+ return specs.map(([key, kind]) => ({ key, kind }));
6
+ }
7
+ describe("useKeyStability", () => {
8
+ let warnSpy;
9
+ beforeEach(() => {
10
+ warnSpy = vi.spyOn(console, "warn").mockImplementation(() => { });
11
+ });
12
+ afterEach(() => {
13
+ warnSpy.mockRestore();
14
+ });
15
+ it("stays silent on first render (no previous to compare)", () => {
16
+ renderHook(() => useKeyStability(makeItems(["msg-1", "message"], ["msg-2", "message"])));
17
+ expect(warnSpy).not.toHaveBeenCalled();
18
+ });
19
+ it("stays silent when keys are stable across renders", () => {
20
+ const { rerender } = renderHook(({ items }) => useKeyStability(items), {
21
+ initialProps: {
22
+ items: makeItems(["msg-1", "message"], ["msg-2", "message"]),
23
+ },
24
+ });
25
+ rerender({
26
+ items: makeItems(["msg-1", "message"], ["msg-2", "message"]),
27
+ });
28
+ expect(warnSpy).not.toHaveBeenCalled();
29
+ });
30
+ it("stays silent when items are appended (no swap)", () => {
31
+ const { rerender } = renderHook(({ items }) => useKeyStability(items), {
32
+ initialProps: {
33
+ items: makeItems(["msg-1", "message"]),
34
+ },
35
+ });
36
+ rerender({
37
+ items: makeItems(["msg-1", "message"], ["msg-2", "message"]),
38
+ });
39
+ expect(warnSpy).not.toHaveBeenCalled();
40
+ });
41
+ it("warns when a key is swapped at the same index", () => {
42
+ const { rerender } = renderHook(({ items }) => useKeyStability(items), {
43
+ initialProps: {
44
+ items: makeItems(["e0-m0", "message"], ["e0-m1", "message"]),
45
+ },
46
+ });
47
+ rerender({
48
+ items: makeItems(["e0-m0", "message"], ["e1-m0", "message"]),
49
+ });
50
+ expect(warnSpy).toHaveBeenCalledTimes(1);
51
+ const msg = warnSpy.mock.calls[0][0];
52
+ expect(msg).toContain("[stgm:perf:keys]");
53
+ expect(msg).toContain("e0-m1");
54
+ expect(msg).toContain("e1-m0");
55
+ expect(msg).toContain("index 1");
56
+ });
57
+ it("warns about widespread instability when many keys swap", () => {
58
+ const { rerender } = renderHook(({ items }) => useKeyStability(items), {
59
+ initialProps: {
60
+ items: makeItems(["a", "message"], ["b", "message"], ["c", "message"], ["d", "message"]),
61
+ },
62
+ });
63
+ rerender({
64
+ items: makeItems(["w", "message"], ["x", "message"], ["y", "message"], ["z", "message"]),
65
+ });
66
+ const calls = warnSpy.mock.calls.map((c) => c[0]);
67
+ const summaryCall = calls.find((m) => m.includes("key swaps detected"));
68
+ expect(summaryCall).toBeDefined();
69
+ expect(summaryCall).toContain("4 key swaps");
70
+ });
71
+ });
72
+ //# sourceMappingURL=use-key-stability.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-key-stability.test.js","sourceRoot":"","sources":["../../../../src/internal/dev/__tests__/use-key-stability.test.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,2 @@
1
+ export {};
2
+ //# sourceMappingURL=use-render-tracer.test.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-render-tracer.test.d.ts","sourceRoot":"","sources":["../../../../src/internal/dev/__tests__/use-render-tracer.test.ts"],"names":[],"mappings":""}
@@ -0,0 +1,55 @@
1
+ import { describe, it, expect, vi, beforeEach, afterEach } from "vitest";
2
+ import { renderHook } from "@testing-library/react";
3
+ import { useRenderTracer } from "../use-render-tracer";
4
+ describe("useRenderTracer", () => {
5
+ let debugSpy;
6
+ beforeEach(() => {
7
+ debugSpy = vi.spyOn(console, "debug").mockImplementation(() => { });
8
+ });
9
+ afterEach(() => {
10
+ debugSpy.mockRestore();
11
+ });
12
+ it("logs on the first render", () => {
13
+ renderHook(() => useRenderTracer("TestComponent", { foo: "bar" }));
14
+ expect(debugSpy).toHaveBeenCalledTimes(1);
15
+ const msg = debugSpy.mock.calls[0][0];
16
+ expect(msg).toContain("[stgm:perf:render]");
17
+ expect(msg).toContain("TestComponent");
18
+ expect(msg).toContain("render=#1");
19
+ });
20
+ it("samples output — skips intermediate renders, logs every 10th", () => {
21
+ const { rerender } = renderHook(({ count }) => useRenderTracer("Counter", { count }), { initialProps: { count: 0 } });
22
+ debugSpy.mockClear();
23
+ // Renders 2-9: no output (sampled every 10th, first was #1)
24
+ for (let i = 1; i < 9; i++) {
25
+ rerender({ count: i });
26
+ }
27
+ expect(debugSpy).not.toHaveBeenCalled();
28
+ // Render #10: should log
29
+ rerender({ count: 9 });
30
+ expect(debugSpy).toHaveBeenCalledTimes(1);
31
+ const msg = debugSpy.mock.calls[0][0];
32
+ expect(msg).toContain("render=#10");
33
+ });
34
+ it("reports changed props on sampled renders", () => {
35
+ const stableRef = {};
36
+ const { rerender } = renderHook(({ a, b }) => useRenderTracer("Diff", { a, b }), { initialProps: { a: stableRef, b: 1 } });
37
+ debugSpy.mockClear();
38
+ // Drive to render #10 with only `b` changing each time
39
+ for (let i = 1; i < 9; i++) {
40
+ rerender({ a: stableRef, b: i + 1 });
41
+ }
42
+ // Render #10
43
+ rerender({ a: stableRef, b: 10 });
44
+ expect(debugSpy).toHaveBeenCalledTimes(1);
45
+ const msg = debugSpy.mock.calls[0][0];
46
+ expect(msg).toContain("changed=[b]");
47
+ });
48
+ it("includes primitive prop values in output", () => {
49
+ renderHook(() => useRenderTracer("Props", { count: 42, active: true }));
50
+ const msg = debugSpy.mock.calls[0][0];
51
+ expect(msg).toContain("count=42");
52
+ expect(msg).toContain("active=true");
53
+ });
54
+ });
55
+ //# sourceMappingURL=use-render-tracer.test.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-render-tracer.test.js","sourceRoot":"","sources":["../../../../src/internal/dev/__tests__/use-render-tracer.test.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,QAAQ,EAAE,EAAE,EAAE,MAAM,EAAE,EAAE,EAAE,UAAU,EAAE,SAAS,EAAE,MAAM,QAAQ,CAAC;AACzE,OAAO,EAAE,UAAU,EAAE,MAAM,wBAAwB,CAAC;AACpD,OAAO,EAAE,eAAe,EAAE,MAAM,sBAAsB,CAAC;AAEvD,QAAQ,CAAC,iBAAiB,EAAE,GAAG,EAAE;IAC/B,IAAI,QAAqC,CAAC;IAE1C,UAAU,CAAC,GAAG,EAAE;QACd,QAAQ,GAAG,EAAE,CAAC,KAAK,CAAC,OAAO,EAAE,OAAO,CAAC,CAAC,kBAAkB,CAAC,GAAG,EAAE,GAAE,CAAC,CAAC,CAAC;IACrE,CAAC,CAAC,CAAC;IAEH,SAAS,CAAC,GAAG,EAAE;QACb,QAAQ,CAAC,WAAW,EAAE,CAAC;IACzB,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0BAA0B,EAAE,GAAG,EAAE;QAClC,UAAU,CAAC,GAAG,EAAE,CAAC,eAAe,CAAC,eAAe,EAAE,EAAE,GAAG,EAAE,KAAK,EAAE,CAAC,CAAC,CAAC;QAEnE,MAAM,CAAC,QAAQ,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;QAC1C,MAAM,GAAG,GAAG,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAW,CAAC;QAChD,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,oBAAoB,CAAC,CAAC;QAC5C,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,eAAe,CAAC,CAAC;QACvC,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,WAAW,CAAC,CAAC;IACrC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,8DAA8D,EAAE,GAAG,EAAE;QACtE,MAAM,EAAE,QAAQ,EAAE,GAAG,UAAU,CAC7B,CAAC,EAAE,KAAK,EAAE,EAAE,EAAE,CAAC,eAAe,CAAC,SAAS,EAAE,EAAE,KAAK,EAAE,CAAC,EACpD,EAAE,YAAY,EAAE,EAAE,KAAK,EAAE,CAAC,EAAE,EAAE,CAC/B,CAAC;QAEF,QAAQ,CAAC,SAAS,EAAE,CAAC;QAErB,4DAA4D;QAC5D,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;YAC3B,QAAQ,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC;QACzB,CAAC;QACD,MAAM,CAAC,QAAQ,CAAC,CAAC,GAAG,CAAC,gBAAgB,EAAE,CAAC;QAExC,yBAAyB;QACzB,QAAQ,CAAC,EAAE,KAAK,EAAE,CAAC,EAAE,CAAC,CAAC;QACvB,MAAM,CAAC,QAAQ,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;QAC1C,MAAM,GAAG,GAAG,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAW,CAAC;QAChD,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,YAAY,CAAC,CAAC;IACtC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;QAClD,MAAM,SAAS,GAAG,EAAE,CAAC;QACrB,MAAM,EAAE,QAAQ,EAAE,GAAG,UAAU,CAC7B,CAAC,EAAE,CAAC,EAAE,CAAC,EAAE,EAAE,EAAE,CAAC,eAAe,CAAC,MAAM,EAAE,EAAE,CAAC,EAAE,CAAC,EAAE,CAAC,EAC/C,EAAE,YAAY,EAAE,EAAE,CAAC,EAAE,SAAoB,EAAE,CAAC,EAAE,CAAY,EAAE,EAAE,CAC/D,CAAC;QAEF,QAAQ,CAAC,SAAS,EAAE,CAAC;QAErB,uDAAuD;QACvD,KAAK,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,EAAE,EAAE,CAAC;YAC3B,QAAQ,CAAC,EAAE,CAAC,EAAE,SAAS,EAAE,CAAC,EAAE,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;QACvC,CAAC;QAED,aAAa;QACb,QAAQ,CAAC,EAAE,CAAC,EAAE,SAAS,EAAE,CAAC,EAAE,EAAE,EAAE,CAAC,CAAC;QAClC,MAAM,CAAC,QAAQ,CAAC,CAAC,qBAAqB,CAAC,CAAC,CAAC,CAAC;QAE1C,MAAM,GAAG,GAAG,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAW,CAAC;QAChD,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC;IACvC,CAAC,CAAC,CAAC;IAEH,EAAE,CAAC,0CAA0C,EAAE,GAAG,EAAE;QAClD,UAAU,CAAC,GAAG,EAAE,CACd,eAAe,CAAC,OAAO,EAAE,EAAE,KAAK,EAAE,EAAE,EAAE,MAAM,EAAE,IAAI,EAAE,CAAC,CACtD,CAAC;QAEF,MAAM,GAAG,GAAG,QAAQ,CAAC,IAAI,CAAC,KAAK,CAAC,CAAC,CAAC,CAAC,CAAC,CAAW,CAAC;QAChD,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,UAAU,CAAC,CAAC;QAClC,MAAM,CAAC,GAAG,CAAC,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC;IACvC,CAAC,CAAC,CAAC;AACL,CAAC,CAAC,CAAC"}
@@ -0,0 +1,14 @@
1
+ import { type RefObject } from "react";
2
+ /**
3
+ * Dev-only hook that periodically counts DOM nodes under a container
4
+ * element and logs the result.
5
+ *
6
+ * Measurement runs inside `requestIdleCallback` (with a `setTimeout`
7
+ * fallback for environments that don't support it) so it never blocks
8
+ * rendering or scroll.
9
+ *
10
+ * @param containerRef - Ref to the DOM element whose subtree to count.
11
+ * @param label - Identifier for the console output.
12
+ */
13
+ export declare function useDomNodeCount(containerRef: RefObject<HTMLElement | null>, label: string): void;
14
+ //# sourceMappingURL=dom-counter.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dom-counter.d.ts","sourceRoot":"","sources":["../../../src/internal/dev/dom-counter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,KAAK,SAAS,EAAqB,MAAM,OAAO,CAAC;AAO1D;;;;;;;;;;GAUG;AACH,wBAAgB,eAAe,CAC7B,YAAY,EAAE,SAAS,CAAC,WAAW,GAAG,IAAI,CAAC,EAC3C,KAAK,EAAE,MAAM,GACZ,IAAI,CAyBN"}
@@ -0,0 +1,39 @@
1
+ import { useEffect, useRef } from "react";
2
+ const DEV = process.env.NODE_ENV !== "production";
3
+ /** Log the node count every Nth trigger. */
4
+ const LOG_EVERY = 10;
5
+ /**
6
+ * Dev-only hook that periodically counts DOM nodes under a container
7
+ * element and logs the result.
8
+ *
9
+ * Measurement runs inside `requestIdleCallback` (with a `setTimeout`
10
+ * fallback for environments that don't support it) so it never blocks
11
+ * rendering or scroll.
12
+ *
13
+ * @param containerRef - Ref to the DOM element whose subtree to count.
14
+ * @param label - Identifier for the console output.
15
+ */
16
+ export function useDomNodeCount(containerRef, label) {
17
+ const triggerCountRef = useRef(0);
18
+ useEffect(() => {
19
+ if (!DEV)
20
+ return;
21
+ triggerCountRef.current += 1;
22
+ if (triggerCountRef.current % LOG_EVERY !== 0)
23
+ return;
24
+ const el = containerRef.current;
25
+ if (!el)
26
+ return;
27
+ const measure = () => {
28
+ const count = el.querySelectorAll("*").length;
29
+ console.debug(`[stgm:perf:dom] ${label} nodes=${count}`);
30
+ };
31
+ if (typeof requestIdleCallback === "function") {
32
+ const handle = requestIdleCallback(measure);
33
+ return () => cancelIdleCallback(handle);
34
+ }
35
+ const handle = setTimeout(measure, 0);
36
+ return () => clearTimeout(handle);
37
+ });
38
+ }
39
+ //# sourceMappingURL=dom-counter.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"dom-counter.js","sourceRoot":"","sources":["../../../src/internal/dev/dom-counter.ts"],"names":[],"mappings":"AAAA,OAAO,EAAkB,SAAS,EAAE,MAAM,EAAE,MAAM,OAAO,CAAC;AAE1D,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,YAAY,CAAC;AAElD,4CAA4C;AAC5C,MAAM,SAAS,GAAG,EAAE,CAAC;AAErB;;;;;;;;;;GAUG;AACH,MAAM,UAAU,eAAe,CAC7B,YAA2C,EAC3C,KAAa;IAEb,MAAM,eAAe,GAAG,MAAM,CAAC,CAAC,CAAC,CAAC;IAElC,SAAS,CAAC,GAAG,EAAE;QACb,IAAI,CAAC,GAAG;YAAE,OAAO;QAEjB,eAAe,CAAC,OAAO,IAAI,CAAC,CAAC;QAC7B,IAAI,eAAe,CAAC,OAAO,GAAG,SAAS,KAAK,CAAC;YAAE,OAAO;QAEtD,MAAM,EAAE,GAAG,YAAY,CAAC,OAAO,CAAC;QAChC,IAAI,CAAC,EAAE;YAAE,OAAO;QAEhB,MAAM,OAAO,GAAG,GAAG,EAAE;YACnB,MAAM,KAAK,GAAG,EAAE,CAAC,gBAAgB,CAAC,GAAG,CAAC,CAAC,MAAM,CAAC;YAC9C,OAAO,CAAC,KAAK,CAAC,mBAAmB,KAAK,WAAW,KAAK,EAAE,CAAC,CAAC;QAC5D,CAAC,CAAC;QAEF,IAAI,OAAO,mBAAmB,KAAK,UAAU,EAAE,CAAC;YAC9C,MAAM,MAAM,GAAG,mBAAmB,CAAC,OAAO,CAAC,CAAC;YAC5C,OAAO,GAAG,EAAE,CAAC,kBAAkB,CAAC,MAAM,CAAC,CAAC;QAC1C,CAAC;QAED,MAAM,MAAM,GAAG,UAAU,CAAC,OAAO,EAAE,CAAC,CAAC,CAAC;QACtC,OAAO,GAAG,EAAE,CAAC,YAAY,CAAC,MAAM,CAAC,CAAC;IACpC,CAAC,CAAC,CAAC;AACL,CAAC"}
@@ -0,0 +1,6 @@
1
+ export { useRenderTracer } from "./use-render-tracer";
2
+ export { useKeyStability } from "./use-key-stability";
3
+ export { useStreamRate, type StreamRateTracker } from "./use-stream-rate";
4
+ export { DevProfiler } from "./profiler-wrapper";
5
+ export { useDomNodeCount } from "./dom-counter";
6
+ //# sourceMappingURL=index.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../../../src/internal/dev/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,aAAa,EAAE,KAAK,iBAAiB,EAAE,MAAM,mBAAmB,CAAC;AAC1E,OAAO,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AACjD,OAAO,EAAE,eAAe,EAAE,MAAM,eAAe,CAAC"}
@@ -0,0 +1,6 @@
1
+ export { useRenderTracer } from "./use-render-tracer";
2
+ export { useKeyStability } from "./use-key-stability";
3
+ export { useStreamRate } from "./use-stream-rate";
4
+ export { DevProfiler } from "./profiler-wrapper";
5
+ export { useDomNodeCount } from "./dom-counter";
6
+ //# sourceMappingURL=index.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"index.js","sourceRoot":"","sources":["../../../src/internal/dev/index.ts"],"names":[],"mappings":"AAAA,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,eAAe,EAAE,MAAM,qBAAqB,CAAC;AACtD,OAAO,EAAE,aAAa,EAA0B,MAAM,mBAAmB,CAAC;AAC1E,OAAO,EAAE,WAAW,EAAE,MAAM,oBAAoB,CAAC;AACjD,OAAO,EAAE,eAAe,EAAE,MAAM,eAAe,CAAC"}
@@ -0,0 +1,16 @@
1
+ import { type ReactNode } from "react";
2
+ /**
3
+ * Thin wrapper around React's `<Profiler>`.
4
+ *
5
+ * In dev mode, logs commit-level timing (`actualDuration`,
6
+ * `baseDuration`, mount vs update phase) with sampled output.
7
+ *
8
+ * In production, React strips Profiler callbacks automatically, and
9
+ * our dev gate ensures this component renders children with zero
10
+ * overhead regardless.
11
+ */
12
+ export declare function DevProfiler({ id, children, }: {
13
+ readonly id: string;
14
+ readonly children: ReactNode;
15
+ }): string | number | bigint | boolean | Iterable<ReactNode> | Promise<string | number | bigint | boolean | import("react").ReactPortal | import("react").ReactElement<unknown, string | import("react").JSXElementConstructor<any>> | Iterable<ReactNode> | null | undefined> | import("react/jsx-runtime").JSX.Element | null | undefined;
16
+ //# sourceMappingURL=profiler-wrapper.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"profiler-wrapper.d.ts","sourceRoot":"","sources":["../../../src/internal/dev/profiler-wrapper.tsx"],"names":[],"mappings":"AAEA,OAAO,EAA2C,KAAK,SAAS,EAAE,MAAM,OAAO,CAAC;AAyBhF;;;;;;;;;GASG;AACH,wBAAgB,WAAW,CAAC,EAC1B,EAAE,EACF,QAAQ,GACT,EAAE;IACD,QAAQ,CAAC,EAAE,EAAE,MAAM,CAAC;IACpB,QAAQ,CAAC,QAAQ,EAAE,SAAS,CAAC;CAC9B,2UAQA"}
@@ -0,0 +1,31 @@
1
+ "use client";
2
+ import { jsx as _jsx } from "react/jsx-runtime";
3
+ import { Profiler } from "react";
4
+ const DEV = process.env.NODE_ENV !== "production";
5
+ /** Log every Nth commit to avoid console flood during streaming. */
6
+ const LOG_EVERY = 10;
7
+ let commitCount = 0;
8
+ const onRender = (id, phase, actualDuration, baseDuration) => {
9
+ commitCount += 1;
10
+ if (commitCount % LOG_EVERY === 0 || commitCount === 1) {
11
+ console.debug(`[stgm:perf:profiler] ${id} phase=${phase} ` +
12
+ `actualDuration=${actualDuration.toFixed(1)}ms ` +
13
+ `baseDuration=${baseDuration.toFixed(1)}ms`);
14
+ }
15
+ };
16
+ /**
17
+ * Thin wrapper around React's `<Profiler>`.
18
+ *
19
+ * In dev mode, logs commit-level timing (`actualDuration`,
20
+ * `baseDuration`, mount vs update phase) with sampled output.
21
+ *
22
+ * In production, React strips Profiler callbacks automatically, and
23
+ * our dev gate ensures this component renders children with zero
24
+ * overhead regardless.
25
+ */
26
+ export function DevProfiler({ id, children, }) {
27
+ if (!DEV)
28
+ return children;
29
+ return (_jsx(Profiler, { id: id, onRender: onRender, children: children }));
30
+ }
31
+ //# sourceMappingURL=profiler-wrapper.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"profiler-wrapper.js","sourceRoot":"","sources":["../../../src/internal/dev/profiler-wrapper.tsx"],"names":[],"mappings":"AAAA,YAAY,CAAC;;AAEb,OAAO,EAAE,QAAQ,EAAiD,MAAM,OAAO,CAAC;AAEhF,MAAM,GAAG,GAAG,OAAO,CAAC,GAAG,CAAC,QAAQ,KAAK,YAAY,CAAC;AAElD,oEAAoE;AACpE,MAAM,SAAS,GAAG,EAAE,CAAC;AAErB,IAAI,WAAW,GAAG,CAAC,CAAC;AAEpB,MAAM,QAAQ,GAA6B,CACzC,EAAE,EACF,KAAK,EACL,cAAc,EACd,YAAY,EACZ,EAAE;IACF,WAAW,IAAI,CAAC,CAAC;IACjB,IAAI,WAAW,GAAG,SAAS,KAAK,CAAC,IAAI,WAAW,KAAK,CAAC,EAAE,CAAC;QACvD,OAAO,CAAC,KAAK,CACX,wBAAwB,EAAE,WAAW,KAAK,IAAI;YAC5C,kBAAkB,cAAc,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM;YACjD,gBAAgB,YAAY,CAAC,OAAO,CAAC,CAAC,CAAC,IAAI,CAC9C,CAAC;IACJ,CAAC;AACH,CAAC,CAAC;AAEF;;;;;;;;;GASG;AACH,MAAM,UAAU,WAAW,CAAC,EAC1B,EAAE,EACF,QAAQ,GAIT;IACC,IAAI,CAAC,GAAG;QAAE,OAAO,QAAQ,CAAC;IAE1B,OAAO,CACL,KAAC,QAAQ,IAAC,EAAE,EAAE,EAAE,EAAE,QAAQ,EAAE,QAAQ,YACjC,QAAQ,GACA,CACZ,CAAC;AACJ,CAAC"}
@@ -0,0 +1,22 @@
1
+ /** Minimal shape expected from a keyed thread item. */
2
+ interface KeyedItem {
3
+ readonly key: string;
4
+ readonly kind: string;
5
+ }
6
+ /**
7
+ * Dev-only hook that detects key instability in a list of keyed items.
8
+ *
9
+ * Compares the current render's key set against the previous render's
10
+ * and warns when:
11
+ *
12
+ * - Duplicate keys exist in the current render (collision bug).
13
+ * - A key disappears and a new key appears at the same list index
14
+ * (suggests a remount caused by key change, not a logical add/remove).
15
+ * - The total number of key replacements in a single render exceeds a
16
+ * threshold, indicating widespread instability.
17
+ *
18
+ * In production builds the entire function body is dead-code-eliminated.
19
+ */
20
+ export declare function useKeyStability(items: readonly KeyedItem[]): void;
21
+ export {};
22
+ //# sourceMappingURL=use-key-stability.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-key-stability.d.ts","sourceRoot":"","sources":["../../../src/internal/dev/use-key-stability.ts"],"names":[],"mappings":"AAIA,uDAAuD;AACvD,UAAU,SAAS;IACjB,QAAQ,CAAC,GAAG,EAAE,MAAM,CAAC;IACrB,QAAQ,CAAC,IAAI,EAAE,MAAM,CAAC;CACvB;AAED;;;;;;;;;;;;;GAaG;AACH,wBAAgB,eAAe,CAAC,KAAK,EAAE,SAAS,SAAS,EAAE,GAAG,IAAI,CA6DjE"}
@@ -0,0 +1,67 @@
1
+ import { useRef } from "react";
2
+ const DEV = process.env.NODE_ENV !== "production";
3
+ /**
4
+ * Dev-only hook that detects key instability in a list of keyed items.
5
+ *
6
+ * Compares the current render's key set against the previous render's
7
+ * and warns when:
8
+ *
9
+ * - Duplicate keys exist in the current render (collision bug).
10
+ * - A key disappears and a new key appears at the same list index
11
+ * (suggests a remount caused by key change, not a logical add/remove).
12
+ * - The total number of key replacements in a single render exceeds a
13
+ * threshold, indicating widespread instability.
14
+ *
15
+ * In production builds the entire function body is dead-code-eliminated.
16
+ */
17
+ export function useKeyStability(items) {
18
+ const prevRef = useRef(null);
19
+ if (!DEV)
20
+ return;
21
+ // Detect duplicate keys within the current render.
22
+ const seen = new Map();
23
+ for (let i = 0; i < items.length; i++) {
24
+ const prevIdx = seen.get(items[i].key);
25
+ if (prevIdx !== undefined) {
26
+ console.warn(`[stgm:perf:keys] Duplicate key "${items[i].key}" at indices ${prevIdx} and ${i} ` +
27
+ `(${items[prevIdx].kind}, ${items[i].kind}). React will silently drop one.`);
28
+ }
29
+ else {
30
+ seen.set(items[i].key, i);
31
+ }
32
+ }
33
+ const prev = prevRef.current;
34
+ prevRef.current = items;
35
+ if (!prev || prev.length === 0)
36
+ return;
37
+ const prevKeys = new Map();
38
+ for (let i = 0; i < prev.length; i++) {
39
+ prevKeys.set(prev[i].key, i);
40
+ }
41
+ const curKeys = new Set(seen.keys());
42
+ const removed = [];
43
+ for (const key of prevKeys.keys()) {
44
+ if (!curKeys.has(key))
45
+ removed.push(key);
46
+ }
47
+ if (removed.length === 0)
48
+ return;
49
+ let swapCount = 0;
50
+ for (const removedKey of removed) {
51
+ const idx = prevKeys.get(removedKey);
52
+ if (idx < items.length) {
53
+ const replacement = items[idx];
54
+ if (!prevKeys.has(replacement.key)) {
55
+ swapCount++;
56
+ console.warn(`[stgm:perf:keys] Key swap at index ${idx}: ` +
57
+ `"${removedKey}" (${prev[idx].kind}) → "${replacement.key}" (${replacement.kind}). ` +
58
+ "This causes React to unmount/remount the row.");
59
+ }
60
+ }
61
+ }
62
+ if (swapCount > 3) {
63
+ console.warn(`[stgm:perf:keys] ${swapCount} key swaps detected in a single render. ` +
64
+ "Thread items may be using unstable keys.");
65
+ }
66
+ }
67
+ //# sourceMappingURL=use-key-stability.js.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-key-stability.js","sourceRoot":"","sources":["../../../src/internal/dev/use-key-stability.ts"],"names":[],"mappings":"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"}
@@ -0,0 +1,13 @@
1
+ /**
2
+ * Dev-only hook that tracks render count and reports which props
3
+ * changed (by shallow referential equality) since the last render.
4
+ *
5
+ * In production builds the function body is dead-code-eliminated by
6
+ * bundlers that replace `process.env.NODE_ENV` with `"production"`.
7
+ *
8
+ * @param componentName - Stable label for console output.
9
+ * @param props - Key/value map of props to track. Pass only the props
10
+ * you care about — typically the ones that drive re-renders.
11
+ */
12
+ export declare function useRenderTracer(componentName: string, props: Record<string, unknown>): void;
13
+ //# sourceMappingURL=use-render-tracer.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"use-render-tracer.d.ts","sourceRoot":"","sources":["../../../src/internal/dev/use-render-tracer.ts"],"names":[],"mappings":"AAUA;;;;;;;;;;GAUG;AACH,wBAAgB,eAAe,CAC7B,aAAa,EAAE,MAAM,EACrB,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,GAC7B,IAAI,CA6CN"}
@@ -0,0 +1,57 @@
1
+ import { useRef } from "react";
2
+ const DEV = process.env.NODE_ENV !== "production";
3
+ /**
4
+ * Sampling interval: log every Nth render to avoid flooding the
5
+ * console during high-frequency streaming (10-15 ticks/s).
6
+ */
7
+ const LOG_EVERY = 10;
8
+ /**
9
+ * Dev-only hook that tracks render count and reports which props
10
+ * changed (by shallow referential equality) since the last render.
11
+ *
12
+ * In production builds the function body is dead-code-eliminated by
13
+ * bundlers that replace `process.env.NODE_ENV` with `"production"`.
14
+ *
15
+ * @param componentName - Stable label for console output.
16
+ * @param props - Key/value map of props to track. Pass only the props
17
+ * you care about — typically the ones that drive re-renders.
18
+ */
19
+ export function useRenderTracer(componentName, props) {
20
+ const countRef = useRef(0);
21
+ const prevRef = useRef(null);
22
+ if (!DEV)
23
+ return;
24
+ countRef.current += 1;
25
+ const count = countRef.current;
26
+ const prev = prevRef.current;
27
+ if (count % LOG_EVERY === 0 || count === 1) {
28
+ const changed = [];
29
+ if (prev) {
30
+ for (const key of Object.keys(props)) {
31
+ if (!Object.is(props[key], prev[key])) {
32
+ changed.push(key);
33
+ }
34
+ }
35
+ }
36
+ const parts = [
37
+ `[stgm:perf:render] ${componentName}`,
38
+ `render=#${count}`,
39
+ ];
40
+ for (const [key, value] of Object.entries(props)) {
41
+ if (typeof value === "string") {
42
+ parts.push(`${key}=${value.length > 40 ? value.slice(0, 40) + "…" : value}`);
43
+ }
44
+ else if (typeof value === "number" || typeof value === "boolean") {
45
+ parts.push(`${key}=${String(value)}`);
46
+ }
47
+ }
48
+ if (prev) {
49
+ parts.push(changed.length > 0
50
+ ? `changed=[${changed.join(",")}]`
51
+ : "changed=[]");
52
+ }
53
+ console.debug(parts.join(" "));
54
+ }
55
+ prevRef.current = { ...props };
56
+ }
57
+ //# sourceMappingURL=use-render-tracer.js.map