@desplega.ai/agent-swarm 1.49.0 → 1.52.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/README.md +1 -1
- package/openapi.json +2070 -728
- package/package.json +10 -1
- package/src/agentmail/handlers.ts +65 -10
- package/src/agentmail/templates.ts +111 -0
- package/src/be/db.ts +1233 -7
- package/src/be/migrations/014_prompt_templates.sql +33 -0
- package/src/be/migrations/015_workflow_workspace.sql +3 -0
- package/src/be/migrations/016_active_session_runner_session.sql +4 -0
- package/src/be/migrations/017_channel_activity_cursors.sql +6 -0
- package/src/be/migrations/018_fix_seed_double_version.sql +30 -0
- package/src/be/migrations/019_skills.sql +65 -0
- package/src/be/migrations/020_approval_requests.sql +41 -0
- package/src/be/seed.ts +62 -0
- package/src/be/skill-parser.ts +70 -0
- package/src/be/skill-sync.ts +106 -0
- package/src/commands/runner.ts +320 -132
- package/src/commands/templates.ts +172 -0
- package/src/github/handlers.ts +292 -77
- package/src/github/mentions-aliases.test.ts +73 -0
- package/src/github/mentions.test.ts +3 -3
- package/src/github/mentions.ts +32 -6
- package/src/github/templates.ts +398 -0
- package/src/gitlab/handlers.ts +63 -22
- package/src/gitlab/templates.ts +140 -0
- package/src/heartbeat/heartbeat.ts +19 -10
- package/src/heartbeat/templates.ts +30 -0
- package/src/http/active-sessions.ts +27 -0
- package/src/http/approval-requests.ts +247 -0
- package/src/http/config.ts +3 -3
- package/src/http/index.ts +9 -2
- package/src/http/poll.ts +135 -14
- package/src/http/prompt-templates.ts +412 -0
- package/src/http/schedules.ts +35 -0
- package/src/http/skills.ts +479 -0
- package/src/http/workflows.ts +8 -0
- package/src/linear/sync.ts +28 -4
- package/src/linear/templates.ts +47 -0
- package/src/prompts/base-prompt.ts +41 -490
- package/src/prompts/registry.ts +57 -0
- package/src/prompts/resolver.ts +296 -0
- package/src/prompts/session-templates.ts +604 -0
- package/src/providers/claude-adapter.ts +15 -2
- package/src/providers/pi-mono-extension.ts +5 -1
- package/src/scheduler/scheduler.ts +125 -91
- package/src/server.ts +44 -0
- package/src/slack/assistant.ts +7 -4
- package/src/slack/channel-activity.ts +177 -0
- package/src/slack/handlers.ts +21 -6
- package/src/slack/templates.ts +55 -0
- package/src/tests/approval-requests.test.ts +735 -0
- package/src/tests/artifact-sdk.test.ts +12 -12
- package/src/tests/base-prompt.test.ts +49 -49
- package/src/tests/channel-activity.test.ts +363 -0
- package/src/tests/heartbeat.test.ts +1 -0
- package/src/tests/linear-webhook.test.ts +7 -3
- package/src/tests/pool-session-logs.test.ts +199 -0
- package/src/tests/prompt-template-github.test.ts +682 -0
- package/src/tests/prompt-template-remaining.test.ts +504 -0
- package/src/tests/prompt-template-resolver.test.ts +621 -0
- package/src/tests/prompt-template-session.test.ts +363 -0
- package/src/tests/prompt-templates-db.test.ts +616 -0
- package/src/tests/self-improvement.test.ts +8 -7
- package/src/tests/skill-parser.test.ts +178 -0
- package/src/tests/skill-sync.test.ts +171 -0
- package/src/tests/slack-metadata-inheritance.test.ts +1 -1
- package/src/tests/slack-thread-followups.test.ts +1 -1
- package/src/tests/structured-output.test.ts +0 -4
- package/src/tests/tool-annotations.test.ts +2 -1
- package/src/tests/update-profile-agentid.test.ts +248 -0
- package/src/tests/update-profile-auth.test.ts +195 -0
- package/src/tests/workflow-async-v2.test.ts +126 -4
- package/src/tests/workflow-definition-validation.test.ts +76 -0
- package/src/tests/workflow-executors.test.ts +4 -2
- package/src/tests/workflow-retry-v2.test.ts +1 -1
- package/src/tests/workflow-schedule-trigger.test.ts +104 -0
- package/src/tests/workflow-workspace.test.ts +272 -0
- package/src/tools/prompt-templates/delete.ts +86 -0
- package/src/tools/prompt-templates/get.ts +89 -0
- package/src/tools/prompt-templates/index.ts +5 -0
- package/src/tools/prompt-templates/list.ts +95 -0
- package/src/tools/prompt-templates/preview.ts +84 -0
- package/src/tools/prompt-templates/set.ts +117 -0
- package/src/tools/request-human-input.ts +106 -0
- package/src/tools/skills/index.ts +11 -0
- package/src/tools/skills/skill-create.ts +105 -0
- package/src/tools/skills/skill-delete.ts +67 -0
- package/src/tools/skills/skill-get.ts +75 -0
- package/src/tools/skills/skill-install-remote.ts +152 -0
- package/src/tools/skills/skill-install.ts +101 -0
- package/src/tools/skills/skill-list.ts +77 -0
- package/src/tools/skills/skill-publish.ts +123 -0
- package/src/tools/skills/skill-search.ts +43 -0
- package/src/tools/skills/skill-sync-remote.ts +128 -0
- package/src/tools/skills/skill-uninstall.ts +60 -0
- package/src/tools/skills/skill-update.ts +128 -0
- package/src/tools/store-progress.ts +22 -4
- package/src/tools/task-action.ts +20 -0
- package/src/tools/templates.ts +53 -0
- package/src/tools/tool-config.ts +23 -0
- package/src/tools/update-profile.ts +106 -34
- package/src/tools/workflows/create-workflow.ts +19 -1
- package/src/tools/workflows/update-workflow.ts +16 -1
- package/src/types.ts +109 -2
- package/src/workflows/definition.ts +30 -12
- package/src/workflows/engine.ts +40 -14
- package/src/workflows/executors/agent-task.ts +14 -3
- package/src/workflows/executors/human-in-the-loop.ts +160 -0
- package/src/workflows/executors/registry.ts +2 -0
- package/src/workflows/index.ts +1 -1
- package/src/workflows/recovery.ts +72 -0
- package/src/workflows/resume.ts +162 -12
- package/src/workflows/triggers.ts +31 -2
- package/src/workflows/version.ts +2 -0
- package/.claude/settings.json +0 -84
- package/.claude/settings.local.json +0 -117
- package/.dockerignore +0 -61
- package/.editorconfig +0 -15
- package/.entire/settings.json +0 -4
- package/.env.docker.example +0 -56
- package/.env.example +0 -78
- package/.github/ISSUE_TEMPLATE/bug_report.yml +0 -78
- package/.github/ISSUE_TEMPLATE/community-template.yml +0 -77
- package/.github/ISSUE_TEMPLATE/config.yml +0 -8
- package/.github/ISSUE_TEMPLATE/feature_request.yml +0 -60
- package/.github/PULL_REQUEST_TEMPLATE/community-template.md +0 -29
- package/.github/workflows/ci.yml +0 -52
- package/.github/workflows/docker-and-deploy.yml +0 -132
- package/.github/workflows/merge-gate.yml +0 -233
- package/.opencode/plugins/entire.ts +0 -133
- package/.superset/config.json +0 -6
- package/.wts-config.json +0 -4
- package/.wts-setup.ts +0 -171
- package/CHANGELOG.md +0 -447
- package/CLAUDE.md +0 -521
- package/CONTRIBUTING.md +0 -315
- package/DEPLOYMENT.md +0 -622
- package/Dockerfile +0 -65
- package/Dockerfile.worker +0 -189
- package/MCP.md +0 -841
- package/UI.md +0 -40
- package/api-entrypoint.sh +0 -56
- package/assets/agent-swarm-logo-orange.png +0 -0
- package/assets/agent-swarm-logo.png +0 -0
- package/assets/agent-swarm.mp4 +0 -0
- package/assets/agent-swarm.png +0 -0
- package/biome.json +0 -39
- package/deploy/DEPLOY.md +0 -60
- package/deploy/agent-swarm.service +0 -17
- package/deploy/docker-push.ts +0 -30
- package/deploy/install.ts +0 -85
- package/deploy/prod-db.ts +0 -42
- package/deploy/uninstall.ts +0 -12
- package/deploy/update.ts +0 -21
- package/depot.json +0 -1
- package/docker-compose.example.yml +0 -350
- package/docker-compose.local.yml +0 -119
- package/docker-entrypoint.sh +0 -632
- package/docs-site/app/api/search/route.ts +0 -4
- package/docs-site/app/docs/[[...slug]]/page.tsx +0 -87
- package/docs-site/app/docs/layout.tsx +0 -12
- package/docs-site/app/globals.css +0 -24
- package/docs-site/app/layout.config.tsx +0 -34
- package/docs-site/app/layout.tsx +0 -119
- package/docs-site/app/llms-full.txt/route.ts +0 -11
- package/docs-site/app/llms.mdx/docs/[[...slug]]/route.ts +0 -24
- package/docs-site/app/llms.txt/route.ts +0 -8
- package/docs-site/app/page.tsx +0 -5
- package/docs-site/app/robots.ts +0 -13
- package/docs-site/app/sitemap.ts +0 -37
- package/docs-site/components/api-page.client.tsx +0 -4
- package/docs-site/components/api-page.tsx +0 -7
- package/docs-site/components/mdx/mermaid.tsx +0 -55
- package/docs-site/content/docs/(documentation)/architecture/agents.mdx +0 -117
- package/docs-site/content/docs/(documentation)/architecture/hooks.mdx +0 -77
- package/docs-site/content/docs/(documentation)/architecture/memory.mdx +0 -96
- package/docs-site/content/docs/(documentation)/architecture/meta.json +0 -4
- package/docs-site/content/docs/(documentation)/architecture/overview.mdx +0 -172
- package/docs-site/content/docs/(documentation)/concepts/epics.mdx +0 -98
- package/docs-site/content/docs/(documentation)/concepts/meta.json +0 -4
- package/docs-site/content/docs/(documentation)/concepts/scheduling.mdx +0 -136
- package/docs-site/content/docs/(documentation)/concepts/services.mdx +0 -104
- package/docs-site/content/docs/(documentation)/concepts/task-lifecycle.mdx +0 -148
- package/docs-site/content/docs/(documentation)/concepts/workflows.mdx +0 -209
- package/docs-site/content/docs/(documentation)/contributing.mdx +0 -158
- package/docs-site/content/docs/(documentation)/getting-started.mdx +0 -157
- package/docs-site/content/docs/(documentation)/guides/agentmail-integration.mdx +0 -79
- package/docs-site/content/docs/(documentation)/guides/deployment.mdx +0 -171
- package/docs-site/content/docs/(documentation)/guides/github-integration.mdx +0 -81
- package/docs-site/content/docs/(documentation)/guides/gitlab-integration.mdx +0 -93
- package/docs-site/content/docs/(documentation)/guides/linear-integration.mdx +0 -98
- package/docs-site/content/docs/(documentation)/guides/meta.json +0 -13
- package/docs-site/content/docs/(documentation)/guides/sentry-integration.mdx +0 -52
- package/docs-site/content/docs/(documentation)/guides/slack-integration.mdx +0 -179
- package/docs-site/content/docs/(documentation)/guides/x402-payments.mdx +0 -154
- package/docs-site/content/docs/(documentation)/index.mdx +0 -65
- package/docs-site/content/docs/(documentation)/meta.json +0 -19
- package/docs-site/content/docs/(documentation)/reference/cli.mdx +0 -241
- package/docs-site/content/docs/(documentation)/reference/environment-variables.mdx +0 -205
- package/docs-site/content/docs/(documentation)/reference/mcp-tools.mdx +0 -449
- package/docs-site/content/docs/(documentation)/reference/meta.json +0 -4
- package/docs-site/content/docs/api-reference/active-sessions.mdx +0 -9
- package/docs-site/content/docs/api-reference/agents.mdx +0 -9
- package/docs-site/content/docs/api-reference/channels.mdx +0 -9
- package/docs-site/content/docs/api-reference/config.mdx +0 -9
- package/docs-site/content/docs/api-reference/debug.mdx +0 -9
- package/docs-site/content/docs/api-reference/ecosystem.mdx +0 -9
- package/docs-site/content/docs/api-reference/epics.mdx +0 -9
- package/docs-site/content/docs/api-reference/index.mdx +0 -32
- package/docs-site/content/docs/api-reference/memory.mdx +0 -9
- package/docs-site/content/docs/api-reference/meta.json +0 -25
- package/docs-site/content/docs/api-reference/poll.mdx +0 -9
- package/docs-site/content/docs/api-reference/repos.mdx +0 -9
- package/docs-site/content/docs/api-reference/schedules.mdx +0 -9
- package/docs-site/content/docs/api-reference/session-data.mdx +0 -9
- package/docs-site/content/docs/api-reference/stats.mdx +0 -9
- package/docs-site/content/docs/api-reference/tasks.mdx +0 -9
- package/docs-site/content/docs/api-reference/trackers.mdx +0 -9
- package/docs-site/content/docs/api-reference/webhooks.mdx +0 -9
- package/docs-site/content/docs/api-reference/workflows.mdx +0 -9
- package/docs-site/content/docs/meta.json +0 -3
- package/docs-site/lib/get-llm-text.ts +0 -10
- package/docs-site/lib/openapi.ts +0 -23
- package/docs-site/lib/source.ts +0 -8
- package/docs-site/mdx-components.tsx +0 -13
- package/docs-site/next.config.mjs +0 -29
- package/docs-site/package.json +0 -35
- package/docs-site/pnpm-lock.yaml +0 -5407
- package/docs-site/postcss.config.mjs +0 -8
- package/docs-site/public/logo.png +0 -0
- package/docs-site/scripts/generate-docs.ts +0 -171
- package/docs-site/source.config.ts +0 -17
- package/docs-site/tsconfig.json +0 -46
- package/ecosystem.config.cjs +0 -66
- package/landing/next.config.ts +0 -14
- package/landing/package.json +0 -31
- package/landing/pnpm-lock.yaml +0 -1091
- package/landing/postcss.config.mjs +0 -8
- package/landing/public/apple-touch-icon.png +0 -0
- package/landing/public/favicon.ico +0 -0
- package/landing/public/logo.png +0 -0
- package/landing/public/og-image.png +0 -0
- package/landing/public/omghost-desplega.svg +0 -30
- package/landing/public/omghost-openfort.svg +0 -9
- package/landing/src/app/actions/waitlist.ts +0 -25
- package/landing/src/app/blog/openfort-hackathon/page.tsx +0 -863
- package/landing/src/app/blog/page.tsx +0 -162
- package/landing/src/app/blog/swarm-metrics/page.tsx +0 -685
- package/landing/src/app/examples/page.tsx +0 -174
- package/landing/src/app/examples/x402/page.tsx +0 -456
- package/landing/src/app/globals.css +0 -122
- package/landing/src/app/layout.tsx +0 -134
- package/landing/src/app/page.tsx +0 -27
- package/landing/src/app/robots.ts +0 -13
- package/landing/src/app/sitemap.ts +0 -44
- package/landing/src/components/architecture.tsx +0 -163
- package/landing/src/components/cta.tsx +0 -52
- package/landing/src/components/features.tsx +0 -160
- package/landing/src/components/footer.tsx +0 -100
- package/landing/src/components/hero.tsx +0 -217
- package/landing/src/components/how-it-works.tsx +0 -165
- package/landing/src/components/navbar.tsx +0 -147
- package/landing/src/components/waitlist.tsx +0 -110
- package/landing/src/components/why-choose.tsx +0 -149
- package/landing/src/components/workshops.tsx +0 -328
- package/landing/src/lib/utils.ts +0 -6
- package/landing/tsconfig.json +0 -41
- package/misc/transcripts/2026-03-09-pi-mono-e2e-verification.md +0 -154
- package/new-ui/CLAUDE.md +0 -92
- package/new-ui/README.md +0 -73
- package/new-ui/biome.json +0 -42
- package/new-ui/components.json +0 -21
- package/new-ui/index.html +0 -25
- package/new-ui/package.json +0 -49
- package/new-ui/pnpm-lock.yaml +0 -4845
- package/new-ui/public/logo.png +0 -0
- package/new-ui/src/api/client.ts +0 -814
- package/new-ui/src/api/hooks/index.ts +0 -64
- package/new-ui/src/api/hooks/use-agents.ts +0 -58
- package/new-ui/src/api/hooks/use-channels.ts +0 -115
- package/new-ui/src/api/hooks/use-config-api.ts +0 -46
- package/new-ui/src/api/hooks/use-costs.ts +0 -122
- package/new-ui/src/api/hooks/use-db-query.ts +0 -29
- package/new-ui/src/api/hooks/use-epics.ts +0 -75
- package/new-ui/src/api/hooks/use-repos.ts +0 -61
- package/new-ui/src/api/hooks/use-schedules.ts +0 -81
- package/new-ui/src/api/hooks/use-services.ts +0 -16
- package/new-ui/src/api/hooks/use-stats.ts +0 -27
- package/new-ui/src/api/hooks/use-tasks.ts +0 -89
- package/new-ui/src/api/hooks/use-workflows.ts +0 -109
- package/new-ui/src/api/types.ts +0 -549
- package/new-ui/src/app/App.tsx +0 -13
- package/new-ui/src/app/providers.tsx +0 -32
- package/new-ui/src/app/router.tsx +0 -52
- package/new-ui/src/components/layout/app-header.tsx +0 -47
- package/new-ui/src/components/layout/app-sidebar.tsx +0 -128
- package/new-ui/src/components/layout/breadcrumbs.tsx +0 -57
- package/new-ui/src/components/layout/config-guard.tsx +0 -22
- package/new-ui/src/components/layout/root-layout.tsx +0 -40
- package/new-ui/src/components/layout/swarm-switcher.tsx +0 -85
- package/new-ui/src/components/shared/command-menu.tsx +0 -131
- package/new-ui/src/components/shared/data-grid.tsx +0 -141
- package/new-ui/src/components/shared/empty-state.tsx +0 -24
- package/new-ui/src/components/shared/error-boundary.tsx +0 -72
- package/new-ui/src/components/shared/json-viewer.tsx +0 -47
- package/new-ui/src/components/shared/name-connection-modal.tsx +0 -99
- package/new-ui/src/components/shared/page-skeleton.tsx +0 -16
- package/new-ui/src/components/shared/session-log-viewer.tsx +0 -364
- package/new-ui/src/components/shared/stats-bar.tsx +0 -132
- package/new-ui/src/components/shared/status-badge.tsx +0 -131
- package/new-ui/src/components/shared/usage-summary.tsx +0 -179
- package/new-ui/src/components/ui/alert-dialog.tsx +0 -176
- package/new-ui/src/components/ui/alert.tsx +0 -60
- package/new-ui/src/components/ui/avatar.tsx +0 -96
- package/new-ui/src/components/ui/badge.tsx +0 -46
- package/new-ui/src/components/ui/button.tsx +0 -62
- package/new-ui/src/components/ui/card.tsx +0 -75
- package/new-ui/src/components/ui/command.tsx +0 -160
- package/new-ui/src/components/ui/dialog.tsx +0 -143
- package/new-ui/src/components/ui/dropdown-menu.tsx +0 -226
- package/new-ui/src/components/ui/input.tsx +0 -21
- package/new-ui/src/components/ui/label.tsx +0 -19
- package/new-ui/src/components/ui/progress.tsx +0 -26
- package/new-ui/src/components/ui/scroll-area.tsx +0 -54
- package/new-ui/src/components/ui/select.tsx +0 -175
- package/new-ui/src/components/ui/separator.tsx +0 -28
- package/new-ui/src/components/ui/sheet.tsx +0 -132
- package/new-ui/src/components/ui/sidebar.tsx +0 -691
- package/new-ui/src/components/ui/skeleton.tsx +0 -13
- package/new-ui/src/components/ui/sonner.tsx +0 -35
- package/new-ui/src/components/ui/switch.tsx +0 -33
- package/new-ui/src/components/ui/table.tsx +0 -92
- package/new-ui/src/components/ui/tabs.tsx +0 -79
- package/new-ui/src/components/ui/textarea.tsx +0 -18
- package/new-ui/src/components/ui/tooltip.tsx +0 -51
- package/new-ui/src/components/workflows/action-node.tsx +0 -53
- package/new-ui/src/components/workflows/condition-node.tsx +0 -50
- package/new-ui/src/components/workflows/graph-utils.ts +0 -124
- package/new-ui/src/components/workflows/json-tree.tsx +0 -189
- package/new-ui/src/components/workflows/node-styles.ts +0 -10
- package/new-ui/src/components/workflows/step-detail-sheet.tsx +0 -87
- package/new-ui/src/components/workflows/trigger-node.tsx +0 -41
- package/new-ui/src/components/workflows/workflow-graph.tsx +0 -65
- package/new-ui/src/hooks/use-auto-scroll.ts +0 -82
- package/new-ui/src/hooks/use-config.ts +0 -203
- package/new-ui/src/hooks/use-keyboard-shortcuts.ts +0 -41
- package/new-ui/src/hooks/use-mobile.ts +0 -19
- package/new-ui/src/hooks/use-theme.ts +0 -60
- package/new-ui/src/lib/config.ts +0 -188
- package/new-ui/src/lib/slugs.ts +0 -71
- package/new-ui/src/lib/utils.ts +0 -120
- package/new-ui/src/main.tsx +0 -11
- package/new-ui/src/pages/agents/[id]/page.tsx +0 -492
- package/new-ui/src/pages/agents/page.tsx +0 -134
- package/new-ui/src/pages/chat/page.tsx +0 -674
- package/new-ui/src/pages/config/page.tsx +0 -1109
- package/new-ui/src/pages/dashboard/page.tsx +0 -454
- package/new-ui/src/pages/debug/page.tsx +0 -275
- package/new-ui/src/pages/epics/[id]/page.tsx +0 -809
- package/new-ui/src/pages/epics/page.tsx +0 -321
- package/new-ui/src/pages/not-found/page.tsx +0 -18
- package/new-ui/src/pages/repos/page.tsx +0 -369
- package/new-ui/src/pages/schedules/[id]/page.tsx +0 -664
- package/new-ui/src/pages/schedules/page.tsx +0 -477
- package/new-ui/src/pages/services/page.tsx +0 -128
- package/new-ui/src/pages/tasks/[id]/page.tsx +0 -670
- package/new-ui/src/pages/tasks/page.tsx +0 -592
- package/new-ui/src/pages/usage/page.tsx +0 -195
- package/new-ui/src/pages/workflow-runs/[id]/page.tsx +0 -363
- package/new-ui/src/pages/workflows/[id]/page.tsx +0 -417
- package/new-ui/src/pages/workflows/page.tsx +0 -266
- package/new-ui/src/styles/ag-grid.css +0 -36
- package/new-ui/src/styles/globals.css +0 -213
- package/new-ui/test-results/.last-run.json +0 -4
- package/new-ui/tsconfig.app.json +0 -34
- package/new-ui/tsconfig.json +0 -4
- package/new-ui/tsconfig.node.json +0 -26
- package/new-ui/vercel.json +0 -4
- package/new-ui/vite.config.ts +0 -28
- package/plugin/README.md +0 -1
- package/plugin/build-pi-skills.ts +0 -233
- package/plugin/hooks/hooks.json +0 -71
- package/prek.toml +0 -75
- package/pyproject.toml +0 -9
- package/scripts/check-db-boundary.sh +0 -60
- package/scripts/e2e-docker-provider.ts +0 -820
- package/scripts/e2e-io-schemas-test.ts +0 -807
- package/scripts/e2e-provider-test.ts +0 -220
- package/scripts/e2e-workflow-redesign.sh +0 -229
- package/scripts/e2e-workflow-test.sh +0 -285
- package/scripts/e2e-workflow-test.ts +0 -857
- package/scripts/generate-mcp-docs.ts +0 -415
- package/scripts/generate-openapi.ts +0 -26
- package/scripts/measure-tool-tokens.ts +0 -118
- package/scripts/x402-e2e-test.ts +0 -195
- package/scripts/x402-test-server.ts +0 -236
- package/scripts/x402-testnet-e2e.ts +0 -668
- package/slack-manifest.json +0 -88
- package/templates-ui/README.md +0 -46
- package/templates-ui/components.json +0 -17
- package/templates-ui/eslint.config.mjs +0 -18
- package/templates-ui/next.config.ts +0 -7
- package/templates-ui/package.json +0 -35
- package/templates-ui/pnpm-lock.yaml +0 -4571
- package/templates-ui/postcss.config.mjs +0 -7
- package/templates-ui/public/file.svg +0 -1
- package/templates-ui/public/globe.svg +0 -1
- package/templates-ui/public/logo.png +0 -0
- package/templates-ui/public/next.svg +0 -1
- package/templates-ui/public/vercel.svg +0 -1
- package/templates-ui/public/window.svg +0 -1
- package/templates-ui/src/app/[category]/[name]/page.tsx +0 -89
- package/templates-ui/src/app/api/templates/[...slug]/route.ts +0 -52
- package/templates-ui/src/app/api/templates/route.ts +0 -18
- package/templates-ui/src/app/builder/page.tsx +0 -37
- package/templates-ui/src/app/globals.css +0 -94
- package/templates-ui/src/app/layout.tsx +0 -79
- package/templates-ui/src/app/page.tsx +0 -38
- package/templates-ui/src/app/robots.ts +0 -11
- package/templates-ui/src/app/sitemap.ts +0 -31
- package/templates-ui/src/components/compose-builder.tsx +0 -442
- package/templates-ui/src/components/compose-preview.tsx +0 -117
- package/templates-ui/src/components/file-preview.tsx +0 -77
- package/templates-ui/src/components/footer.tsx +0 -40
- package/templates-ui/src/components/header.tsx +0 -41
- package/templates-ui/src/components/template-card.tsx +0 -87
- package/templates-ui/src/components/template-detail.tsx +0 -125
- package/templates-ui/src/components/template-gallery.tsx +0 -263
- package/templates-ui/src/components/ui/badge.tsx +0 -36
- package/templates-ui/src/components/ui/button.tsx +0 -57
- package/templates-ui/src/components/ui/card.tsx +0 -76
- package/templates-ui/src/components/ui/separator.tsx +0 -31
- package/templates-ui/src/components/ui/tooltip.tsx +0 -32
- package/templates-ui/src/lib/compose-generator.ts +0 -241
- package/templates-ui/src/lib/templates.ts +0 -137
- package/templates-ui/src/lib/utils.ts +0 -6
- package/templates-ui/tsconfig.json +0 -34
- package/thoughts/research/2026-02-28-openfort-viem-x402-research.md +0 -679
- package/thoughts/research/2026-02-28-x402-payments-research.md +0 -686
- package/thoughts/researcher/plans/2026-02-20-agent-self-improvement-plan.md +0 -282
- package/thoughts/researcher/research/2026-02-20-agent-self-improvement.md +0 -492
- package/thoughts/shared/plans/.gitkeep +0 -0
- package/thoughts/shared/plans/2025-12-18-slack-integration.md +0 -1195
- package/thoughts/shared/plans/2025-12-19-agent-log-streaming.md +0 -732
- package/thoughts/shared/plans/2025-12-19-role-based-swarm-plugin.md +0 -361
- package/thoughts/shared/plans/2025-12-20-mobile-responsive-ui.md +0 -501
- package/thoughts/shared/plans/2025-12-20-startup-team-swarm.md +0 -560
- package/thoughts/shared/plans/2025-12-23-runner-level-polling.md +0 -934
- package/thoughts/shared/plans/2025-12-23-runner-session-logs.md +0 -1000
- package/thoughts/shared/plans/2025-12-23-worker-lead-spawn-triggers.md +0 -568
- package/thoughts/shared/plans/2026-01-09-inverse-teleport.md +0 -1516
- package/thoughts/shared/plans/2026-01-12-agent-rename-pm2-control.md +0 -1133
- package/thoughts/shared/plans/2026-01-12-github-app-integration.md +0 -380
- package/thoughts/shared/plans/2026-01-12-lead-inbox-model.md +0 -876
- package/thoughts/shared/plans/2026-01-12-ralph-wiggum-integration.md +0 -463
- package/thoughts/shared/plans/2026-01-13-agent-concurrency.md +0 -691
- package/thoughts/shared/plans/2026-01-13-github-assignment-handling.md +0 -690
- package/thoughts/shared/plans/2026-01-13-prevent-duplicate-trigger-processing.md +0 -1071
- package/thoughts/shared/plans/2026-01-14-fix-slack-thread-context.md +0 -507
- package/thoughts/shared/plans/2026-01-15-scheduled-tasks-implementation.md +0 -565
- package/thoughts/shared/plans/2026-01-15-usage-cost-tracking-ui.md +0 -1479
- package/thoughts/shared/plans/2026-01-16-epics-feature-implementation.md +0 -1230
- package/thoughts/shared/plans/2026-02-26-mcp-tool-context-reduction.md +0 -282
- package/thoughts/shared/plans/2026-03-02-claude-context-mode-integration.md +0 -328
- package/thoughts/shared/plans/2026-03-02-code-level-heartbeat.md +0 -224
- package/thoughts/shared/research/.gitkeep +0 -0
- package/thoughts/shared/research/2025-01-09-inverse-teleport-plan-review.md +0 -420
- package/thoughts/shared/research/2025-12-18-slack-integration.md +0 -442
- package/thoughts/shared/research/2025-12-19-agent-log-streaming.md +0 -339
- package/thoughts/shared/research/2025-12-19-agent-secrets-cli-research.md +0 -390
- package/thoughts/shared/research/2025-12-21-gemini-cli-integration.md +0 -376
- package/thoughts/shared/research/2025-12-22-runner-loop-architecture.md +0 -582
- package/thoughts/shared/research/2025-12-22-setup-experience-improvements.md +0 -264
- package/thoughts/shared/research/2026-01-13-lead-duplicate-trigger-processing.md +0 -223
- package/thoughts/shared/research/2026-01-14-lead-slack-thread-context.md +0 -277
- package/thoughts/shared/research/2026-01-15-ai-tracker-agent-swarm-integration.md +0 -376
- package/thoughts/shared/research/2026-01-15-auto-starting-processes-in-worker-containers.md +0 -787
- package/thoughts/shared/research/2026-01-15-scheduled-tasks.md +0 -390
- package/thoughts/shared/research/2026-01-16-epics-feature-research.md +0 -437
- package/thoughts/shared/research/2026-02-26-cliffy-mcp-tools.md +0 -159
- package/thoughts/shared/research/2026-03-03-database-migration-system-refactor.md +0 -337
- package/thoughts/swarm-researcher/plans/2026-02-23-openclaw-improvements-plan.md +0 -778
- package/thoughts/swarm-researcher/plans/2026-02-26-artifacts-localtunnel-plan.md +0 -1269
- package/thoughts/swarm-researcher/research/2026-02-23-openclaw-vs-agent-swarm-comparison.md +0 -411
- package/thoughts/swarm-researcher/research/2026-02-26-artifacts-localtunnel.md +0 -724
- package/thoughts/taras/brainstorms/2026-03-20-prompt-template-registry.md +0 -443
- package/thoughts/taras/brainstorms/2026-03-20-setup-cli-onboarding.md +0 -307
- package/thoughts/taras/plans/2026-01-22-agent-swarm-schemas.md +0 -98
- package/thoughts/taras/plans/2026-01-28-per-worker-claude-md.md +0 -617
- package/thoughts/taras/plans/2026-01-28-sentry-cli-integration.md +0 -214
- package/thoughts/taras/plans/2026-02-20-auto-improvement.md +0 -803
- package/thoughts/taras/plans/2026-02-20-env-management.md +0 -538
- package/thoughts/taras/plans/2026-02-20-memory-system.md +0 -882
- package/thoughts/taras/plans/2026-02-20-repos-knowledge.md +0 -806
- package/thoughts/taras/plans/2026-02-20-session-attach.md +0 -647
- package/thoughts/taras/plans/2026-02-20-worker-identity.md +0 -820
- package/thoughts/taras/plans/2026-02-25-feat-new-ui-visual-redesign-plan.md +0 -768
- package/thoughts/taras/plans/2026-03-04-fix-buildSystemPrompt-missing-fields.md +0 -77
- package/thoughts/taras/plans/2026-03-04-new-ui-missing-actions.md +0 -543
- package/thoughts/taras/plans/2026-03-06-one-time-scheduled-tasks.md +0 -373
- package/thoughts/taras/plans/2026-03-08-memory-self-improvement-enhancements.md +0 -512
- package/thoughts/taras/plans/2026-03-08-pi-mono-provider-implementation.md +0 -919
- package/thoughts/taras/plans/2026-03-09-templates-registry.md +0 -723
- package/thoughts/taras/plans/2026-03-10-task-working-directory.md +0 -371
- package/thoughts/taras/plans/2026-03-11-archil-per-agent-write-strategy.md +0 -621
- package/thoughts/taras/plans/2026-03-12-eliminate-inbox-route-to-tasks.md +0 -61
- package/thoughts/taras/plans/2026-03-12-slack-thread-followup-additive.md +0 -488
- package/thoughts/taras/plans/2026-03-13-slack-ai-improvements.md +0 -644
- package/thoughts/taras/plans/2026-03-16-route-wrapper-openapi.md +0 -636
- package/thoughts/taras/plans/2026-03-17-multi-api-config.md +0 -444
- package/thoughts/taras/plans/2026-03-18-agent-fs-integration.md +0 -591
- package/thoughts/taras/plans/2026-03-18-debug-db-explorer.md +0 -446
- package/thoughts/taras/plans/2026-03-18-workflow-redesign.md +0 -987
- package/thoughts/taras/plans/2026-03-19-compound-learnings.md +0 -403
- package/thoughts/taras/plans/2026-03-19-ticket-tracker-linear-integration.md +0 -860
- package/thoughts/taras/plans/2026-03-19-workflow-io-schemas-and-bugs.md +0 -899
- package/thoughts/taras/plans/2026-03-20-setup-cli-onboarding.md +0 -874
- package/thoughts/taras/plans/2026-03-20-workflow-structured-output-validation-workspace.md +0 -723
- package/thoughts/taras/research/2026-01-22-vercel-cli-integration.md +0 -287
- package/thoughts/taras/research/2026-01-27-excessive-polling-issue.md +0 -311
- package/thoughts/taras/research/2026-01-28-per-worker-claude-md.md +0 -383
- package/thoughts/taras/research/2026-01-28-sentry-cli-integration.md +0 -240
- package/thoughts/taras/research/2026-02-19-agent-native-swarm-architecture.md +0 -390
- package/thoughts/taras/research/2026-02-19-swarm-gaps-implementation.md +0 -594
- package/thoughts/taras/research/2026-02-25-dashboard-ui-design-best-practices.md +0 -825
- package/thoughts/taras/research/2026-02-26-task-detail-page-redesign.md +0 -393
- package/thoughts/taras/research/2026-03-03-new-ui-missing-actions.md +0 -168
- package/thoughts/taras/research/2026-03-05-pi-mono-provider-research.md +0 -230
- package/thoughts/taras/research/2026-03-06-workflow-engine-design.md +0 -445
- package/thoughts/taras/research/2026-03-08-drive-loop-concept.md +0 -375
- package/thoughts/taras/research/2026-03-08-pi-mono-deep-dive.md +0 -869
- package/thoughts/taras/research/2026-03-09-templates-registry.md +0 -373
- package/thoughts/taras/research/2026-03-10-agent-working-directory.md +0 -223
- package/thoughts/taras/research/2026-03-10-configurable-event-prompts.md +0 -339
- package/thoughts/taras/research/2026-03-11-archil-production-setup.md +0 -181
- package/thoughts/taras/research/2026-03-11-archil-shared-disk-write-strategies.md +0 -437
- package/thoughts/taras/research/2026-03-13-slack-ai-features.md +0 -258
- package/thoughts/taras/research/2026-03-16-openapi-docs-generation.md +0 -335
- package/thoughts/taras/research/2026-03-16-route-wrapper-openapi.md +0 -670
- package/thoughts/taras/research/2026-03-16-slack-thread-followups-e2e.md +0 -54
- package/thoughts/taras/research/2026-03-18-agent-fs-integration.md +0 -558
- package/thoughts/taras/research/2026-03-18-linear-integration-finalization.md +0 -526
- package/thoughts/taras/research/2026-03-18-workflow-redesign.md +0 -797
- package/thoughts/taras/research/2026-03-19-workflow-node-io-schemas-and-bugs.md +0 -563
- package/thoughts/taras/research/2026-03-19-workflow-structured-output-validation-workspace.md +0 -486
- package/thoughts/taras/research/2026-03-20-prompt-template-registry.md +0 -469
- package/tsconfig.json +0 -37
|
@@ -1,132 +0,0 @@
|
|
|
1
|
-
import type { LucideIcon } from "lucide-react";
|
|
2
|
-
import { Bot, CheckCircle2, Clock, DollarSign, Heart, Loader2, XCircle, Zap } from "lucide-react";
|
|
3
|
-
import { Link } from "react-router-dom";
|
|
4
|
-
import { cn } from "@/lib/utils";
|
|
5
|
-
|
|
6
|
-
interface StatItemProps {
|
|
7
|
-
icon: LucideIcon;
|
|
8
|
-
label: string;
|
|
9
|
-
value: number | string;
|
|
10
|
-
variant?: "default" | "success" | "warning" | "danger";
|
|
11
|
-
to?: string;
|
|
12
|
-
}
|
|
13
|
-
|
|
14
|
-
const variantStyles = {
|
|
15
|
-
default: "text-foreground",
|
|
16
|
-
success: "text-emerald-500",
|
|
17
|
-
warning: "text-amber-500",
|
|
18
|
-
danger: "text-red-500",
|
|
19
|
-
} as const;
|
|
20
|
-
|
|
21
|
-
function StatItem({ icon: Icon, label, value, variant = "default", to }: StatItemProps) {
|
|
22
|
-
const content = (
|
|
23
|
-
<>
|
|
24
|
-
<Icon className={cn("h-3.5 w-3.5 shrink-0", variantStyles[variant])} />
|
|
25
|
-
<span className="hidden sm:inline text-xs text-muted-foreground">{label}</span>
|
|
26
|
-
<span
|
|
27
|
-
className={cn(
|
|
28
|
-
"text-xs sm:text-sm font-bold font-mono tabular-nums shrink-0",
|
|
29
|
-
variantStyles[variant],
|
|
30
|
-
)}
|
|
31
|
-
>
|
|
32
|
-
{value}
|
|
33
|
-
</span>
|
|
34
|
-
</>
|
|
35
|
-
);
|
|
36
|
-
|
|
37
|
-
if (to) {
|
|
38
|
-
return (
|
|
39
|
-
<Link
|
|
40
|
-
to={to}
|
|
41
|
-
className="flex items-center justify-center gap-1.5 sm:gap-2 px-2 sm:px-3 py-1.5 hover:bg-muted/50 transition-colors"
|
|
42
|
-
title={label}
|
|
43
|
-
>
|
|
44
|
-
{content}
|
|
45
|
-
</Link>
|
|
46
|
-
);
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
return (
|
|
50
|
-
<div
|
|
51
|
-
className="flex items-center justify-center gap-1.5 sm:gap-2 px-2 sm:px-3 py-1.5"
|
|
52
|
-
title={label}
|
|
53
|
-
>
|
|
54
|
-
{content}
|
|
55
|
-
</div>
|
|
56
|
-
);
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
interface StatsBarProps {
|
|
60
|
-
agents?: { total: number; idle: number; busy: number; offline: number };
|
|
61
|
-
tasks?: {
|
|
62
|
-
total: number;
|
|
63
|
-
pending: number;
|
|
64
|
-
in_progress: number;
|
|
65
|
-
completed: number;
|
|
66
|
-
failed: number;
|
|
67
|
-
};
|
|
68
|
-
epics?: { active: number };
|
|
69
|
-
healthy?: boolean;
|
|
70
|
-
costToday?: number;
|
|
71
|
-
costMtd?: number;
|
|
72
|
-
}
|
|
73
|
-
|
|
74
|
-
function formatCostCompact(usd: number): string {
|
|
75
|
-
if (usd < 0.01) return "$0";
|
|
76
|
-
return `$${usd.toFixed(2)}`;
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
export function StatsBar({ agents, tasks, healthy, costToday, costMtd }: StatsBarProps) {
|
|
80
|
-
return (
|
|
81
|
-
<div className="grid grid-cols-3 sm:flex sm:items-center gap-0 rounded-lg border border-border bg-muted/30 sm:divide-x divide-border">
|
|
82
|
-
<StatItem icon={Bot} label="Agents" value={agents?.total ?? 0} to="/agents" />
|
|
83
|
-
<StatItem
|
|
84
|
-
icon={Zap}
|
|
85
|
-
label="Busy"
|
|
86
|
-
value={agents?.busy ?? 0}
|
|
87
|
-
variant={(agents?.busy ?? 0) > 0 ? "warning" : "default"}
|
|
88
|
-
to="/agents?status=busy"
|
|
89
|
-
/>
|
|
90
|
-
<StatItem
|
|
91
|
-
icon={Clock}
|
|
92
|
-
label="Pending"
|
|
93
|
-
value={tasks?.pending ?? 0}
|
|
94
|
-
to="/tasks?status=pending"
|
|
95
|
-
/>
|
|
96
|
-
<StatItem
|
|
97
|
-
icon={Loader2}
|
|
98
|
-
label="Running"
|
|
99
|
-
value={tasks?.in_progress ?? 0}
|
|
100
|
-
variant={(tasks?.in_progress ?? 0) > 0 ? "warning" : "default"}
|
|
101
|
-
to="/tasks?status=in_progress"
|
|
102
|
-
/>
|
|
103
|
-
<StatItem
|
|
104
|
-
icon={CheckCircle2}
|
|
105
|
-
label="Done"
|
|
106
|
-
value={tasks?.completed ?? 0}
|
|
107
|
-
variant="success"
|
|
108
|
-
to="/tasks?status=completed"
|
|
109
|
-
/>
|
|
110
|
-
<StatItem
|
|
111
|
-
icon={XCircle}
|
|
112
|
-
label="Failed"
|
|
113
|
-
value={tasks?.failed ?? 0}
|
|
114
|
-
variant={(tasks?.failed ?? 0) > 0 ? "danger" : "default"}
|
|
115
|
-
to="/tasks?status=failed"
|
|
116
|
-
/>
|
|
117
|
-
<StatItem
|
|
118
|
-
icon={DollarSign}
|
|
119
|
-
label="Today"
|
|
120
|
-
value={formatCostCompact(costToday ?? 0)}
|
|
121
|
-
to="/usage"
|
|
122
|
-
/>
|
|
123
|
-
<StatItem icon={DollarSign} label="MTD" value={formatCostCompact(costMtd ?? 0)} to="/usage" />
|
|
124
|
-
<StatItem
|
|
125
|
-
icon={Heart}
|
|
126
|
-
label="Health"
|
|
127
|
-
value={healthy ? "OK" : "ERR"}
|
|
128
|
-
variant={healthy ? "success" : "danger"}
|
|
129
|
-
/>
|
|
130
|
-
</div>
|
|
131
|
-
);
|
|
132
|
-
}
|
|
@@ -1,131 +0,0 @@
|
|
|
1
|
-
import { Loader2 } from "lucide-react";
|
|
2
|
-
import type {
|
|
3
|
-
AgentStatus,
|
|
4
|
-
AgentTaskStatus,
|
|
5
|
-
EpicStatus,
|
|
6
|
-
ServiceStatus,
|
|
7
|
-
WorkflowRunStatus,
|
|
8
|
-
WorkflowRunStepStatus,
|
|
9
|
-
} from "@/api/types";
|
|
10
|
-
import { Badge } from "@/components/ui/badge";
|
|
11
|
-
import { cn } from "@/lib/utils";
|
|
12
|
-
|
|
13
|
-
type Status =
|
|
14
|
-
| AgentStatus
|
|
15
|
-
| AgentTaskStatus
|
|
16
|
-
| EpicStatus
|
|
17
|
-
| ServiceStatus
|
|
18
|
-
| WorkflowRunStatus
|
|
19
|
-
| WorkflowRunStepStatus;
|
|
20
|
-
|
|
21
|
-
interface StatusConfig {
|
|
22
|
-
label: string;
|
|
23
|
-
dot: string;
|
|
24
|
-
text: string;
|
|
25
|
-
spinner?: boolean;
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
const statusConfig: Record<string, StatusConfig> = {
|
|
29
|
-
// Agent statuses
|
|
30
|
-
idle: { label: "IDLE", dot: "bg-emerald-500", text: "text-emerald-600 dark:text-emerald-400" },
|
|
31
|
-
busy: {
|
|
32
|
-
label: "BUSY",
|
|
33
|
-
dot: "bg-amber-500",
|
|
34
|
-
text: "text-amber-600 dark:text-amber-400",
|
|
35
|
-
spinner: true,
|
|
36
|
-
},
|
|
37
|
-
offline: { label: "OFFLINE", dot: "bg-zinc-400", text: "text-zinc-500 dark:text-zinc-400" },
|
|
38
|
-
|
|
39
|
-
// Task statuses
|
|
40
|
-
backlog: { label: "BACKLOG", dot: "bg-zinc-400", text: "text-zinc-500 dark:text-zinc-400" },
|
|
41
|
-
unassigned: { label: "UNASSIGNED", dot: "bg-zinc-400", text: "text-zinc-500 dark:text-zinc-400" },
|
|
42
|
-
offered: {
|
|
43
|
-
label: "OFFERED",
|
|
44
|
-
dot: "bg-amber-500",
|
|
45
|
-
text: "text-amber-600 dark:text-amber-400",
|
|
46
|
-
spinner: true,
|
|
47
|
-
},
|
|
48
|
-
reviewing: { label: "REVIEWING", dot: "bg-blue-500", text: "text-blue-600 dark:text-blue-400" },
|
|
49
|
-
pending: { label: "PENDING", dot: "bg-yellow-500", text: "text-yellow-600 dark:text-yellow-400" },
|
|
50
|
-
in_progress: {
|
|
51
|
-
label: "IN PROGRESS",
|
|
52
|
-
dot: "bg-amber-500",
|
|
53
|
-
text: "text-amber-600 dark:text-amber-400",
|
|
54
|
-
spinner: true,
|
|
55
|
-
},
|
|
56
|
-
paused: { label: "PAUSED", dot: "bg-blue-500", text: "text-blue-600 dark:text-blue-400" },
|
|
57
|
-
completed: {
|
|
58
|
-
label: "COMPLETED",
|
|
59
|
-
dot: "bg-emerald-500",
|
|
60
|
-
text: "text-emerald-600 dark:text-emerald-400",
|
|
61
|
-
},
|
|
62
|
-
failed: { label: "FAILED", dot: "bg-red-500", text: "text-red-600 dark:text-red-400" },
|
|
63
|
-
cancelled: { label: "CANCELLED", dot: "bg-zinc-400", text: "text-zinc-500 dark:text-zinc-400" },
|
|
64
|
-
|
|
65
|
-
// Epic statuses
|
|
66
|
-
draft: { label: "DRAFT", dot: "bg-zinc-400", text: "text-zinc-500 dark:text-zinc-400" },
|
|
67
|
-
active: {
|
|
68
|
-
label: "ACTIVE",
|
|
69
|
-
dot: "bg-amber-500",
|
|
70
|
-
text: "text-amber-600 dark:text-amber-400",
|
|
71
|
-
spinner: true,
|
|
72
|
-
},
|
|
73
|
-
|
|
74
|
-
// Service statuses
|
|
75
|
-
starting: {
|
|
76
|
-
label: "STARTING",
|
|
77
|
-
dot: "bg-yellow-500",
|
|
78
|
-
text: "text-yellow-600 dark:text-yellow-400",
|
|
79
|
-
},
|
|
80
|
-
healthy: {
|
|
81
|
-
label: "HEALTHY",
|
|
82
|
-
dot: "bg-emerald-500",
|
|
83
|
-
text: "text-emerald-600 dark:text-emerald-400",
|
|
84
|
-
},
|
|
85
|
-
unhealthy: { label: "UNHEALTHY", dot: "bg-red-500", text: "text-red-600 dark:text-red-400" },
|
|
86
|
-
stopped: { label: "STOPPED", dot: "bg-zinc-400", text: "text-zinc-500 dark:text-zinc-400" },
|
|
87
|
-
|
|
88
|
-
// Workflow run statuses
|
|
89
|
-
running: {
|
|
90
|
-
label: "RUNNING",
|
|
91
|
-
dot: "bg-amber-500",
|
|
92
|
-
text: "text-amber-600 dark:text-amber-400",
|
|
93
|
-
spinner: true,
|
|
94
|
-
},
|
|
95
|
-
waiting: { label: "WAITING", dot: "bg-yellow-500", text: "text-yellow-600 dark:text-yellow-400" },
|
|
96
|
-
|
|
97
|
-
// Workflow step statuses
|
|
98
|
-
skipped: { label: "SKIPPED", dot: "bg-zinc-400", text: "text-zinc-500 dark:text-zinc-400" },
|
|
99
|
-
} satisfies Record<string, StatusConfig>;
|
|
100
|
-
|
|
101
|
-
interface StatusBadgeProps {
|
|
102
|
-
status: Status;
|
|
103
|
-
size?: "sm" | "md";
|
|
104
|
-
className?: string;
|
|
105
|
-
}
|
|
106
|
-
|
|
107
|
-
export function StatusBadge({ status, size = "sm", className }: StatusBadgeProps) {
|
|
108
|
-
const config = statusConfig[status] ?? {
|
|
109
|
-
label: status,
|
|
110
|
-
dot: "bg-zinc-400",
|
|
111
|
-
text: "text-zinc-500 dark:text-zinc-400",
|
|
112
|
-
};
|
|
113
|
-
|
|
114
|
-
return (
|
|
115
|
-
<Badge
|
|
116
|
-
variant="outline"
|
|
117
|
-
className={cn(
|
|
118
|
-
"gap-1.5 font-medium leading-none items-center",
|
|
119
|
-
size === "sm" ? "text-[9px] px-1.5 py-0 h-5" : "text-[10px] px-2 py-0 h-6",
|
|
120
|
-
className,
|
|
121
|
-
)}
|
|
122
|
-
>
|
|
123
|
-
{config.spinner ? (
|
|
124
|
-
<Loader2 className={cn("h-3 w-3 shrink-0 animate-spin", config.text)} />
|
|
125
|
-
) : (
|
|
126
|
-
<span className={cn("h-1.5 w-1.5 rounded-full shrink-0", config.dot)} />
|
|
127
|
-
)}
|
|
128
|
-
<span className={config.text}>{config.label}</span>
|
|
129
|
-
</Badge>
|
|
130
|
-
);
|
|
131
|
-
}
|
|
@@ -1,179 +0,0 @@
|
|
|
1
|
-
import { Activity, Clock, Coins, DollarSign, TrendingUp } from "lucide-react";
|
|
2
|
-
import { useMemo } from "react";
|
|
3
|
-
import {
|
|
4
|
-
CartesianGrid,
|
|
5
|
-
Line,
|
|
6
|
-
LineChart,
|
|
7
|
-
ResponsiveContainer,
|
|
8
|
-
Tooltip,
|
|
9
|
-
XAxis,
|
|
10
|
-
YAxis,
|
|
11
|
-
} from "recharts";
|
|
12
|
-
import type { SessionCost, UsageSummaryDailyRow, UsageSummaryTotals } from "@/api/types";
|
|
13
|
-
import { formatCompactNumber, formatCurrency, formatDuration } from "@/lib/utils";
|
|
14
|
-
|
|
15
|
-
function StatCard({
|
|
16
|
-
label,
|
|
17
|
-
value,
|
|
18
|
-
icon: Icon,
|
|
19
|
-
}: {
|
|
20
|
-
label: string;
|
|
21
|
-
value: string;
|
|
22
|
-
icon: React.ElementType;
|
|
23
|
-
}) {
|
|
24
|
-
return (
|
|
25
|
-
<div className="flex items-center gap-3 rounded-lg border border-border px-4 py-3">
|
|
26
|
-
<div className="flex h-8 w-8 items-center justify-center rounded-md bg-muted">
|
|
27
|
-
<Icon className="h-4 w-4 text-muted-foreground" />
|
|
28
|
-
</div>
|
|
29
|
-
<div>
|
|
30
|
-
<p className="text-[11px] text-muted-foreground uppercase tracking-wider">{label}</p>
|
|
31
|
-
<p className="text-lg font-bold font-mono">{value}</p>
|
|
32
|
-
</div>
|
|
33
|
-
</div>
|
|
34
|
-
);
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
const tooltipStyle = {
|
|
38
|
-
background: "var(--color-card)",
|
|
39
|
-
border: "1px solid var(--color-border)",
|
|
40
|
-
borderRadius: 8,
|
|
41
|
-
fontSize: 12,
|
|
42
|
-
color: "var(--color-foreground)",
|
|
43
|
-
};
|
|
44
|
-
|
|
45
|
-
// New interface: accepts pre-aggregated data from server
|
|
46
|
-
interface UsageSummaryAggregatedProps {
|
|
47
|
-
totals: UsageSummaryTotals;
|
|
48
|
-
dailyData: UsageSummaryDailyRow[];
|
|
49
|
-
daysBack?: number;
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
// Legacy interface: accepts raw costs (used by agent detail page)
|
|
53
|
-
interface UsageSummaryRawProps {
|
|
54
|
-
costs: SessionCost[];
|
|
55
|
-
daysBack?: number;
|
|
56
|
-
}
|
|
57
|
-
|
|
58
|
-
type UsageSummaryProps = UsageSummaryAggregatedProps | UsageSummaryRawProps;
|
|
59
|
-
|
|
60
|
-
function isAggregatedProps(props: UsageSummaryProps): props is UsageSummaryAggregatedProps {
|
|
61
|
-
return "totals" in props;
|
|
62
|
-
}
|
|
63
|
-
|
|
64
|
-
export function UsageSummary(props: UsageSummaryProps) {
|
|
65
|
-
const daysBack = props.daysBack ?? 30;
|
|
66
|
-
const totals = isAggregatedProps(props) ? props.totals : undefined;
|
|
67
|
-
const costs = isAggregatedProps(props) ? undefined : props.costs;
|
|
68
|
-
|
|
69
|
-
// Compute stats from either pre-aggregated or raw data
|
|
70
|
-
const stats = useMemo(() => {
|
|
71
|
-
if (totals) {
|
|
72
|
-
return {
|
|
73
|
-
totalCost: totals.totalCostUsd,
|
|
74
|
-
totalTokens: totals.totalInputTokens + totals.totalOutputTokens,
|
|
75
|
-
sessions: totals.totalSessions,
|
|
76
|
-
totalDuration: totals.totalDurationMs,
|
|
77
|
-
avgCost: totals.avgCostPerSession,
|
|
78
|
-
};
|
|
79
|
-
}
|
|
80
|
-
// Legacy: compute from raw costs
|
|
81
|
-
const c = costs ?? [];
|
|
82
|
-
const totalCost = c.reduce((s, x) => s + x.totalCostUsd, 0);
|
|
83
|
-
const totalTokens = c.reduce((s, x) => s + x.inputTokens + x.outputTokens, 0);
|
|
84
|
-
const totalDuration = c.reduce((s, x) => s + x.durationMs, 0);
|
|
85
|
-
return {
|
|
86
|
-
totalCost,
|
|
87
|
-
totalTokens,
|
|
88
|
-
sessions: c.length,
|
|
89
|
-
totalDuration,
|
|
90
|
-
avgCost: c.length > 0 ? totalCost / c.length : 0,
|
|
91
|
-
};
|
|
92
|
-
}, [totals, costs]);
|
|
93
|
-
|
|
94
|
-
const aggregatedDailyData = isAggregatedProps(props) ? props.dailyData : undefined;
|
|
95
|
-
|
|
96
|
-
// Compute daily chart data
|
|
97
|
-
const dailyData = useMemo(() => {
|
|
98
|
-
if (aggregatedDailyData) {
|
|
99
|
-
// Use pre-aggregated daily data, just format dates for display
|
|
100
|
-
return aggregatedDailyData.map((d) => ({
|
|
101
|
-
date: d.date.slice(5),
|
|
102
|
-
cost: Math.round(d.costUsd * 1000) / 1000,
|
|
103
|
-
}));
|
|
104
|
-
}
|
|
105
|
-
// Legacy: compute from raw costs
|
|
106
|
-
const c = costs ?? [];
|
|
107
|
-
const now = new Date();
|
|
108
|
-
const start = new Date(now.getFullYear(), now.getMonth(), now.getDate() - daysBack);
|
|
109
|
-
|
|
110
|
-
const dayMap = new Map<string, number>();
|
|
111
|
-
for (let d = new Date(start); d <= now; d.setDate(d.getDate() + 1)) {
|
|
112
|
-
dayMap.set(d.toISOString().slice(0, 10), 0);
|
|
113
|
-
}
|
|
114
|
-
|
|
115
|
-
for (const x of c) {
|
|
116
|
-
const day = x.createdAt.slice(0, 10);
|
|
117
|
-
if (dayMap.has(day)) {
|
|
118
|
-
dayMap.set(day, (dayMap.get(day) ?? 0) + x.totalCostUsd);
|
|
119
|
-
}
|
|
120
|
-
}
|
|
121
|
-
|
|
122
|
-
return Array.from(dayMap.entries()).map(([date, cost]) => ({
|
|
123
|
-
date: date.slice(5),
|
|
124
|
-
cost: Math.round(cost * 1000) / 1000,
|
|
125
|
-
}));
|
|
126
|
-
}, [aggregatedDailyData, costs, daysBack]);
|
|
127
|
-
|
|
128
|
-
const isEmpty = totals ? totals.totalSessions === 0 : (costs ?? []).length === 0;
|
|
129
|
-
|
|
130
|
-
if (isEmpty) {
|
|
131
|
-
return (
|
|
132
|
-
<p className="text-sm text-muted-foreground py-8 text-center">No usage data available</p>
|
|
133
|
-
);
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
return (
|
|
137
|
-
<div className="space-y-4">
|
|
138
|
-
{/* Stats Strip */}
|
|
139
|
-
<div className="grid gap-3 sm:grid-cols-2 lg:grid-cols-5">
|
|
140
|
-
<StatCard label="Total Cost" value={formatCurrency(stats.totalCost)} icon={DollarSign} />
|
|
141
|
-
<StatCard label="Tokens" value={formatCompactNumber(stats.totalTokens)} icon={Coins} />
|
|
142
|
-
<StatCard label="Sessions" value={String(stats.sessions)} icon={Activity} />
|
|
143
|
-
<StatCard label="Total Time" value={formatDuration(stats.totalDuration)} icon={Clock} />
|
|
144
|
-
<StatCard label="Avg/Session" value={formatCurrency(stats.avgCost)} icon={TrendingUp} />
|
|
145
|
-
</div>
|
|
146
|
-
|
|
147
|
-
{/* Daily Cost Chart */}
|
|
148
|
-
<div className="rounded-lg border border-border p-4">
|
|
149
|
-
<p className="text-xs text-muted-foreground uppercase tracking-wider mb-3">Daily Cost</p>
|
|
150
|
-
<ResponsiveContainer width="100%" height={200}>
|
|
151
|
-
<LineChart data={dailyData}>
|
|
152
|
-
<CartesianGrid strokeDasharray="3 3" stroke="var(--color-border)" />
|
|
153
|
-
<XAxis
|
|
154
|
-
dataKey="date"
|
|
155
|
-
tick={{ fontSize: 10, fill: "var(--color-muted-foreground)" }}
|
|
156
|
-
interval={Math.max(0, Math.floor(dailyData.length / 10))}
|
|
157
|
-
/>
|
|
158
|
-
<YAxis
|
|
159
|
-
tick={{ fontSize: 10, fill: "var(--color-muted-foreground)" }}
|
|
160
|
-
tickFormatter={(v) => `$${v}`}
|
|
161
|
-
width={50}
|
|
162
|
-
/>
|
|
163
|
-
<Tooltip
|
|
164
|
-
contentStyle={tooltipStyle}
|
|
165
|
-
formatter={(value) => [`$${Number(value).toFixed(3)}`, "Cost"]}
|
|
166
|
-
/>
|
|
167
|
-
<Line
|
|
168
|
-
type="monotone"
|
|
169
|
-
dataKey="cost"
|
|
170
|
-
stroke="var(--color-primary)"
|
|
171
|
-
strokeWidth={2}
|
|
172
|
-
dot={false}
|
|
173
|
-
/>
|
|
174
|
-
</LineChart>
|
|
175
|
-
</ResponsiveContainer>
|
|
176
|
-
</div>
|
|
177
|
-
</div>
|
|
178
|
-
);
|
|
179
|
-
}
|
|
@@ -1,176 +0,0 @@
|
|
|
1
|
-
import { AlertDialog as AlertDialogPrimitive } from "radix-ui";
|
|
2
|
-
import type * as React from "react";
|
|
3
|
-
import { Button } from "@/components/ui/button";
|
|
4
|
-
import { cn } from "@/lib/utils";
|
|
5
|
-
|
|
6
|
-
function AlertDialog({ ...props }: React.ComponentProps<typeof AlertDialogPrimitive.Root>) {
|
|
7
|
-
return <AlertDialogPrimitive.Root data-slot="alert-dialog" {...props} />;
|
|
8
|
-
}
|
|
9
|
-
|
|
10
|
-
function AlertDialogTrigger({
|
|
11
|
-
...props
|
|
12
|
-
}: React.ComponentProps<typeof AlertDialogPrimitive.Trigger>) {
|
|
13
|
-
return <AlertDialogPrimitive.Trigger data-slot="alert-dialog-trigger" {...props} />;
|
|
14
|
-
}
|
|
15
|
-
|
|
16
|
-
function AlertDialogPortal({ ...props }: React.ComponentProps<typeof AlertDialogPrimitive.Portal>) {
|
|
17
|
-
return <AlertDialogPrimitive.Portal data-slot="alert-dialog-portal" {...props} />;
|
|
18
|
-
}
|
|
19
|
-
|
|
20
|
-
function AlertDialogOverlay({
|
|
21
|
-
className,
|
|
22
|
-
...props
|
|
23
|
-
}: React.ComponentProps<typeof AlertDialogPrimitive.Overlay>) {
|
|
24
|
-
return (
|
|
25
|
-
<AlertDialogPrimitive.Overlay
|
|
26
|
-
data-slot="alert-dialog-overlay"
|
|
27
|
-
className={cn(
|
|
28
|
-
"data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 fixed inset-0 z-50 bg-black/50",
|
|
29
|
-
className,
|
|
30
|
-
)}
|
|
31
|
-
{...props}
|
|
32
|
-
/>
|
|
33
|
-
);
|
|
34
|
-
}
|
|
35
|
-
|
|
36
|
-
function AlertDialogContent({
|
|
37
|
-
className,
|
|
38
|
-
size = "default",
|
|
39
|
-
...props
|
|
40
|
-
}: React.ComponentProps<typeof AlertDialogPrimitive.Content> & {
|
|
41
|
-
size?: "default" | "sm";
|
|
42
|
-
}) {
|
|
43
|
-
return (
|
|
44
|
-
<AlertDialogPortal>
|
|
45
|
-
<AlertDialogOverlay />
|
|
46
|
-
<AlertDialogPrimitive.Content
|
|
47
|
-
data-slot="alert-dialog-content"
|
|
48
|
-
data-size={size}
|
|
49
|
-
className={cn(
|
|
50
|
-
"bg-background data-[state=open]:animate-in data-[state=closed]:animate-out data-[state=closed]:fade-out-0 data-[state=open]:fade-in-0 data-[state=closed]:zoom-out-95 data-[state=open]:zoom-in-95 group/alert-dialog-content fixed top-[50%] left-[50%] z-50 grid w-full max-w-[calc(100%-2rem)] translate-x-[-50%] translate-y-[-50%] gap-4 rounded-lg border p-6 shadow-lg duration-200 data-[size=sm]:max-w-xs data-[size=default]:sm:max-w-lg",
|
|
51
|
-
className,
|
|
52
|
-
)}
|
|
53
|
-
{...props}
|
|
54
|
-
/>
|
|
55
|
-
</AlertDialogPortal>
|
|
56
|
-
);
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
function AlertDialogHeader({ className, ...props }: React.ComponentProps<"div">) {
|
|
60
|
-
return (
|
|
61
|
-
<div
|
|
62
|
-
data-slot="alert-dialog-header"
|
|
63
|
-
className={cn(
|
|
64
|
-
"grid grid-rows-[auto_1fr] place-items-center gap-1.5 text-center has-data-[slot=alert-dialog-media]:grid-rows-[auto_auto_1fr] has-data-[slot=alert-dialog-media]:gap-x-6 sm:group-data-[size=default]/alert-dialog-content:place-items-start sm:group-data-[size=default]/alert-dialog-content:text-left sm:group-data-[size=default]/alert-dialog-content:has-data-[slot=alert-dialog-media]:grid-rows-[auto_1fr]",
|
|
65
|
-
className,
|
|
66
|
-
)}
|
|
67
|
-
{...props}
|
|
68
|
-
/>
|
|
69
|
-
);
|
|
70
|
-
}
|
|
71
|
-
|
|
72
|
-
function AlertDialogFooter({ className, ...props }: React.ComponentProps<"div">) {
|
|
73
|
-
return (
|
|
74
|
-
<div
|
|
75
|
-
data-slot="alert-dialog-footer"
|
|
76
|
-
className={cn(
|
|
77
|
-
"flex flex-col-reverse gap-2 group-data-[size=sm]/alert-dialog-content:grid group-data-[size=sm]/alert-dialog-content:grid-cols-2 sm:flex-row sm:justify-end",
|
|
78
|
-
className,
|
|
79
|
-
)}
|
|
80
|
-
{...props}
|
|
81
|
-
/>
|
|
82
|
-
);
|
|
83
|
-
}
|
|
84
|
-
|
|
85
|
-
function AlertDialogTitle({
|
|
86
|
-
className,
|
|
87
|
-
...props
|
|
88
|
-
}: React.ComponentProps<typeof AlertDialogPrimitive.Title>) {
|
|
89
|
-
return (
|
|
90
|
-
<AlertDialogPrimitive.Title
|
|
91
|
-
data-slot="alert-dialog-title"
|
|
92
|
-
className={cn(
|
|
93
|
-
"text-lg font-semibold sm:group-data-[size=default]/alert-dialog-content:group-has-data-[slot=alert-dialog-media]/alert-dialog-content:col-start-2",
|
|
94
|
-
className,
|
|
95
|
-
)}
|
|
96
|
-
{...props}
|
|
97
|
-
/>
|
|
98
|
-
);
|
|
99
|
-
}
|
|
100
|
-
|
|
101
|
-
function AlertDialogDescription({
|
|
102
|
-
className,
|
|
103
|
-
...props
|
|
104
|
-
}: React.ComponentProps<typeof AlertDialogPrimitive.Description>) {
|
|
105
|
-
return (
|
|
106
|
-
<AlertDialogPrimitive.Description
|
|
107
|
-
data-slot="alert-dialog-description"
|
|
108
|
-
className={cn("text-muted-foreground text-sm", className)}
|
|
109
|
-
{...props}
|
|
110
|
-
/>
|
|
111
|
-
);
|
|
112
|
-
}
|
|
113
|
-
|
|
114
|
-
function AlertDialogMedia({ className, ...props }: React.ComponentProps<"div">) {
|
|
115
|
-
return (
|
|
116
|
-
<div
|
|
117
|
-
data-slot="alert-dialog-media"
|
|
118
|
-
className={cn(
|
|
119
|
-
"bg-muted mb-2 inline-flex size-16 items-center justify-center rounded-md sm:group-data-[size=default]/alert-dialog-content:row-span-2 *:[svg:not([class*='size-'])]:size-8",
|
|
120
|
-
className,
|
|
121
|
-
)}
|
|
122
|
-
{...props}
|
|
123
|
-
/>
|
|
124
|
-
);
|
|
125
|
-
}
|
|
126
|
-
|
|
127
|
-
function AlertDialogAction({
|
|
128
|
-
className,
|
|
129
|
-
variant = "default",
|
|
130
|
-
size = "default",
|
|
131
|
-
...props
|
|
132
|
-
}: React.ComponentProps<typeof AlertDialogPrimitive.Action> &
|
|
133
|
-
Pick<React.ComponentProps<typeof Button>, "variant" | "size">) {
|
|
134
|
-
return (
|
|
135
|
-
<Button variant={variant} size={size} asChild>
|
|
136
|
-
<AlertDialogPrimitive.Action
|
|
137
|
-
data-slot="alert-dialog-action"
|
|
138
|
-
className={cn(className)}
|
|
139
|
-
{...props}
|
|
140
|
-
/>
|
|
141
|
-
</Button>
|
|
142
|
-
);
|
|
143
|
-
}
|
|
144
|
-
|
|
145
|
-
function AlertDialogCancel({
|
|
146
|
-
className,
|
|
147
|
-
variant = "outline",
|
|
148
|
-
size = "default",
|
|
149
|
-
...props
|
|
150
|
-
}: React.ComponentProps<typeof AlertDialogPrimitive.Cancel> &
|
|
151
|
-
Pick<React.ComponentProps<typeof Button>, "variant" | "size">) {
|
|
152
|
-
return (
|
|
153
|
-
<Button variant={variant} size={size} asChild>
|
|
154
|
-
<AlertDialogPrimitive.Cancel
|
|
155
|
-
data-slot="alert-dialog-cancel"
|
|
156
|
-
className={cn(className)}
|
|
157
|
-
{...props}
|
|
158
|
-
/>
|
|
159
|
-
</Button>
|
|
160
|
-
);
|
|
161
|
-
}
|
|
162
|
-
|
|
163
|
-
export {
|
|
164
|
-
AlertDialog,
|
|
165
|
-
AlertDialogAction,
|
|
166
|
-
AlertDialogCancel,
|
|
167
|
-
AlertDialogContent,
|
|
168
|
-
AlertDialogDescription,
|
|
169
|
-
AlertDialogFooter,
|
|
170
|
-
AlertDialogHeader,
|
|
171
|
-
AlertDialogMedia,
|
|
172
|
-
AlertDialogOverlay,
|
|
173
|
-
AlertDialogPortal,
|
|
174
|
-
AlertDialogTitle,
|
|
175
|
-
AlertDialogTrigger,
|
|
176
|
-
};
|
|
@@ -1,60 +0,0 @@
|
|
|
1
|
-
import { cva, type VariantProps } from "class-variance-authority";
|
|
2
|
-
import type * as React from "react";
|
|
3
|
-
|
|
4
|
-
import { cn } from "@/lib/utils";
|
|
5
|
-
|
|
6
|
-
const alertVariants = cva(
|
|
7
|
-
"relative w-full rounded-lg border px-4 py-3 text-sm grid has-[>svg]:grid-cols-[calc(var(--spacing)*4)_1fr] grid-cols-[0_1fr] has-[>svg]:gap-x-3 gap-y-0.5 items-start [&>svg]:size-4 [&>svg]:translate-y-0.5 [&>svg]:text-current",
|
|
8
|
-
{
|
|
9
|
-
variants: {
|
|
10
|
-
variant: {
|
|
11
|
-
default: "bg-card text-card-foreground",
|
|
12
|
-
destructive:
|
|
13
|
-
"text-destructive bg-card [&>svg]:text-current *:data-[slot=alert-description]:text-destructive/90",
|
|
14
|
-
},
|
|
15
|
-
},
|
|
16
|
-
defaultVariants: {
|
|
17
|
-
variant: "default",
|
|
18
|
-
},
|
|
19
|
-
},
|
|
20
|
-
);
|
|
21
|
-
|
|
22
|
-
function Alert({
|
|
23
|
-
className,
|
|
24
|
-
variant,
|
|
25
|
-
...props
|
|
26
|
-
}: React.ComponentProps<"div"> & VariantProps<typeof alertVariants>) {
|
|
27
|
-
return (
|
|
28
|
-
<div
|
|
29
|
-
data-slot="alert"
|
|
30
|
-
role="alert"
|
|
31
|
-
className={cn(alertVariants({ variant }), className)}
|
|
32
|
-
{...props}
|
|
33
|
-
/>
|
|
34
|
-
);
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
function AlertTitle({ className, ...props }: React.ComponentProps<"div">) {
|
|
38
|
-
return (
|
|
39
|
-
<div
|
|
40
|
-
data-slot="alert-title"
|
|
41
|
-
className={cn("col-start-2 line-clamp-1 min-h-4 font-medium tracking-tight", className)}
|
|
42
|
-
{...props}
|
|
43
|
-
/>
|
|
44
|
-
);
|
|
45
|
-
}
|
|
46
|
-
|
|
47
|
-
function AlertDescription({ className, ...props }: React.ComponentProps<"div">) {
|
|
48
|
-
return (
|
|
49
|
-
<div
|
|
50
|
-
data-slot="alert-description"
|
|
51
|
-
className={cn(
|
|
52
|
-
"text-muted-foreground col-start-2 grid justify-items-start gap-1 text-sm [&_p]:leading-relaxed",
|
|
53
|
-
className,
|
|
54
|
-
)}
|
|
55
|
-
{...props}
|
|
56
|
-
/>
|
|
57
|
-
);
|
|
58
|
-
}
|
|
59
|
-
|
|
60
|
-
export { Alert, AlertTitle, AlertDescription };
|