@stigmer/react 0.3.4 → 0.4.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/billing/AutoRechargeCard.d.ts +38 -0
- package/billing/AutoRechargeCard.d.ts.map +1 -0
- package/billing/AutoRechargeCard.js +90 -0
- package/billing/AutoRechargeCard.js.map +1 -0
- package/billing/BillingSection.d.ts +32 -0
- package/billing/BillingSection.d.ts.map +1 -0
- package/billing/BillingSection.js +81 -0
- package/billing/BillingSection.js.map +1 -0
- package/billing/CreditBalanceCard.d.ts +25 -0
- package/billing/CreditBalanceCard.d.ts.map +1 -0
- package/billing/CreditBalanceCard.js +28 -0
- package/billing/CreditBalanceCard.js.map +1 -0
- package/billing/CreditLedgerTable.d.ts +22 -0
- package/billing/CreditLedgerTable.d.ts.map +1 -0
- package/billing/CreditLedgerTable.js +75 -0
- package/billing/CreditLedgerTable.js.map +1 -0
- package/billing/CreditPackGrid.d.ts +31 -0
- package/billing/CreditPackGrid.d.ts.map +1 -0
- package/billing/CreditPackGrid.js +35 -0
- package/billing/CreditPackGrid.js.map +1 -0
- package/billing/LowBalanceBanner.d.ts +26 -0
- package/billing/LowBalanceBanner.d.ts.map +1 -0
- package/billing/LowBalanceBanner.js +33 -0
- package/billing/LowBalanceBanner.js.map +1 -0
- package/billing/PaymentMethodCard.d.ts +35 -0
- package/billing/PaymentMethodCard.d.ts.map +1 -0
- package/billing/PaymentMethodCard.js +48 -0
- package/billing/PaymentMethodCard.js.map +1 -0
- package/billing/credit-packs.d.ts +25 -0
- package/billing/credit-packs.d.ts.map +1 -0
- package/billing/credit-packs.js +39 -0
- package/billing/credit-packs.js.map +1 -0
- package/billing/format.d.ts +39 -0
- package/billing/format.d.ts.map +1 -0
- package/billing/format.js +90 -0
- package/billing/format.js.map +1 -0
- package/billing/index.d.ts +32 -0
- package/billing/index.d.ts.map +1 -0
- package/billing/index.js +21 -0
- package/billing/index.js.map +1 -0
- package/billing/useBillingAccount.d.ts +40 -0
- package/billing/useBillingAccount.d.ts.map +1 -0
- package/billing/useBillingAccount.js +35 -0
- package/billing/useBillingAccount.js.map +1 -0
- package/billing/useBillingUsageReport.d.ts +42 -0
- package/billing/useBillingUsageReport.d.ts.map +1 -0
- package/billing/useBillingUsageReport.js +43 -0
- package/billing/useBillingUsageReport.js.map +1 -0
- package/billing/useCreateBillingPortalSession.d.ts +35 -0
- package/billing/useCreateBillingPortalSession.d.ts.map +1 -0
- package/billing/useCreateBillingPortalSession.js +50 -0
- package/billing/useCreateBillingPortalSession.js.map +1 -0
- package/billing/useCreateCheckoutSession.d.ts +54 -0
- package/billing/useCreateCheckoutSession.d.ts.map +1 -0
- package/billing/useCreateCheckoutSession.js +58 -0
- package/billing/useCreateCheckoutSession.js.map +1 -0
- package/billing/useCreditLedger.d.ts +48 -0
- package/billing/useCreditLedger.d.ts.map +1 -0
- package/billing/useCreditLedger.js +39 -0
- package/billing/useCreditLedger.js.map +1 -0
- package/billing/useCustomerModelPricing.d.ts +41 -0
- package/billing/useCustomerModelPricing.d.ts.map +1 -0
- package/billing/useCustomerModelPricing.js +37 -0
- package/billing/useCustomerModelPricing.js.map +1 -0
- package/billing/useSetAutoRechargeConfig.d.ts +50 -0
- package/billing/useSetAutoRechargeConfig.d.ts.map +1 -0
- package/billing/useSetAutoRechargeConfig.js +53 -0
- package/billing/useSetAutoRechargeConfig.js.map +1 -0
- package/composer/ComposerToolbar.js +1 -1
- package/composer/ComposerToolbar.js.map +1 -1
- package/composer/SessionComposer.d.ts +1 -1
- package/composer/SessionComposer.d.ts.map +1 -1
- package/composer/SessionComposer.js +19 -4
- package/composer/SessionComposer.js.map +1 -1
- package/composer/__tests__/SessionComposer-memo.test.d.ts +2 -0
- package/composer/__tests__/SessionComposer-memo.test.d.ts.map +1 -0
- package/composer/__tests__/SessionComposer-memo.test.js +23 -0
- package/composer/__tests__/SessionComposer-memo.test.js.map +1 -0
- package/execution/ApprovalCard.d.ts +5 -1
- package/execution/ApprovalCard.d.ts.map +1 -1
- package/execution/ApprovalCard.js +7 -3
- package/execution/ApprovalCard.js.map +1 -1
- package/execution/ExecutionPhaseBadge.d.ts +1 -1
- package/execution/ExecutionPhaseBadge.d.ts.map +1 -1
- package/execution/ExecutionPhaseBadge.js +3 -2
- package/execution/ExecutionPhaseBadge.js.map +1 -1
- package/execution/MessageEntry.d.ts +7 -3
- package/execution/MessageEntry.d.ts.map +1 -1
- package/execution/MessageEntry.js +19 -8
- package/execution/MessageEntry.js.map +1 -1
- package/execution/MessageThread.d.ts +84 -3
- package/execution/MessageThread.d.ts.map +1 -1
- package/execution/MessageThread.js +113 -65
- package/execution/MessageThread.js.map +1 -1
- package/execution/SetupProgress.d.ts +1 -1
- package/execution/SetupProgress.d.ts.map +1 -1
- package/execution/SetupProgress.js +3 -3
- package/execution/SetupProgress.js.map +1 -1
- package/execution/SubAgentSection.d.ts +5 -1
- package/execution/SubAgentSection.d.ts.map +1 -1
- package/execution/SubAgentSection.js +13 -7
- package/execution/SubAgentSection.js.map +1 -1
- package/execution/ThreadSkeleton.d.ts +22 -0
- package/execution/ThreadSkeleton.d.ts.map +1 -0
- package/execution/ThreadSkeleton.js +26 -0
- package/execution/ThreadSkeleton.js.map +1 -0
- package/execution/ToolCallGroup.d.ts +16 -1
- package/execution/ToolCallGroup.d.ts.map +1 -1
- package/execution/ToolCallGroup.js +31 -3
- package/execution/ToolCallGroup.js.map +1 -1
- package/execution/UsageWidget.d.ts +1 -1
- package/execution/__tests__/message-entry.test.d.ts +2 -0
- package/execution/__tests__/message-entry.test.d.ts.map +1 -0
- package/execution/__tests__/message-entry.test.js +178 -0
- package/execution/__tests__/message-entry.test.js.map +1 -0
- package/execution/__tests__/thread-keys.test.d.ts +2 -0
- package/execution/__tests__/thread-keys.test.d.ts.map +1 -0
- package/execution/__tests__/thread-keys.test.js +289 -0
- package/execution/__tests__/thread-keys.test.js.map +1 -0
- package/execution/__tests__/thread-memoization.test.d.ts +2 -0
- package/execution/__tests__/thread-memoization.test.d.ts.map +1 -0
- package/execution/__tests__/thread-memoization.test.js +262 -0
- package/execution/__tests__/thread-memoization.test.js.map +1 -0
- package/execution/__tests__/thread-skeleton.test.d.ts +2 -0
- package/execution/__tests__/thread-skeleton.test.d.ts.map +1 -0
- package/execution/__tests__/thread-skeleton.test.js +35 -0
- package/execution/__tests__/thread-skeleton.test.js.map +1 -0
- package/execution/__tests__/useExecutionStream.test.js +73 -10
- package/execution/__tests__/useExecutionStream.test.js.map +1 -1
- package/execution/__tests__/useSessionVariables-stability.test.d.ts +2 -0
- package/execution/__tests__/useSessionVariables-stability.test.d.ts.map +1 -0
- package/execution/__tests__/useSessionVariables-stability.test.js +69 -0
- package/execution/__tests__/useSessionVariables-stability.test.js.map +1 -0
- package/execution/__tests__/virtualized-thread.test.d.ts +2 -0
- package/execution/__tests__/virtualized-thread.test.d.ts.map +1 -0
- package/execution/__tests__/virtualized-thread.test.js +274 -0
- package/execution/__tests__/virtualized-thread.test.js.map +1 -0
- package/execution/index.d.ts +2 -0
- package/execution/index.d.ts.map +1 -1
- package/execution/index.js +1 -0
- package/execution/index.js.map +1 -1
- package/execution/useExecutionStream.d.ts +35 -10
- package/execution/useExecutionStream.d.ts.map +1 -1
- package/execution/useExecutionStream.js +79 -40
- package/execution/useExecutionStream.js.map +1 -1
- package/execution/useSessionVariables.d.ts.map +1 -1
- package/execution/useSessionVariables.js +4 -3
- package/execution/useSessionVariables.js.map +1 -1
- package/github/useGitHubConnection.d.ts.map +1 -1
- package/github/useGitHubConnection.js +5 -4
- package/github/useGitHubConnection.js.map +1 -1
- package/identity-account/index.d.ts +2 -0
- package/identity-account/index.d.ts.map +1 -0
- package/identity-account/index.js +2 -0
- package/identity-account/index.js.map +1 -0
- package/identity-account/useIdentityAccountGate.d.ts +81 -0
- package/identity-account/useIdentityAccountGate.d.ts.map +1 -0
- package/identity-account/useIdentityAccountGate.js +100 -0
- package/identity-account/useIdentityAccountGate.js.map +1 -0
- package/index.d.ts +10 -4
- package/index.d.ts.map +1 -1
- package/index.js +8 -2
- package/index.js.map +1 -1
- package/internal/FetchCacheProvider.d.ts +44 -0
- package/internal/FetchCacheProvider.d.ts.map +1 -0
- package/internal/FetchCacheProvider.js +61 -0
- package/internal/FetchCacheProvider.js.map +1 -0
- package/internal/JumpToLatestButton.d.ts +14 -0
- package/internal/JumpToLatestButton.d.ts.map +1 -0
- package/internal/JumpToLatestButton.js +19 -0
- package/internal/JumpToLatestButton.js.map +1 -0
- package/internal/ThreadItemWrapper.d.ts +20 -0
- package/internal/ThreadItemWrapper.d.ts.map +1 -0
- package/internal/ThreadItemWrapper.js +44 -0
- package/internal/ThreadItemWrapper.js.map +1 -0
- package/internal/VirtualizedThread.d.ts +25 -0
- package/internal/VirtualizedThread.d.ts.map +1 -0
- package/internal/VirtualizedThread.js +58 -0
- package/internal/VirtualizedThread.js.map +1 -0
- package/internal/__tests__/fetch-cache.test.d.ts +2 -0
- package/internal/__tests__/fetch-cache.test.d.ts.map +1 -0
- package/internal/__tests__/fetch-cache.test.js +182 -0
- package/internal/__tests__/fetch-cache.test.js.map +1 -0
- package/internal/__tests__/stream-controller.test.d.ts +2 -0
- package/internal/__tests__/stream-controller.test.d.ts.map +1 -0
- package/internal/__tests__/stream-controller.test.js +294 -0
- package/internal/__tests__/stream-controller.test.js.map +1 -0
- package/internal/__tests__/thread-animation.test.d.ts +2 -0
- package/internal/__tests__/thread-animation.test.d.ts.map +1 -0
- package/internal/__tests__/thread-animation.test.js +79 -0
- package/internal/__tests__/thread-animation.test.js.map +1 -0
- package/internal/__tests__/useAutoScroll.test.d.ts +2 -0
- package/internal/__tests__/useAutoScroll.test.d.ts.map +1 -0
- package/internal/__tests__/useAutoScroll.test.js +188 -0
- package/internal/__tests__/useAutoScroll.test.js.map +1 -0
- package/internal/__tests__/useFetch-cache.test.d.ts +2 -0
- package/internal/__tests__/useFetch-cache.test.d.ts.map +1 -0
- package/internal/__tests__/useFetch-cache.test.js +137 -0
- package/internal/__tests__/useFetch-cache.test.js.map +1 -0
- package/internal/dev/__tests__/use-key-stability.test.d.ts +2 -0
- package/internal/dev/__tests__/use-key-stability.test.d.ts.map +1 -0
- package/internal/dev/__tests__/use-key-stability.test.js +72 -0
- package/internal/dev/__tests__/use-key-stability.test.js.map +1 -0
- package/internal/dev/__tests__/use-render-tracer.test.d.ts +2 -0
- package/internal/dev/__tests__/use-render-tracer.test.d.ts.map +1 -0
- package/internal/dev/__tests__/use-render-tracer.test.js +55 -0
- package/internal/dev/__tests__/use-render-tracer.test.js.map +1 -0
- package/internal/dev/dom-counter.d.ts +14 -0
- package/internal/dev/dom-counter.d.ts.map +1 -0
- package/internal/dev/dom-counter.js +39 -0
- package/internal/dev/dom-counter.js.map +1 -0
- package/internal/dev/index.d.ts +6 -0
- package/internal/dev/index.d.ts.map +1 -0
- package/internal/dev/index.js +6 -0
- package/internal/dev/index.js.map +1 -0
- package/internal/dev/profiler-wrapper.d.ts +16 -0
- package/internal/dev/profiler-wrapper.d.ts.map +1 -0
- package/internal/dev/profiler-wrapper.js +31 -0
- package/internal/dev/profiler-wrapper.js.map +1 -0
- package/internal/dev/use-key-stability.d.ts +22 -0
- package/internal/dev/use-key-stability.d.ts.map +1 -0
- package/internal/dev/use-key-stability.js +67 -0
- package/internal/dev/use-key-stability.js.map +1 -0
- package/internal/dev/use-render-tracer.d.ts +13 -0
- package/internal/dev/use-render-tracer.d.ts.map +1 -0
- package/internal/dev/use-render-tracer.js +57 -0
- package/internal/dev/use-render-tracer.js.map +1 -0
- package/internal/dev/use-stream-rate.d.ts +23 -0
- package/internal/dev/use-stream-rate.d.ts.map +1 -0
- package/internal/dev/use-stream-rate.js +94 -0
- package/internal/dev/use-stream-rate.js.map +1 -0
- package/internal/fetch-cache.d.ts +72 -0
- package/internal/fetch-cache.d.ts.map +1 -0
- package/internal/fetch-cache.js +118 -0
- package/internal/fetch-cache.js.map +1 -0
- package/internal/store/__tests__/conversation-store.test.d.ts +2 -0
- package/internal/store/__tests__/conversation-store.test.d.ts.map +1 -0
- package/internal/store/__tests__/conversation-store.test.js +200 -0
- package/internal/store/__tests__/conversation-store.test.js.map +1 -0
- package/internal/store/__tests__/structural-share.test.d.ts +2 -0
- package/internal/store/__tests__/structural-share.test.d.ts.map +1 -0
- package/internal/store/__tests__/structural-share.test.js +368 -0
- package/internal/store/__tests__/structural-share.test.js.map +1 -0
- package/internal/store/conversation-store.d.ts +62 -0
- package/internal/store/conversation-store.d.ts.map +1 -0
- package/internal/store/conversation-store.js +95 -0
- package/internal/store/conversation-store.js.map +1 -0
- package/internal/store/index.d.ts +31 -0
- package/internal/store/index.d.ts.map +1 -0
- package/internal/store/index.js +54 -0
- package/internal/store/index.js.map +1 -0
- package/internal/store/structural-share.d.ts +13 -0
- package/internal/store/structural-share.d.ts.map +1 -0
- package/internal/store/structural-share.js +240 -0
- package/internal/store/structural-share.js.map +1 -0
- package/internal/stream-controller.d.ts +85 -0
- package/internal/stream-controller.d.ts.map +1 -0
- package/internal/stream-controller.js +146 -0
- package/internal/stream-controller.js.map +1 -0
- package/internal/useAutoScroll.d.ts +32 -0
- package/internal/useAutoScroll.d.ts.map +1 -0
- package/internal/useAutoScroll.js +97 -0
- package/internal/useAutoScroll.js.map +1 -0
- package/internal/useFetch.d.ts +14 -0
- package/internal/useFetch.d.ts.map +1 -1
- package/internal/useFetch.js +32 -2
- package/internal/useFetch.js.map +1 -1
- package/package.json +7 -5
- package/session/__tests__/useNewSessionFlow.test.js +16 -0
- package/session/__tests__/useNewSessionFlow.test.js.map +1 -1
- package/session/__tests__/usePersistedModel.test.d.ts +2 -0
- package/session/__tests__/usePersistedModel.test.d.ts.map +1 -0
- package/session/__tests__/usePersistedModel.test.js +82 -0
- package/session/__tests__/usePersistedModel.test.js.map +1 -0
- package/session/__tests__/useSession.test.d.ts +2 -0
- package/session/__tests__/useSession.test.d.ts.map +1 -0
- package/session/__tests__/useSession.test.js +130 -0
- package/session/__tests__/useSession.test.js.map +1 -0
- package/session/useNewSessionFlow.d.ts.map +1 -1
- package/session/useNewSessionFlow.js +12 -6
- package/session/useNewSessionFlow.js.map +1 -1
- package/session/usePersistedModel.d.ts +3 -0
- package/session/usePersistedModel.d.ts.map +1 -1
- package/session/usePersistedModel.js +27 -2
- package/session/usePersistedModel.js.map +1 -1
- package/session/useSession.d.ts.map +1 -1
- package/session/useSession.js +1 -1
- package/session/useSession.js.map +1 -1
- package/session/useSessionConversation.d.ts.map +1 -1
- package/session/useSessionConversation.js +9 -1
- package/session/useSessionConversation.js.map +1 -1
- package/session/useSessionExecutions.d.ts.map +1 -1
- package/session/useSessionExecutions.js +1 -1
- package/session/useSessionExecutions.js.map +1 -1
- package/session/useSessionPageFlow.js +1 -1
- package/session/useSessionPageFlow.js.map +1 -1
- package/session/useSessionUsage.d.ts +24 -40
- package/session/useSessionUsage.d.ts.map +1 -1
- package/session/useSessionUsage.js +64 -97
- package/session/useSessionUsage.js.map +1 -1
- package/settings/BillingSection.d.ts +3 -0
- package/settings/BillingSection.d.ts.map +1 -0
- package/settings/BillingSection.js +3 -0
- package/settings/BillingSection.js.map +1 -0
- package/settings/index.d.ts +2 -0
- package/settings/index.d.ts.map +1 -1
- package/settings/index.js +1 -0
- package/settings/index.js.map +1 -1
- package/settings/settings-nav.js +1 -1
- package/settings/settings-nav.js.map +1 -1
- package/src/billing/AutoRechargeCard.tsx +274 -0
- package/src/billing/BillingSection.tsx +255 -0
- package/src/billing/CreditBalanceCard.tsx +81 -0
- package/src/billing/CreditLedgerTable.tsx +281 -0
- package/src/billing/CreditPackGrid.tsx +132 -0
- package/src/billing/LowBalanceBanner.tsx +67 -0
- package/src/billing/PaymentMethodCard.tsx +133 -0
- package/src/billing/credit-packs.ts +54 -0
- package/src/billing/format.ts +97 -0
- package/src/billing/index.ts +51 -0
- package/src/billing/useBillingAccount.ts +64 -0
- package/src/billing/useBillingUsageReport.ts +73 -0
- package/src/billing/useCreateBillingPortalSession.ts +76 -0
- package/src/billing/useCreateCheckoutSession.ts +101 -0
- package/src/billing/useCreditLedger.ts +79 -0
- package/src/billing/useCustomerModelPricing.ts +67 -0
- package/src/billing/useSetAutoRechargeConfig.ts +90 -0
- package/src/composer/ComposerToolbar.tsx +1 -1
- package/src/composer/SessionComposer.tsx +22 -4
- package/src/composer/__tests__/SessionComposer-memo.test.ts +26 -0
- package/src/execution/ApprovalCard.tsx +7 -3
- package/src/execution/ExecutionPhaseBadge.tsx +3 -2
- package/src/execution/MessageEntry.tsx +27 -16
- package/src/execution/MessageThread.tsx +308 -131
- package/src/execution/SetupProgress.tsx +3 -3
- package/src/execution/SubAgentSection.tsx +14 -6
- package/src/execution/ThreadSkeleton.tsx +73 -0
- package/src/execution/ToolCallGroup.tsx +36 -3
- package/src/execution/UsageWidget.tsx +1 -1
- package/src/execution/__tests__/message-entry.test.tsx +236 -0
- package/src/execution/__tests__/thread-keys.test.ts +409 -0
- package/src/execution/__tests__/thread-memoization.test.ts +320 -0
- package/src/execution/__tests__/thread-skeleton.test.tsx +44 -0
- package/src/execution/__tests__/useExecutionStream.test.tsx +109 -12
- package/src/execution/__tests__/useSessionVariables-stability.test.ts +95 -0
- package/src/execution/__tests__/virtualized-thread.test.tsx +401 -0
- package/src/execution/index.ts +3 -0
- package/src/execution/useExecutionStream.ts +123 -48
- package/src/execution/useSessionVariables.ts +17 -12
- package/src/github/useGitHubConnection.ts +18 -13
- package/src/identity-account/index.ts +5 -0
- package/src/identity-account/useIdentityAccountGate.ts +163 -0
- package/src/index.ts +73 -0
- package/src/internal/FetchCacheProvider.tsx +74 -0
- package/src/internal/JumpToLatestButton.tsx +61 -0
- package/src/internal/ThreadItemWrapper.tsx +65 -0
- package/src/internal/VirtualizedThread.tsx +162 -0
- package/src/internal/__tests__/fetch-cache.test.ts +230 -0
- package/src/internal/__tests__/stream-controller.test.ts +395 -0
- package/src/internal/__tests__/thread-animation.test.tsx +121 -0
- package/src/internal/__tests__/useAutoScroll.test.tsx +261 -0
- package/src/internal/__tests__/useFetch-cache.test.ts +214 -0
- package/src/internal/dev/__tests__/use-key-stability.test.ts +124 -0
- package/src/internal/dev/__tests__/use-render-tracer.test.ts +78 -0
- package/src/internal/dev/dom-counter.ts +47 -0
- package/src/internal/dev/index.ts +5 -0
- package/src/internal/dev/profiler-wrapper.tsx +52 -0
- package/src/internal/dev/use-key-stability.ts +86 -0
- package/src/internal/dev/use-render-tracer.ts +70 -0
- package/src/internal/dev/use-stream-rate.ts +138 -0
- package/src/internal/fetch-cache.ts +155 -0
- package/src/internal/store/__tests__/conversation-store.test.ts +257 -0
- package/src/internal/store/__tests__/structural-share.test.ts +454 -0
- package/src/internal/store/conversation-store.ts +128 -0
- package/src/internal/store/index.ts +68 -0
- package/src/internal/store/structural-share.ts +318 -0
- package/src/internal/stream-controller.ts +201 -0
- package/src/internal/useAutoScroll.ts +121 -0
- package/src/internal/useFetch.ts +51 -2
- package/src/session/__tests__/useNewSessionFlow.test.tsx +22 -0
- package/src/session/__tests__/usePersistedModel.test.tsx +117 -0
- package/src/session/__tests__/useSession.test.tsx +187 -0
- package/src/session/useNewSessionFlow.ts +12 -6
- package/src/session/usePersistedModel.ts +28 -2
- package/src/session/useSession.ts +1 -0
- package/src/session/useSessionConversation.ts +11 -2
- package/src/session/useSessionExecutions.ts +1 -0
- package/src/session/useSessionPageFlow.ts +1 -1
- package/src/session/useSessionUsage.ts +102 -123
- package/src/settings/BillingSection.tsx +4 -0
- package/src/settings/index.ts +2 -0
- package/src/settings/settings-nav.ts +1 -1
- package/src/styles.css +31 -0
- package/src/usage/AgentBreakdownList.tsx +147 -0
- package/src/usage/CreditRunwayIndicator.tsx +71 -0
- package/src/usage/ExportButton.tsx +115 -0
- package/src/usage/HarnessSplitCard.tsx +103 -0
- package/src/usage/OrgUsagePanel.tsx +109 -45
- package/src/usage/index.ts +15 -0
- package/src/usage/useExportCSV.ts +115 -0
- package/src/usage/useOrgUsageReport.ts +2 -1
- package/src/workspace/__tests__/useWorkspaceEntries-stability.test.ts +76 -0
- package/src/workspace/useWorkspaceEntries.ts +16 -11
- package/styles.css +1 -1
- package/usage/AgentBreakdownList.d.ts +21 -0
- package/usage/AgentBreakdownList.d.ts.map +1 -0
- package/usage/AgentBreakdownList.js +44 -0
- package/usage/AgentBreakdownList.js.map +1 -0
- package/usage/CreditRunwayIndicator.d.ts +21 -0
- package/usage/CreditRunwayIndicator.d.ts.map +1 -0
- package/usage/CreditRunwayIndicator.js +38 -0
- package/usage/CreditRunwayIndicator.js.map +1 -0
- package/usage/ExportButton.d.ts +20 -0
- package/usage/ExportButton.d.ts.map +1 -0
- package/usage/ExportButton.js +36 -0
- package/usage/ExportButton.js.map +1 -0
- package/usage/HarnessSplitCard.d.ts +17 -0
- package/usage/HarnessSplitCard.d.ts.map +1 -0
- package/usage/HarnessSplitCard.js +38 -0
- package/usage/HarnessSplitCard.js.map +1 -0
- package/usage/OrgUsagePanel.d.ts.map +1 -1
- package/usage/OrgUsagePanel.js +30 -22
- package/usage/OrgUsagePanel.js.map +1 -1
- package/usage/index.d.ts +10 -0
- package/usage/index.d.ts.map +1 -1
- package/usage/index.js +5 -0
- package/usage/index.js.map +1 -1
- package/usage/useExportCSV.d.ts +23 -0
- package/usage/useExportCSV.d.ts.map +1 -0
- package/usage/useExportCSV.js +81 -0
- package/usage/useExportCSV.js.map +1 -0
- package/usage/useOrgUsageReport.d.ts +2 -1
- package/usage/useOrgUsageReport.d.ts.map +1 -1
- package/usage/useOrgUsageReport.js +2 -1
- package/usage/useOrgUsageReport.js.map +1 -1
- package/workspace/__tests__/useWorkspaceEntries-stability.test.d.ts +2 -0
- package/workspace/__tests__/useWorkspaceEntries-stability.test.d.ts.map +1 -0
- package/workspace/__tests__/useWorkspaceEntries-stability.test.js +57 -0
- package/workspace/__tests__/useWorkspaceEntries-stability.test.js.map +1 -0
- package/workspace/useWorkspaceEntries.d.ts.map +1 -1
- package/workspace/useWorkspaceEntries.js +5 -4
- 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 @@
|
|
|
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 @@
|
|
|
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":"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"}
|
|
@@ -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
|