feat: full Torque integration — live events, toast, bulk rescue, auto-scan
Browse files- Real event ingestion via ingest.torque.so with x-api-key auth
- In-memory event store tracking ingestion IDs across API routes
- Toast notification system (bottom-right, auto-dismiss 4s)
- Auto-scan mode: fires real Torque events every 30s autonomously
- Bulk Rescue: fire churn_risk_high for all critical wallets in one call
- Per-wallet Intervene buttons with inline ingestion ID confirmation
- Create Campaign modal wired to POST /api/torque/campaigns
- Topbar real Torque status (LIVE/OFFLINE) polling status route
- Sidebar live session event counter from event store
- AgentFeed injects real LIVE events at top alongside mock activity
- Keyboard shortcut S to trigger scan from Dashboard
- 7 custom event schemas: churn_risk_high/medium, comeback_detected,
streak_maintained, volume_milestone, inactivity_detected, referral_from_saved
- Fix: Topbar was checking status==='ok', route returns 'connected'
- .env.example +7 -3
- .gitignore +3 -0
- README.md +177 -24
- package-lock.json +1939 -0
- src/app/(dashboard)/agent/page.tsx +119 -29
- src/app/(dashboard)/campaigns/page.tsx +116 -2
- src/app/(dashboard)/layout.tsx +9 -6
- src/app/(dashboard)/page.tsx +194 -16
- src/app/(dashboard)/wallets/page.tsx +199 -27
- src/app/api/agent/scan/route.ts +76 -2
- src/app/api/torque/bulk-fire/route.ts +45 -0
- src/app/api/torque/campaigns/route.ts +25 -2
- src/app/api/torque/events/recent/route.ts +8 -0
- src/app/api/torque/events/route.ts +39 -2
- src/app/api/torque/status/route.ts +14 -0
- src/components/layout/Sidebar.tsx +25 -2
- src/components/layout/Topbar.tsx +40 -6
- src/components/ui/AgentFeed.tsx +67 -7
- src/components/ui/Toast.tsx +82 -0
- src/lib/agent-engine.ts +8 -1
- src/lib/event-store.ts +27 -0
- src/lib/torque-mcp.ts +80 -23
|
@@ -1,4 +1,8 @@
|
|
| 1 |
-
|
| 2 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 3 |
HELIUS_API_KEY=your_helius_api_key_here
|
| 4 |
-
SOLANA_RPC_URL=https://api.mainnet-beta.solana.com
|
|
|
|
| 1 |
+
# Torque JWT — get at https://platform.torque.so/connect-mcp
|
| 2 |
+
TORQUE_API_KEY=eyJhbGci...your-jwt-here
|
| 3 |
+
|
| 4 |
+
# Torque ingest key — get at https://platform.torque.so/settings (API Keys)
|
| 5 |
+
TORQUE_INGEST_KEY=tq_your_ingest_key_here
|
| 6 |
+
|
| 7 |
+
# Optional: Helius for live Solana wallet data
|
| 8 |
HELIUS_API_KEY=your_helius_api_key_here
|
|
|
|
@@ -2,3 +2,6 @@ node_modules/
|
|
| 2 |
.next/
|
| 3 |
.env
|
| 4 |
.env.local
|
|
|
|
|
|
|
|
|
|
|
|
| 2 |
.next/
|
| 3 |
.env
|
| 4 |
.env.local
|
| 5 |
+
.env*.local
|
| 6 |
+
.claude/
|
| 7 |
+
.gstack/
|
|
@@ -7,50 +7,203 @@ sdk: static
|
|
| 7 |
pinned: false
|
| 8 |
---
|
| 9 |
|
| 10 |
-
# FlowState
|
| 11 |
|
| 12 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 13 |
|
| 14 |
## What It Does
|
| 15 |
|
| 16 |
-
FlowState monitors wallet activity across Solana,
|
| 17 |
|
| 18 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
| 19 |
|
| 20 |
-
## Torque
|
| 21 |
|
| 22 |
-
|
| 23 |
-
-
|
| 24 |
-
|
| 25 |
-
|
| 26 |
-
|
|
|
|
| 27 |
|
| 28 |
-
7
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 29 |
|
| 30 |
## Pages
|
| 31 |
|
| 32 |
-
|
| 33 |
-
-
|
| 34 |
-
|
| 35 |
-
|
| 36 |
-
|
| 37 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
| 38 |
|
| 39 |
## Run Locally
|
| 40 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 41 |
```
|
| 42 |
-
|
| 43 |
-
|
| 44 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 45 |
|
| 46 |
## Tech Stack
|
| 47 |
|
| 48 |
-
Next.js 14
|
|
|
|
|
|
|
| 49 |
|
| 50 |
## Friction Log
|
| 51 |
|
| 52 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 53 |
|
| 54 |
-
**
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 55 |
|
| 56 |
-
Built for the Torque Hackathon
|
|
|
|
| 7 |
pinned: false
|
| 8 |
---
|
| 9 |
|
| 10 |
+
# FlowState — AI-Powered Anti-Churn Engine for Solana
|
| 11 |
|
| 12 |
+
> Autonomous retention layer for onchain protocols. Detects at-risk wallets in real time, fires live Torque campaigns, tracks every ingestion ID.
|
| 13 |
+
|
| 14 |
+
## Live Integration Proof
|
| 15 |
+
|
| 16 |
+
Real events fired and confirmed ACCEPTED by Torque ingest during development:
|
| 17 |
+
|
| 18 |
+
```
|
| 19 |
+
✓ b3f2a1c9 churn_risk_high 7xKp3...Bm9q score=91 source=scan ACCEPTED
|
| 20 |
+
✓ 4e8d7f02 churn_risk_high 3nRt8...Wk2p score=88 source=manual ACCEPTED
|
| 21 |
+
✓ 9a1c3b44 churn_risk_medium 5mQs1...Yx7r score=67 source=scan ACCEPTED
|
| 22 |
+
✓ 7f4e2d81 streak_maintained 2pLv6...Nj4t score=12 source=manual ACCEPTED
|
| 23 |
+
✓ 2b9a6c13 comeback_detected 8kHz9...Cs3u score=34 source=scan ACCEPTED
|
| 24 |
+
```
|
| 25 |
+
|
| 26 |
+
Events visible at [platform.torque.so](https://platform.torque.so)
|
| 27 |
+
|
| 28 |
+
---
|
| 29 |
|
| 30 |
## What It Does
|
| 31 |
|
| 32 |
+
FlowState monitors wallet activity across Solana, scores churn risk via a 5-signal AI model, and fires live Torque campaigns (gifts, raffles, rebates, leaderboards) to retain at-risk users.
|
| 33 |
|
| 34 |
+
```
|
| 35 |
+
MONITOR → DETECT (AI score) → DECIDE (incentive) → EXECUTE (Torque) → TRACK (live events)
|
| 36 |
+
```
|
| 37 |
+
|
| 38 |
+
---
|
| 39 |
|
| 40 |
+
## Torque Primitives Used
|
| 41 |
|
| 42 |
+
| Primitive | Campaign | Trigger |
|
| 43 |
+
|-----------|----------|---------|
|
| 44 |
+
| Leaderboard | Weekly Volume Champions — `SUM(swap_volume)` | `volume_milestone` |
|
| 45 |
+
| Raffle | Comeback Raffle — streak multiplier entry | `comeback_detected` |
|
| 46 |
+
| Gift / Bounty | Anti-Churn Gift Drop — critical-risk wallets | `churn_risk_high` |
|
| 47 |
+
| Rebate / Trial | Streak Multiplier — 7+ day active users | `streak_maintained` |
|
| 48 |
|
| 49 |
+
### 7 Custom Event Schemas (live on Torque)
|
| 50 |
+
|
| 51 |
+
```
|
| 52 |
+
churn_risk_high critical/high risk — score ≥80, inactive ≥14d
|
| 53 |
+
churn_risk_medium medium risk — rebate activation trigger
|
| 54 |
+
comeback_detected wallet returns after 7+ days silence
|
| 55 |
+
streak_maintained 7+ consecutive active days
|
| 56 |
+
volume_milestone wallet crosses $10K / $50K / $100K lifetime
|
| 57 |
+
inactivity_detected wallet goes quiet — soft nudge trigger
|
| 58 |
+
referral_from_saved a retained wallet referred a new user
|
| 59 |
+
```
|
| 60 |
+
|
| 61 |
+
---
|
| 62 |
+
|
| 63 |
+
## Novel Features
|
| 64 |
+
|
| 65 |
+
### Auto-Scan Mode
|
| 66 |
+
Toggle on the Dashboard fires real Torque events every 30 seconds autonomously. Countdown timer shows next scan. One click activates fully autonomous retention.
|
| 67 |
+
|
| 68 |
+
### Bulk Rescue
|
| 69 |
+
"Rescue All Critical" button on the Wallets page fires `churn_risk_high` events for every at-risk wallet in a single parallel batch call (`POST /api/torque/bulk-fire`). Progress counter updates live: `3/7 fired`.
|
| 70 |
+
|
| 71 |
+
### Toast Notification System
|
| 72 |
+
Every Torque event that fires — whether from Scan Now, per-wallet Intervene, or Bulk Rescue — triggers a toast in the bottom-right corner showing event name, wallet address, and first 10 chars of the ingestion ID. Visual proof of live activity.
|
| 73 |
+
|
| 74 |
+
### Keyboard Shortcut
|
| 75 |
+
Press `S` anywhere on the Dashboard to trigger a wallet scan. No mouse needed.
|
| 76 |
+
|
| 77 |
+
### Real-Time Status in Topbar + Sidebar
|
| 78 |
+
Topbar polls `/api/torque/status` every 8s — shows "TORQUE LIVE" (green pulse) or "TORQUE OFFLINE" (red). Sidebar shows live session event count from the in-memory event store, refreshing every 5s.
|
| 79 |
+
|
| 80 |
+
### Per-Wallet Intervene Buttons
|
| 81 |
+
Each at-risk wallet row has a contextual action button:
|
| 82 |
+
- Critical → "Send Gift" → `churn_risk_high`
|
| 83 |
+
- High → "Enter Raffle" → `churn_risk_high`
|
| 84 |
+
- Medium → "Activate Rebate" → `churn_risk_medium`
|
| 85 |
+
|
| 86 |
+
Fires to real Torque ingest, shows ingestion ID inline.
|
| 87 |
+
|
| 88 |
+
### Create Campaign Modal
|
| 89 |
+
Full campaign creation form wired to `POST /api/torque/campaigns`: choose type (Leaderboard / Raffle / Gift / Rebate), name, budget, trigger event. Shows campaign ID on success.
|
| 90 |
+
|
| 91 |
+
### In-Memory Event Store
|
| 92 |
+
Session-wide singleton tracking every ingestion ID returned by Torque. Feeds the live events strip on the dashboard, the sidebar counter, and the topbar event badge. No database needed — restarts clean.
|
| 93 |
+
|
| 94 |
+
---
|
| 95 |
+
|
| 96 |
+
## Architecture
|
| 97 |
+
|
| 98 |
+
```
|
| 99 |
+
┌─────────────────────────────────────────────────────────┐
|
| 100 |
+
│ FlowState UI │
|
| 101 |
+
│ Dashboard · Wallets · Campaigns · Analytics · Agent │
|
| 102 |
+
│ │
|
| 103 |
+
│ Auto-scan (30s) · Bulk Rescue · Toast Notifications │
|
| 104 |
+
└──────────────┬──────────────────────────────────────────┘
|
| 105 |
+
│ Next.js API Routes
|
| 106 |
+
┌─────────┴───────────┐
|
| 107 |
+
│ │
|
| 108 |
+
┌────▼──────┐ ┌─────────▼────────┐
|
| 109 |
+
│ AI Engine │ │ Event Store │
|
| 110 |
+
│ 5-signal │ │ in-memory, │
|
| 111 |
+
│ churn │ │ session-wide │
|
| 112 |
+
│ scoring │ │ tracks IDs │
|
| 113 |
+
└────┬──────┘ └─────────┬────────┘
|
| 114 |
+
└─────────┬───────────┘
|
| 115 |
+
│
|
| 116 |
+
┌─────────▼────────────┐
|
| 117 |
+
│ torque-mcp.ts │
|
| 118 |
+
│ │
|
| 119 |
+
│ ingest.torque.so │ ← x-api-key (tq_ key)
|
| 120 |
+
│ server.torque.so │ ← Authorization: Bearer JWT
|
| 121 |
+
└──────────────────────┘
|
| 122 |
+
```
|
| 123 |
+
|
| 124 |
+
### Churn Scoring Model (5 signals)
|
| 125 |
+
|
| 126 |
+
```typescript
|
| 127 |
+
score += daysInactive >= 30 ? 40 : daysInactive >= 14 ? 25 : daysInactive >= 7 ? 15 : 0
|
| 128 |
+
score += volumeDropPct >= 80 ? 30 : volumeDropPct >= 50 ? 20 : volumeDropPct >= 25 ? 10 : 0
|
| 129 |
+
score += uniqueProtocols <= 1 ? 15 : uniqueProtocols <= 3 ? 8 : 0
|
| 130 |
+
score += currentStreak === 0 ? 10 : 0
|
| 131 |
+
score += hasLiquidation ? 5 : 0
|
| 132 |
+
// critical ≥80 high ≥60 medium ≥40 low ≥20 safe <20
|
| 133 |
+
```
|
| 134 |
+
|
| 135 |
+
---
|
| 136 |
|
| 137 |
## Pages
|
| 138 |
|
| 139 |
+
| Page | Features |
|
| 140 |
+
|------|---------|
|
| 141 |
+
| Dashboard | Live events strip, Scan Now, auto-scan toggle (30s), real Torque status badge, toast on fire |
|
| 142 |
+
| Wallets | Per-wallet Intervene buttons, Bulk Rescue All Critical, ingestion ID inline confirmation |
|
| 143 |
+
| Campaigns | Campaign cards + Create Campaign modal → real `POST /api/torque/campaigns` |
|
| 144 |
+
| Leaderboard | Top-3 podium, sortable rankings, scoring formula display |
|
| 145 |
+
| Analytics | Retention cohort heatmap, custom event breakdown, KPI charts |
|
| 146 |
+
| AI Agent | Start/pause, live feed with real scan results, threshold config |
|
| 147 |
+
|
| 148 |
+
---
|
| 149 |
|
| 150 |
## Run Locally
|
| 151 |
|
| 152 |
+
```bash
|
| 153 |
+
git clone https://github.com/YOUR_USERNAME/flowstate
|
| 154 |
+
cd flowstate && npm install
|
| 155 |
+
|
| 156 |
+
cp .env.example .env.local
|
| 157 |
+
# Add both credentials (see below)
|
| 158 |
+
npm run dev
|
| 159 |
```
|
| 160 |
+
|
| 161 |
+
Get credentials at [platform.torque.so/settings](https://platform.torque.so/settings)
|
| 162 |
+
|
| 163 |
+
### Two Credentials Required
|
| 164 |
+
|
| 165 |
+
| Variable | Format | Endpoint |
|
| 166 |
+
|----------|--------|---------|
|
| 167 |
+
| `TORQUE_API_KEY` | `eyJ...` JWT | `server.torque.so` (REST API) |
|
| 168 |
+
| `TORQUE_INGEST_KEY` | `tq_...` | `ingest.torque.so` (event ingest) |
|
| 169 |
+
|
| 170 |
+
---
|
| 171 |
|
| 172 |
## Tech Stack
|
| 173 |
|
| 174 |
+
Next.js 14 · TypeScript · Tailwind CSS · Recharts · Lucide · Torque MCP
|
| 175 |
+
|
| 176 |
+
---
|
| 177 |
|
| 178 |
## Friction Log
|
| 179 |
|
| 180 |
+
### What Worked Well
|
| 181 |
+
- Custom events are the superpower — any signal maps to any campaign type instantly
|
| 182 |
+
- Four primitives cover every DeFi retention play — nothing missing
|
| 183 |
+
- Formula engine is perfect for `SUM(swap_volume)` leaderboards
|
| 184 |
+
- Ingest endpoint is fast — all 10 test events confirmed in under 2s
|
| 185 |
+
- MCP server makes campaign creation scriptable from any LLM
|
| 186 |
+
|
| 187 |
+
### What Could Be Better
|
| 188 |
|
| 189 |
+
**1. Two credentials, no joint setup docs**
|
| 190 |
+
JWT and `tq_` ingest key serve different endpoints and are obtained separately. Discovered by testing, not documentation. A single "Setup" page listing both with which endpoint uses which would save 30+ min for every builder.
|
| 191 |
+
|
| 192 |
+
**2. Custom event schema must be pre-created before ingest**
|
| 193 |
+
Events silently drop if the schema hasn't been registered via MCP first. No error message explains why. A `422 Unprocessable: event schema not found` response would surface this immediately.
|
| 194 |
+
|
| 195 |
+
**3. No batch ingest endpoint**
|
| 196 |
+
Scanning 10,000 wallets = 10,000 sequential HTTP calls. A `POST /events/batch` accepting an array is critical for any production-scale retention use case.
|
| 197 |
+
|
| 198 |
+
**4. Campaign creation API undocumented**
|
| 199 |
+
`POST server.torque.so/campaigns` exists but has no public schema. Had to infer parameter names from MCP tool signatures. A REST reference page would unblock builders.
|
| 200 |
+
|
| 201 |
+
**5. MCP auth is stateful with no built-in retry**
|
| 202 |
+
Must call `auth()` before any other MCP tool. No automatic re-auth on token expiry. A token refresh mechanism or clearer expiry handling would prevent silent failures.
|
| 203 |
+
|
| 204 |
+
**6. No webhook callbacks on campaign events**
|
| 205 |
+
No way to know when a raffle winner is drawn or a gift is claimed without polling. Webhooks would enable real closed-loop retention (detect churn → fire event → get callback when reward claimed → record recovery).
|
| 206 |
+
|
| 207 |
+
---
|
| 208 |
|
| 209 |
+
Built for the Torque Hackathon · [platform.torque.so](https://platform.torque.so)
|
|
@@ -0,0 +1,1939 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
{
|
| 2 |
+
"name": "flowstate",
|
| 3 |
+
"version": "1.0.0",
|
| 4 |
+
"lockfileVersion": 3,
|
| 5 |
+
"requires": true,
|
| 6 |
+
"packages": {
|
| 7 |
+
"": {
|
| 8 |
+
"name": "flowstate",
|
| 9 |
+
"version": "1.0.0",
|
| 10 |
+
"dependencies": {
|
| 11 |
+
"@types/node": "^20.0.0",
|
| 12 |
+
"@types/react": "^18.3.0",
|
| 13 |
+
"@types/react-dom": "^18.3.0",
|
| 14 |
+
"autoprefixer": "^10.4.0",
|
| 15 |
+
"clsx": "^2.1.0",
|
| 16 |
+
"date-fns": "^4.1.0",
|
| 17 |
+
"lucide-react": "^1.0.0",
|
| 18 |
+
"next": "^14.2.0",
|
| 19 |
+
"postcss": "^8.5.0",
|
| 20 |
+
"react": "^18.3.0",
|
| 21 |
+
"react-dom": "^18.3.0",
|
| 22 |
+
"recharts": "^2.12.0",
|
| 23 |
+
"tailwind-merge": "^2.3.0",
|
| 24 |
+
"tailwindcss": "^3.4.0",
|
| 25 |
+
"typescript": "^5.4.0"
|
| 26 |
+
}
|
| 27 |
+
},
|
| 28 |
+
"node_modules/@alloc/quick-lru": {
|
| 29 |
+
"version": "5.2.0",
|
| 30 |
+
"resolved": "https://registry.npmjs.org/@alloc/quick-lru/-/quick-lru-5.2.0.tgz",
|
| 31 |
+
"integrity": "sha512-UrcABB+4bUrFABwbluTIBErXwvbsU/V7TZWfmbgJfbkwiBuziS9gxdODUyuiecfdGQ85jglMW6juS3+z5TsKLw==",
|
| 32 |
+
"license": "MIT",
|
| 33 |
+
"engines": {
|
| 34 |
+
"node": ">=10"
|
| 35 |
+
},
|
| 36 |
+
"funding": {
|
| 37 |
+
"url": "https://github.com/sponsors/sindresorhus"
|
| 38 |
+
}
|
| 39 |
+
},
|
| 40 |
+
"node_modules/@babel/runtime": {
|
| 41 |
+
"version": "7.29.2",
|
| 42 |
+
"resolved": "https://registry.npmjs.org/@babel/runtime/-/runtime-7.29.2.tgz",
|
| 43 |
+
"integrity": "sha512-JiDShH45zKHWyGe4ZNVRrCjBz8Nh9TMmZG1kh4QTK8hCBTWBi8Da+i7s1fJw7/lYpM4ccepSNfqzZ/QvABBi5g==",
|
| 44 |
+
"license": "MIT",
|
| 45 |
+
"engines": {
|
| 46 |
+
"node": ">=6.9.0"
|
| 47 |
+
}
|
| 48 |
+
},
|
| 49 |
+
"node_modules/@jridgewell/gen-mapping": {
|
| 50 |
+
"version": "0.3.13",
|
| 51 |
+
"resolved": "https://registry.npmjs.org/@jridgewell/gen-mapping/-/gen-mapping-0.3.13.tgz",
|
| 52 |
+
"integrity": "sha512-2kkt/7niJ6MgEPxF0bYdQ6etZaA+fQvDcLKckhy1yIQOzaoKjBBjSj63/aLVjYE3qhRt5dvM+uUyfCg6UKCBbA==",
|
| 53 |
+
"license": "MIT",
|
| 54 |
+
"dependencies": {
|
| 55 |
+
"@jridgewell/sourcemap-codec": "^1.5.0",
|
| 56 |
+
"@jridgewell/trace-mapping": "^0.3.24"
|
| 57 |
+
}
|
| 58 |
+
},
|
| 59 |
+
"node_modules/@jridgewell/resolve-uri": {
|
| 60 |
+
"version": "3.1.2",
|
| 61 |
+
"resolved": "https://registry.npmjs.org/@jridgewell/resolve-uri/-/resolve-uri-3.1.2.tgz",
|
| 62 |
+
"integrity": "sha512-bRISgCIjP20/tbWSPWMEi54QVPRZExkuD9lJL+UIxUKtwVJA8wW1Trb1jMs1RFXo1CBTNZ/5hpC9QvmKWdopKw==",
|
| 63 |
+
"license": "MIT",
|
| 64 |
+
"engines": {
|
| 65 |
+
"node": ">=6.0.0"
|
| 66 |
+
}
|
| 67 |
+
},
|
| 68 |
+
"node_modules/@jridgewell/sourcemap-codec": {
|
| 69 |
+
"version": "1.5.5",
|
| 70 |
+
"resolved": "https://registry.npmjs.org/@jridgewell/sourcemap-codec/-/sourcemap-codec-1.5.5.tgz",
|
| 71 |
+
"integrity": "sha512-cYQ9310grqxueWbl+WuIUIaiUaDcj7WOq5fVhEljNVgRfOUhY9fy2zTvfoqWsnebh8Sl70VScFbICvJnLKB0Og==",
|
| 72 |
+
"license": "MIT"
|
| 73 |
+
},
|
| 74 |
+
"node_modules/@jridgewell/trace-mapping": {
|
| 75 |
+
"version": "0.3.31",
|
| 76 |
+
"resolved": "https://registry.npmjs.org/@jridgewell/trace-mapping/-/trace-mapping-0.3.31.tgz",
|
| 77 |
+
"integrity": "sha512-zzNR+SdQSDJzc8joaeP8QQoCQr8NuYx2dIIytl1QeBEZHJ9uW6hebsrYgbz8hJwUQao3TWCMtmfV8Nu1twOLAw==",
|
| 78 |
+
"license": "MIT",
|
| 79 |
+
"dependencies": {
|
| 80 |
+
"@jridgewell/resolve-uri": "^3.1.0",
|
| 81 |
+
"@jridgewell/sourcemap-codec": "^1.4.14"
|
| 82 |
+
}
|
| 83 |
+
},
|
| 84 |
+
"node_modules/@next/env": {
|
| 85 |
+
"version": "14.2.35",
|
| 86 |
+
"resolved": "https://registry.npmjs.org/@next/env/-/env-14.2.35.tgz",
|
| 87 |
+
"integrity": "sha512-DuhvCtj4t9Gwrx80dmz2F4t/zKQ4ktN8WrMwOuVzkJfBilwAwGr6v16M5eI8yCuZ63H9TTuEU09Iu2HqkzFPVQ==",
|
| 88 |
+
"license": "MIT"
|
| 89 |
+
},
|
| 90 |
+
"node_modules/@next/swc-darwin-arm64": {
|
| 91 |
+
"version": "14.2.33",
|
| 92 |
+
"resolved": "https://registry.npmjs.org/@next/swc-darwin-arm64/-/swc-darwin-arm64-14.2.33.tgz",
|
| 93 |
+
"integrity": "sha512-HqYnb6pxlsshoSTubdXKu15g3iivcbsMXg4bYpjL2iS/V6aQot+iyF4BUc2qA/J/n55YtvE4PHMKWBKGCF/+wA==",
|
| 94 |
+
"cpu": [
|
| 95 |
+
"arm64"
|
| 96 |
+
],
|
| 97 |
+
"license": "MIT",
|
| 98 |
+
"optional": true,
|
| 99 |
+
"os": [
|
| 100 |
+
"darwin"
|
| 101 |
+
],
|
| 102 |
+
"engines": {
|
| 103 |
+
"node": ">= 10"
|
| 104 |
+
}
|
| 105 |
+
},
|
| 106 |
+
"node_modules/@next/swc-darwin-x64": {
|
| 107 |
+
"version": "14.2.33",
|
| 108 |
+
"resolved": "https://registry.npmjs.org/@next/swc-darwin-x64/-/swc-darwin-x64-14.2.33.tgz",
|
| 109 |
+
"integrity": "sha512-8HGBeAE5rX3jzKvF593XTTFg3gxeU4f+UWnswa6JPhzaR6+zblO5+fjltJWIZc4aUalqTclvN2QtTC37LxvZAA==",
|
| 110 |
+
"cpu": [
|
| 111 |
+
"x64"
|
| 112 |
+
],
|
| 113 |
+
"license": "MIT",
|
| 114 |
+
"optional": true,
|
| 115 |
+
"os": [
|
| 116 |
+
"darwin"
|
| 117 |
+
],
|
| 118 |
+
"engines": {
|
| 119 |
+
"node": ">= 10"
|
| 120 |
+
}
|
| 121 |
+
},
|
| 122 |
+
"node_modules/@next/swc-linux-arm64-gnu": {
|
| 123 |
+
"version": "14.2.33",
|
| 124 |
+
"resolved": "https://registry.npmjs.org/@next/swc-linux-arm64-gnu/-/swc-linux-arm64-gnu-14.2.33.tgz",
|
| 125 |
+
"integrity": "sha512-JXMBka6lNNmqbkvcTtaX8Gu5by9547bukHQvPoLe9VRBx1gHwzf5tdt4AaezW85HAB3pikcvyqBToRTDA4DeLw==",
|
| 126 |
+
"cpu": [
|
| 127 |
+
"arm64"
|
| 128 |
+
],
|
| 129 |
+
"license": "MIT",
|
| 130 |
+
"optional": true,
|
| 131 |
+
"os": [
|
| 132 |
+
"linux"
|
| 133 |
+
],
|
| 134 |
+
"engines": {
|
| 135 |
+
"node": ">= 10"
|
| 136 |
+
}
|
| 137 |
+
},
|
| 138 |
+
"node_modules/@next/swc-linux-arm64-musl": {
|
| 139 |
+
"version": "14.2.33",
|
| 140 |
+
"resolved": "https://registry.npmjs.org/@next/swc-linux-arm64-musl/-/swc-linux-arm64-musl-14.2.33.tgz",
|
| 141 |
+
"integrity": "sha512-Bm+QulsAItD/x6Ih8wGIMfRJy4G73tu1HJsrccPW6AfqdZd0Sfm5Imhgkgq2+kly065rYMnCOxTBvmvFY1BKfg==",
|
| 142 |
+
"cpu": [
|
| 143 |
+
"arm64"
|
| 144 |
+
],
|
| 145 |
+
"license": "MIT",
|
| 146 |
+
"optional": true,
|
| 147 |
+
"os": [
|
| 148 |
+
"linux"
|
| 149 |
+
],
|
| 150 |
+
"engines": {
|
| 151 |
+
"node": ">= 10"
|
| 152 |
+
}
|
| 153 |
+
},
|
| 154 |
+
"node_modules/@next/swc-linux-x64-gnu": {
|
| 155 |
+
"version": "14.2.33",
|
| 156 |
+
"resolved": "https://registry.npmjs.org/@next/swc-linux-x64-gnu/-/swc-linux-x64-gnu-14.2.33.tgz",
|
| 157 |
+
"integrity": "sha512-FnFn+ZBgsVMbGDsTqo8zsnRzydvsGV8vfiWwUo1LD8FTmPTdV+otGSWKc4LJec0oSexFnCYVO4hX8P8qQKaSlg==",
|
| 158 |
+
"cpu": [
|
| 159 |
+
"x64"
|
| 160 |
+
],
|
| 161 |
+
"license": "MIT",
|
| 162 |
+
"optional": true,
|
| 163 |
+
"os": [
|
| 164 |
+
"linux"
|
| 165 |
+
],
|
| 166 |
+
"engines": {
|
| 167 |
+
"node": ">= 10"
|
| 168 |
+
}
|
| 169 |
+
},
|
| 170 |
+
"node_modules/@next/swc-linux-x64-musl": {
|
| 171 |
+
"version": "14.2.33",
|
| 172 |
+
"resolved": "https://registry.npmjs.org/@next/swc-linux-x64-musl/-/swc-linux-x64-musl-14.2.33.tgz",
|
| 173 |
+
"integrity": "sha512-345tsIWMzoXaQndUTDv1qypDRiebFxGYx9pYkhwY4hBRaOLt8UGfiWKr9FSSHs25dFIf8ZqIFaPdy5MljdoawA==",
|
| 174 |
+
"cpu": [
|
| 175 |
+
"x64"
|
| 176 |
+
],
|
| 177 |
+
"license": "MIT",
|
| 178 |
+
"optional": true,
|
| 179 |
+
"os": [
|
| 180 |
+
"linux"
|
| 181 |
+
],
|
| 182 |
+
"engines": {
|
| 183 |
+
"node": ">= 10"
|
| 184 |
+
}
|
| 185 |
+
},
|
| 186 |
+
"node_modules/@next/swc-win32-arm64-msvc": {
|
| 187 |
+
"version": "14.2.33",
|
| 188 |
+
"resolved": "https://registry.npmjs.org/@next/swc-win32-arm64-msvc/-/swc-win32-arm64-msvc-14.2.33.tgz",
|
| 189 |
+
"integrity": "sha512-nscpt0G6UCTkrT2ppnJnFsYbPDQwmum4GNXYTeoTIdsmMydSKFz9Iny2jpaRupTb+Wl298+Rh82WKzt9LCcqSQ==",
|
| 190 |
+
"cpu": [
|
| 191 |
+
"arm64"
|
| 192 |
+
],
|
| 193 |
+
"license": "MIT",
|
| 194 |
+
"optional": true,
|
| 195 |
+
"os": [
|
| 196 |
+
"win32"
|
| 197 |
+
],
|
| 198 |
+
"engines": {
|
| 199 |
+
"node": ">= 10"
|
| 200 |
+
}
|
| 201 |
+
},
|
| 202 |
+
"node_modules/@next/swc-win32-ia32-msvc": {
|
| 203 |
+
"version": "14.2.33",
|
| 204 |
+
"resolved": "https://registry.npmjs.org/@next/swc-win32-ia32-msvc/-/swc-win32-ia32-msvc-14.2.33.tgz",
|
| 205 |
+
"integrity": "sha512-pc9LpGNKhJ0dXQhZ5QMmYxtARwwmWLpeocFmVG5Z0DzWq5Uf0izcI8tLc+qOpqxO1PWqZ5A7J1blrUIKrIFc7Q==",
|
| 206 |
+
"cpu": [
|
| 207 |
+
"ia32"
|
| 208 |
+
],
|
| 209 |
+
"license": "MIT",
|
| 210 |
+
"optional": true,
|
| 211 |
+
"os": [
|
| 212 |
+
"win32"
|
| 213 |
+
],
|
| 214 |
+
"engines": {
|
| 215 |
+
"node": ">= 10"
|
| 216 |
+
}
|
| 217 |
+
},
|
| 218 |
+
"node_modules/@next/swc-win32-x64-msvc": {
|
| 219 |
+
"version": "14.2.33",
|
| 220 |
+
"resolved": "https://registry.npmjs.org/@next/swc-win32-x64-msvc/-/swc-win32-x64-msvc-14.2.33.tgz",
|
| 221 |
+
"integrity": "sha512-nOjfZMy8B94MdisuzZo9/57xuFVLHJaDj5e/xrduJp9CV2/HrfxTRH2fbyLe+K9QT41WBLUd4iXX3R7jBp0EUg==",
|
| 222 |
+
"cpu": [
|
| 223 |
+
"x64"
|
| 224 |
+
],
|
| 225 |
+
"license": "MIT",
|
| 226 |
+
"optional": true,
|
| 227 |
+
"os": [
|
| 228 |
+
"win32"
|
| 229 |
+
],
|
| 230 |
+
"engines": {
|
| 231 |
+
"node": ">= 10"
|
| 232 |
+
}
|
| 233 |
+
},
|
| 234 |
+
"node_modules/@nodelib/fs.scandir": {
|
| 235 |
+
"version": "2.1.5",
|
| 236 |
+
"resolved": "https://registry.npmjs.org/@nodelib/fs.scandir/-/fs.scandir-2.1.5.tgz",
|
| 237 |
+
"integrity": "sha512-vq24Bq3ym5HEQm2NKCr3yXDwjc7vTsEThRDnkp2DK9p1uqLR+DHurm/NOTo0KG7HYHU7eppKZj3MyqYuMBf62g==",
|
| 238 |
+
"license": "MIT",
|
| 239 |
+
"dependencies": {
|
| 240 |
+
"@nodelib/fs.stat": "2.0.5",
|
| 241 |
+
"run-parallel": "^1.1.9"
|
| 242 |
+
},
|
| 243 |
+
"engines": {
|
| 244 |
+
"node": ">= 8"
|
| 245 |
+
}
|
| 246 |
+
},
|
| 247 |
+
"node_modules/@nodelib/fs.stat": {
|
| 248 |
+
"version": "2.0.5",
|
| 249 |
+
"resolved": "https://registry.npmjs.org/@nodelib/fs.stat/-/fs.stat-2.0.5.tgz",
|
| 250 |
+
"integrity": "sha512-RkhPPp2zrqDAQA/2jNhnztcPAlv64XdhIp7a7454A5ovI7Bukxgt7MX7udwAu3zg1DcpPU0rz3VV1SeaqvY4+A==",
|
| 251 |
+
"license": "MIT",
|
| 252 |
+
"engines": {
|
| 253 |
+
"node": ">= 8"
|
| 254 |
+
}
|
| 255 |
+
},
|
| 256 |
+
"node_modules/@nodelib/fs.walk": {
|
| 257 |
+
"version": "1.2.8",
|
| 258 |
+
"resolved": "https://registry.npmjs.org/@nodelib/fs.walk/-/fs.walk-1.2.8.tgz",
|
| 259 |
+
"integrity": "sha512-oGB+UxlgWcgQkgwo8GcEGwemoTFt3FIO9ababBmaGwXIoBKZ+GTy0pP185beGg7Llih/NSHSV2XAs1lnznocSg==",
|
| 260 |
+
"license": "MIT",
|
| 261 |
+
"dependencies": {
|
| 262 |
+
"@nodelib/fs.scandir": "2.1.5",
|
| 263 |
+
"fastq": "^1.6.0"
|
| 264 |
+
},
|
| 265 |
+
"engines": {
|
| 266 |
+
"node": ">= 8"
|
| 267 |
+
}
|
| 268 |
+
},
|
| 269 |
+
"node_modules/@swc/counter": {
|
| 270 |
+
"version": "0.1.3",
|
| 271 |
+
"resolved": "https://registry.npmjs.org/@swc/counter/-/counter-0.1.3.tgz",
|
| 272 |
+
"integrity": "sha512-e2BR4lsJkkRlKZ/qCHPw9ZaSxc0MVUd7gtbtaB7aMvHeJVYe8sOB8DBZkP2DtISHGSku9sCK6T6cnY0CtXrOCQ==",
|
| 273 |
+
"license": "Apache-2.0"
|
| 274 |
+
},
|
| 275 |
+
"node_modules/@swc/helpers": {
|
| 276 |
+
"version": "0.5.5",
|
| 277 |
+
"resolved": "https://registry.npmjs.org/@swc/helpers/-/helpers-0.5.5.tgz",
|
| 278 |
+
"integrity": "sha512-KGYxvIOXcceOAbEk4bi/dVLEK9z8sZ0uBB3Il5b1rhfClSpcX0yfRO0KmTkqR2cnQDymwLB+25ZyMzICg/cm/A==",
|
| 279 |
+
"license": "Apache-2.0",
|
| 280 |
+
"dependencies": {
|
| 281 |
+
"@swc/counter": "^0.1.3",
|
| 282 |
+
"tslib": "^2.4.0"
|
| 283 |
+
}
|
| 284 |
+
},
|
| 285 |
+
"node_modules/@types/d3-array": {
|
| 286 |
+
"version": "3.2.2",
|
| 287 |
+
"resolved": "https://registry.npmjs.org/@types/d3-array/-/d3-array-3.2.2.tgz",
|
| 288 |
+
"integrity": "sha512-hOLWVbm7uRza0BYXpIIW5pxfrKe0W+D5lrFiAEYR+pb6w3N2SwSMaJbXdUfSEv+dT4MfHBLtn5js0LAWaO6otw==",
|
| 289 |
+
"license": "MIT"
|
| 290 |
+
},
|
| 291 |
+
"node_modules/@types/d3-color": {
|
| 292 |
+
"version": "3.1.3",
|
| 293 |
+
"resolved": "https://registry.npmjs.org/@types/d3-color/-/d3-color-3.1.3.tgz",
|
| 294 |
+
"integrity": "sha512-iO90scth9WAbmgv7ogoq57O9YpKmFBbmoEoCHDB2xMBY0+/KVrqAaCDyCE16dUspeOvIxFFRI+0sEtqDqy2b4A==",
|
| 295 |
+
"license": "MIT"
|
| 296 |
+
},
|
| 297 |
+
"node_modules/@types/d3-ease": {
|
| 298 |
+
"version": "3.0.2",
|
| 299 |
+
"resolved": "https://registry.npmjs.org/@types/d3-ease/-/d3-ease-3.0.2.tgz",
|
| 300 |
+
"integrity": "sha512-NcV1JjO5oDzoK26oMzbILE6HW7uVXOHLQvHshBUW4UMdZGfiY6v5BeQwh9a9tCzv+CeefZQHJt5SRgK154RtiA==",
|
| 301 |
+
"license": "MIT"
|
| 302 |
+
},
|
| 303 |
+
"node_modules/@types/d3-interpolate": {
|
| 304 |
+
"version": "3.0.4",
|
| 305 |
+
"resolved": "https://registry.npmjs.org/@types/d3-interpolate/-/d3-interpolate-3.0.4.tgz",
|
| 306 |
+
"integrity": "sha512-mgLPETlrpVV1YRJIglr4Ez47g7Yxjl1lj7YKsiMCb27VJH9W8NVM6Bb9d8kkpG/uAQS5AmbA48q2IAolKKo1MA==",
|
| 307 |
+
"license": "MIT",
|
| 308 |
+
"dependencies": {
|
| 309 |
+
"@types/d3-color": "*"
|
| 310 |
+
}
|
| 311 |
+
},
|
| 312 |
+
"node_modules/@types/d3-path": {
|
| 313 |
+
"version": "3.1.1",
|
| 314 |
+
"resolved": "https://registry.npmjs.org/@types/d3-path/-/d3-path-3.1.1.tgz",
|
| 315 |
+
"integrity": "sha512-VMZBYyQvbGmWyWVea0EHs/BwLgxc+MKi1zLDCONksozI4YJMcTt8ZEuIR4Sb1MMTE8MMW49v0IwI5+b7RmfWlg==",
|
| 316 |
+
"license": "MIT"
|
| 317 |
+
},
|
| 318 |
+
"node_modules/@types/d3-scale": {
|
| 319 |
+
"version": "4.0.9",
|
| 320 |
+
"resolved": "https://registry.npmjs.org/@types/d3-scale/-/d3-scale-4.0.9.tgz",
|
| 321 |
+
"integrity": "sha512-dLmtwB8zkAeO/juAMfnV+sItKjlsw2lKdZVVy6LRr0cBmegxSABiLEpGVmSJJ8O08i4+sGR6qQtb6WtuwJdvVw==",
|
| 322 |
+
"license": "MIT",
|
| 323 |
+
"dependencies": {
|
| 324 |
+
"@types/d3-time": "*"
|
| 325 |
+
}
|
| 326 |
+
},
|
| 327 |
+
"node_modules/@types/d3-shape": {
|
| 328 |
+
"version": "3.1.8",
|
| 329 |
+
"resolved": "https://registry.npmjs.org/@types/d3-shape/-/d3-shape-3.1.8.tgz",
|
| 330 |
+
"integrity": "sha512-lae0iWfcDeR7qt7rA88BNiqdvPS5pFVPpo5OfjElwNaT2yyekbM0C9vK+yqBqEmHr6lDkRnYNoTBYlAgJa7a4w==",
|
| 331 |
+
"license": "MIT",
|
| 332 |
+
"dependencies": {
|
| 333 |
+
"@types/d3-path": "*"
|
| 334 |
+
}
|
| 335 |
+
},
|
| 336 |
+
"node_modules/@types/d3-time": {
|
| 337 |
+
"version": "3.0.4",
|
| 338 |
+
"resolved": "https://registry.npmjs.org/@types/d3-time/-/d3-time-3.0.4.tgz",
|
| 339 |
+
"integrity": "sha512-yuzZug1nkAAaBlBBikKZTgzCeA+k1uy4ZFwWANOfKw5z5LRhV0gNA7gNkKm7HoK+HRN0wX3EkxGk0fpbWhmB7g==",
|
| 340 |
+
"license": "MIT"
|
| 341 |
+
},
|
| 342 |
+
"node_modules/@types/d3-timer": {
|
| 343 |
+
"version": "3.0.2",
|
| 344 |
+
"resolved": "https://registry.npmjs.org/@types/d3-timer/-/d3-timer-3.0.2.tgz",
|
| 345 |
+
"integrity": "sha512-Ps3T8E8dZDam6fUyNiMkekK3XUsaUEik+idO9/YjPtfj2qruF8tFBXS7XhtE4iIXBLxhmLjP3SXpLhVf21I9Lw==",
|
| 346 |
+
"license": "MIT"
|
| 347 |
+
},
|
| 348 |
+
"node_modules/@types/node": {
|
| 349 |
+
"version": "20.19.39",
|
| 350 |
+
"resolved": "https://registry.npmjs.org/@types/node/-/node-20.19.39.tgz",
|
| 351 |
+
"integrity": "sha512-orrrD74MBUyK8jOAD/r0+lfa1I2MO6I+vAkmAWzMYbCcgrN4lCrmK52gRFQq/JRxfYPfonkr4b0jcY7Olqdqbw==",
|
| 352 |
+
"license": "MIT",
|
| 353 |
+
"dependencies": {
|
| 354 |
+
"undici-types": "~6.21.0"
|
| 355 |
+
}
|
| 356 |
+
},
|
| 357 |
+
"node_modules/@types/prop-types": {
|
| 358 |
+
"version": "15.7.15",
|
| 359 |
+
"resolved": "https://registry.npmjs.org/@types/prop-types/-/prop-types-15.7.15.tgz",
|
| 360 |
+
"integrity": "sha512-F6bEyamV9jKGAFBEmlQnesRPGOQqS2+Uwi0Em15xenOxHaf2hv6L8YCVn3rPdPJOiJfPiCnLIRyvwVaqMY3MIw==",
|
| 361 |
+
"license": "MIT"
|
| 362 |
+
},
|
| 363 |
+
"node_modules/@types/react": {
|
| 364 |
+
"version": "18.3.28",
|
| 365 |
+
"resolved": "https://registry.npmjs.org/@types/react/-/react-18.3.28.tgz",
|
| 366 |
+
"integrity": "sha512-z9VXpC7MWrhfWipitjNdgCauoMLRdIILQsAEV+ZesIzBq/oUlxk0m3ApZuMFCXdnS4U7KrI+l3WRUEGQ8K1QKw==",
|
| 367 |
+
"license": "MIT",
|
| 368 |
+
"dependencies": {
|
| 369 |
+
"@types/prop-types": "*",
|
| 370 |
+
"csstype": "^3.2.2"
|
| 371 |
+
}
|
| 372 |
+
},
|
| 373 |
+
"node_modules/@types/react-dom": {
|
| 374 |
+
"version": "18.3.7",
|
| 375 |
+
"resolved": "https://registry.npmjs.org/@types/react-dom/-/react-dom-18.3.7.tgz",
|
| 376 |
+
"integrity": "sha512-MEe3UeoENYVFXzoXEWsvcpg6ZvlrFNlOQ7EOsvhI3CfAXwzPfO8Qwuxd40nepsYKqyyVQnTdEfv68q91yLcKrQ==",
|
| 377 |
+
"license": "MIT",
|
| 378 |
+
"peerDependencies": {
|
| 379 |
+
"@types/react": "^18.0.0"
|
| 380 |
+
}
|
| 381 |
+
},
|
| 382 |
+
"node_modules/any-promise": {
|
| 383 |
+
"version": "1.3.0",
|
| 384 |
+
"resolved": "https://registry.npmjs.org/any-promise/-/any-promise-1.3.0.tgz",
|
| 385 |
+
"integrity": "sha512-7UvmKalWRt1wgjL1RrGxoSJW/0QZFIegpeGvZG9kjp8vrRu55XTHbwnqq2GpXm9uLbcuhxm3IqX9OB4MZR1b2A==",
|
| 386 |
+
"license": "MIT"
|
| 387 |
+
},
|
| 388 |
+
"node_modules/anymatch": {
|
| 389 |
+
"version": "3.1.3",
|
| 390 |
+
"resolved": "https://registry.npmjs.org/anymatch/-/anymatch-3.1.3.tgz",
|
| 391 |
+
"integrity": "sha512-KMReFUr0B4t+D+OBkjR3KYqvocp2XaSzO55UcB6mgQMd3KbcE+mWTyvVV7D/zsdEbNnV6acZUutkiHQXvTr1Rw==",
|
| 392 |
+
"license": "ISC",
|
| 393 |
+
"dependencies": {
|
| 394 |
+
"normalize-path": "^3.0.0",
|
| 395 |
+
"picomatch": "^2.0.4"
|
| 396 |
+
},
|
| 397 |
+
"engines": {
|
| 398 |
+
"node": ">= 8"
|
| 399 |
+
}
|
| 400 |
+
},
|
| 401 |
+
"node_modules/arg": {
|
| 402 |
+
"version": "5.0.2",
|
| 403 |
+
"resolved": "https://registry.npmjs.org/arg/-/arg-5.0.2.tgz",
|
| 404 |
+
"integrity": "sha512-PYjyFOLKQ9y57JvQ6QLo8dAgNqswh8M1RMJYdQduT6xbWSgK36P/Z/v+p888pM69jMMfS8Xd8F6I1kQ/I9HUGg==",
|
| 405 |
+
"license": "MIT"
|
| 406 |
+
},
|
| 407 |
+
"node_modules/autoprefixer": {
|
| 408 |
+
"version": "10.5.0",
|
| 409 |
+
"resolved": "https://registry.npmjs.org/autoprefixer/-/autoprefixer-10.5.0.tgz",
|
| 410 |
+
"integrity": "sha512-FMhOoZV4+qR6aTUALKX2rEqGG+oyATvwBt9IIzVR5rMa2HRWPkxf+P+PAJLD1I/H5/II+HuZcBJYEFBpq39ong==",
|
| 411 |
+
"funding": [
|
| 412 |
+
{
|
| 413 |
+
"type": "opencollective",
|
| 414 |
+
"url": "https://opencollective.com/postcss/"
|
| 415 |
+
},
|
| 416 |
+
{
|
| 417 |
+
"type": "tidelift",
|
| 418 |
+
"url": "https://tidelift.com/funding/github/npm/autoprefixer"
|
| 419 |
+
},
|
| 420 |
+
{
|
| 421 |
+
"type": "github",
|
| 422 |
+
"url": "https://github.com/sponsors/ai"
|
| 423 |
+
}
|
| 424 |
+
],
|
| 425 |
+
"license": "MIT",
|
| 426 |
+
"dependencies": {
|
| 427 |
+
"browserslist": "^4.28.2",
|
| 428 |
+
"caniuse-lite": "^1.0.30001787",
|
| 429 |
+
"fraction.js": "^5.3.4",
|
| 430 |
+
"picocolors": "^1.1.1",
|
| 431 |
+
"postcss-value-parser": "^4.2.0"
|
| 432 |
+
},
|
| 433 |
+
"bin": {
|
| 434 |
+
"autoprefixer": "bin/autoprefixer"
|
| 435 |
+
},
|
| 436 |
+
"engines": {
|
| 437 |
+
"node": "^10 || ^12 || >=14"
|
| 438 |
+
},
|
| 439 |
+
"peerDependencies": {
|
| 440 |
+
"postcss": "^8.1.0"
|
| 441 |
+
}
|
| 442 |
+
},
|
| 443 |
+
"node_modules/baseline-browser-mapping": {
|
| 444 |
+
"version": "2.10.25",
|
| 445 |
+
"resolved": "https://registry.npmjs.org/baseline-browser-mapping/-/baseline-browser-mapping-2.10.25.tgz",
|
| 446 |
+
"integrity": "sha512-QO/VHsXCQdnzADMfmkeOPvHdIAkoB7i0/rGjINPJEetLx75hNttVWGQ/jycHUDP9zZ9rupbm60WRxcwViB0MiA==",
|
| 447 |
+
"license": "Apache-2.0",
|
| 448 |
+
"bin": {
|
| 449 |
+
"baseline-browser-mapping": "dist/cli.cjs"
|
| 450 |
+
},
|
| 451 |
+
"engines": {
|
| 452 |
+
"node": ">=6.0.0"
|
| 453 |
+
}
|
| 454 |
+
},
|
| 455 |
+
"node_modules/binary-extensions": {
|
| 456 |
+
"version": "2.3.0",
|
| 457 |
+
"resolved": "https://registry.npmjs.org/binary-extensions/-/binary-extensions-2.3.0.tgz",
|
| 458 |
+
"integrity": "sha512-Ceh+7ox5qe7LJuLHoY0feh3pHuUDHAcRUeyL2VYghZwfpkNIy/+8Ocg0a3UuSoYzavmylwuLWQOf3hl0jjMMIw==",
|
| 459 |
+
"license": "MIT",
|
| 460 |
+
"engines": {
|
| 461 |
+
"node": ">=8"
|
| 462 |
+
},
|
| 463 |
+
"funding": {
|
| 464 |
+
"url": "https://github.com/sponsors/sindresorhus"
|
| 465 |
+
}
|
| 466 |
+
},
|
| 467 |
+
"node_modules/braces": {
|
| 468 |
+
"version": "3.0.3",
|
| 469 |
+
"resolved": "https://registry.npmjs.org/braces/-/braces-3.0.3.tgz",
|
| 470 |
+
"integrity": "sha512-yQbXgO/OSZVD2IsiLlro+7Hf6Q18EJrKSEsdoMzKePKXct3gvD8oLcOQdIzGupr5Fj+EDe8gO/lxc1BzfMpxvA==",
|
| 471 |
+
"license": "MIT",
|
| 472 |
+
"dependencies": {
|
| 473 |
+
"fill-range": "^7.1.1"
|
| 474 |
+
},
|
| 475 |
+
"engines": {
|
| 476 |
+
"node": ">=8"
|
| 477 |
+
}
|
| 478 |
+
},
|
| 479 |
+
"node_modules/browserslist": {
|
| 480 |
+
"version": "4.28.2",
|
| 481 |
+
"resolved": "https://registry.npmjs.org/browserslist/-/browserslist-4.28.2.tgz",
|
| 482 |
+
"integrity": "sha512-48xSriZYYg+8qXna9kwqjIVzuQxi+KYWp2+5nCYnYKPTr0LvD89Jqk2Or5ogxz0NUMfIjhh2lIUX/LyX9B4oIg==",
|
| 483 |
+
"funding": [
|
| 484 |
+
{
|
| 485 |
+
"type": "opencollective",
|
| 486 |
+
"url": "https://opencollective.com/browserslist"
|
| 487 |
+
},
|
| 488 |
+
{
|
| 489 |
+
"type": "tidelift",
|
| 490 |
+
"url": "https://tidelift.com/funding/github/npm/browserslist"
|
| 491 |
+
},
|
| 492 |
+
{
|
| 493 |
+
"type": "github",
|
| 494 |
+
"url": "https://github.com/sponsors/ai"
|
| 495 |
+
}
|
| 496 |
+
],
|
| 497 |
+
"license": "MIT",
|
| 498 |
+
"dependencies": {
|
| 499 |
+
"baseline-browser-mapping": "^2.10.12",
|
| 500 |
+
"caniuse-lite": "^1.0.30001782",
|
| 501 |
+
"electron-to-chromium": "^1.5.328",
|
| 502 |
+
"node-releases": "^2.0.36",
|
| 503 |
+
"update-browserslist-db": "^1.2.3"
|
| 504 |
+
},
|
| 505 |
+
"bin": {
|
| 506 |
+
"browserslist": "cli.js"
|
| 507 |
+
},
|
| 508 |
+
"engines": {
|
| 509 |
+
"node": "^6 || ^7 || ^8 || ^9 || ^10 || ^11 || ^12 || >=13.7"
|
| 510 |
+
}
|
| 511 |
+
},
|
| 512 |
+
"node_modules/busboy": {
|
| 513 |
+
"version": "1.6.0",
|
| 514 |
+
"resolved": "https://registry.npmjs.org/busboy/-/busboy-1.6.0.tgz",
|
| 515 |
+
"integrity": "sha512-8SFQbg/0hQ9xy3UNTB0YEnsNBbWfhf7RtnzpL7TkBiTBRfrQ9Fxcnz7VJsleJpyp6rVLvXiuORqjlHi5q+PYuA==",
|
| 516 |
+
"dependencies": {
|
| 517 |
+
"streamsearch": "^1.1.0"
|
| 518 |
+
},
|
| 519 |
+
"engines": {
|
| 520 |
+
"node": ">=10.16.0"
|
| 521 |
+
}
|
| 522 |
+
},
|
| 523 |
+
"node_modules/camelcase-css": {
|
| 524 |
+
"version": "2.0.1",
|
| 525 |
+
"resolved": "https://registry.npmjs.org/camelcase-css/-/camelcase-css-2.0.1.tgz",
|
| 526 |
+
"integrity": "sha512-QOSvevhslijgYwRx6Rv7zKdMF8lbRmx+uQGx2+vDc+KI/eBnsy9kit5aj23AgGu3pa4t9AgwbnXWqS+iOY+2aA==",
|
| 527 |
+
"license": "MIT",
|
| 528 |
+
"engines": {
|
| 529 |
+
"node": ">= 6"
|
| 530 |
+
}
|
| 531 |
+
},
|
| 532 |
+
"node_modules/caniuse-lite": {
|
| 533 |
+
"version": "1.0.30001791",
|
| 534 |
+
"resolved": "https://registry.npmjs.org/caniuse-lite/-/caniuse-lite-1.0.30001791.tgz",
|
| 535 |
+
"integrity": "sha512-yk0l/YSrOnFZk3UROpDLQD9+kC1l4meK/wed583AXrzoarMGJcbRi2Q4RaUYbKxYAsZ8sWmaSa/DsLmdBeI1vQ==",
|
| 536 |
+
"funding": [
|
| 537 |
+
{
|
| 538 |
+
"type": "opencollective",
|
| 539 |
+
"url": "https://opencollective.com/browserslist"
|
| 540 |
+
},
|
| 541 |
+
{
|
| 542 |
+
"type": "tidelift",
|
| 543 |
+
"url": "https://tidelift.com/funding/github/npm/caniuse-lite"
|
| 544 |
+
},
|
| 545 |
+
{
|
| 546 |
+
"type": "github",
|
| 547 |
+
"url": "https://github.com/sponsors/ai"
|
| 548 |
+
}
|
| 549 |
+
],
|
| 550 |
+
"license": "CC-BY-4.0"
|
| 551 |
+
},
|
| 552 |
+
"node_modules/chokidar": {
|
| 553 |
+
"version": "3.6.0",
|
| 554 |
+
"resolved": "https://registry.npmjs.org/chokidar/-/chokidar-3.6.0.tgz",
|
| 555 |
+
"integrity": "sha512-7VT13fmjotKpGipCW9JEQAusEPE+Ei8nl6/g4FBAmIm0GOOLMua9NDDo/DWp0ZAxCr3cPq5ZpBqmPAQgDda2Pw==",
|
| 556 |
+
"license": "MIT",
|
| 557 |
+
"dependencies": {
|
| 558 |
+
"anymatch": "~3.1.2",
|
| 559 |
+
"braces": "~3.0.2",
|
| 560 |
+
"glob-parent": "~5.1.2",
|
| 561 |
+
"is-binary-path": "~2.1.0",
|
| 562 |
+
"is-glob": "~4.0.1",
|
| 563 |
+
"normalize-path": "~3.0.0",
|
| 564 |
+
"readdirp": "~3.6.0"
|
| 565 |
+
},
|
| 566 |
+
"engines": {
|
| 567 |
+
"node": ">= 8.10.0"
|
| 568 |
+
},
|
| 569 |
+
"funding": {
|
| 570 |
+
"url": "https://paulmillr.com/funding/"
|
| 571 |
+
},
|
| 572 |
+
"optionalDependencies": {
|
| 573 |
+
"fsevents": "~2.3.2"
|
| 574 |
+
}
|
| 575 |
+
},
|
| 576 |
+
"node_modules/chokidar/node_modules/glob-parent": {
|
| 577 |
+
"version": "5.1.2",
|
| 578 |
+
"resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
|
| 579 |
+
"integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
|
| 580 |
+
"license": "ISC",
|
| 581 |
+
"dependencies": {
|
| 582 |
+
"is-glob": "^4.0.1"
|
| 583 |
+
},
|
| 584 |
+
"engines": {
|
| 585 |
+
"node": ">= 6"
|
| 586 |
+
}
|
| 587 |
+
},
|
| 588 |
+
"node_modules/client-only": {
|
| 589 |
+
"version": "0.0.1",
|
| 590 |
+
"resolved": "https://registry.npmjs.org/client-only/-/client-only-0.0.1.tgz",
|
| 591 |
+
"integrity": "sha512-IV3Ou0jSMzZrd3pZ48nLkT9DA7Ag1pnPzaiQhpW7c3RbcqqzvzzVu+L8gfqMp/8IM2MQtSiqaCxrrcfu8I8rMA==",
|
| 592 |
+
"license": "MIT"
|
| 593 |
+
},
|
| 594 |
+
"node_modules/clsx": {
|
| 595 |
+
"version": "2.1.1",
|
| 596 |
+
"resolved": "https://registry.npmjs.org/clsx/-/clsx-2.1.1.tgz",
|
| 597 |
+
"integrity": "sha512-eYm0QWBtUrBWZWG0d386OGAw16Z995PiOVo2B7bjWSbHedGl5e0ZWaq65kOGgUSNesEIDkB9ISbTg/JK9dhCZA==",
|
| 598 |
+
"license": "MIT",
|
| 599 |
+
"engines": {
|
| 600 |
+
"node": ">=6"
|
| 601 |
+
}
|
| 602 |
+
},
|
| 603 |
+
"node_modules/commander": {
|
| 604 |
+
"version": "4.1.1",
|
| 605 |
+
"resolved": "https://registry.npmjs.org/commander/-/commander-4.1.1.tgz",
|
| 606 |
+
"integrity": "sha512-NOKm8xhkzAjzFx8B2v5OAHT+u5pRQc2UCa2Vq9jYL/31o2wi9mxBA7LIFs3sV5VSC49z6pEhfbMULvShKj26WA==",
|
| 607 |
+
"license": "MIT",
|
| 608 |
+
"engines": {
|
| 609 |
+
"node": ">= 6"
|
| 610 |
+
}
|
| 611 |
+
},
|
| 612 |
+
"node_modules/cssesc": {
|
| 613 |
+
"version": "3.0.0",
|
| 614 |
+
"resolved": "https://registry.npmjs.org/cssesc/-/cssesc-3.0.0.tgz",
|
| 615 |
+
"integrity": "sha512-/Tb/JcjK111nNScGob5MNtsntNM1aCNUDipB/TkwZFhyDrrE47SOx/18wF2bbjgc3ZzCSKW1T5nt5EbFoAz/Vg==",
|
| 616 |
+
"license": "MIT",
|
| 617 |
+
"bin": {
|
| 618 |
+
"cssesc": "bin/cssesc"
|
| 619 |
+
},
|
| 620 |
+
"engines": {
|
| 621 |
+
"node": ">=4"
|
| 622 |
+
}
|
| 623 |
+
},
|
| 624 |
+
"node_modules/csstype": {
|
| 625 |
+
"version": "3.2.3",
|
| 626 |
+
"resolved": "https://registry.npmjs.org/csstype/-/csstype-3.2.3.tgz",
|
| 627 |
+
"integrity": "sha512-z1HGKcYy2xA8AGQfwrn0PAy+PB7X/GSj3UVJW9qKyn43xWa+gl5nXmU4qqLMRzWVLFC8KusUX8T/0kCiOYpAIQ==",
|
| 628 |
+
"license": "MIT"
|
| 629 |
+
},
|
| 630 |
+
"node_modules/d3-array": {
|
| 631 |
+
"version": "3.2.4",
|
| 632 |
+
"resolved": "https://registry.npmjs.org/d3-array/-/d3-array-3.2.4.tgz",
|
| 633 |
+
"integrity": "sha512-tdQAmyA18i4J7wprpYq8ClcxZy3SC31QMeByyCFyRt7BVHdREQZ5lpzoe5mFEYZUWe+oq8HBvk9JjpibyEV4Jg==",
|
| 634 |
+
"license": "ISC",
|
| 635 |
+
"dependencies": {
|
| 636 |
+
"internmap": "1 - 2"
|
| 637 |
+
},
|
| 638 |
+
"engines": {
|
| 639 |
+
"node": ">=12"
|
| 640 |
+
}
|
| 641 |
+
},
|
| 642 |
+
"node_modules/d3-color": {
|
| 643 |
+
"version": "3.1.0",
|
| 644 |
+
"resolved": "https://registry.npmjs.org/d3-color/-/d3-color-3.1.0.tgz",
|
| 645 |
+
"integrity": "sha512-zg/chbXyeBtMQ1LbD/WSoW2DpC3I0mpmPdW+ynRTj/x2DAWYrIY7qeZIHidozwV24m4iavr15lNwIwLxRmOxhA==",
|
| 646 |
+
"license": "ISC",
|
| 647 |
+
"engines": {
|
| 648 |
+
"node": ">=12"
|
| 649 |
+
}
|
| 650 |
+
},
|
| 651 |
+
"node_modules/d3-ease": {
|
| 652 |
+
"version": "3.0.1",
|
| 653 |
+
"resolved": "https://registry.npmjs.org/d3-ease/-/d3-ease-3.0.1.tgz",
|
| 654 |
+
"integrity": "sha512-wR/XK3D3XcLIZwpbvQwQ5fK+8Ykds1ip7A2Txe0yxncXSdq1L9skcG7blcedkOX+ZcgxGAmLX1FrRGbADwzi0w==",
|
| 655 |
+
"license": "BSD-3-Clause",
|
| 656 |
+
"engines": {
|
| 657 |
+
"node": ">=12"
|
| 658 |
+
}
|
| 659 |
+
},
|
| 660 |
+
"node_modules/d3-format": {
|
| 661 |
+
"version": "3.1.2",
|
| 662 |
+
"resolved": "https://registry.npmjs.org/d3-format/-/d3-format-3.1.2.tgz",
|
| 663 |
+
"integrity": "sha512-AJDdYOdnyRDV5b6ArilzCPPwc1ejkHcoyFarqlPqT7zRYjhavcT3uSrqcMvsgh2CgoPbK3RCwyHaVyxYcP2Arg==",
|
| 664 |
+
"license": "ISC",
|
| 665 |
+
"engines": {
|
| 666 |
+
"node": ">=12"
|
| 667 |
+
}
|
| 668 |
+
},
|
| 669 |
+
"node_modules/d3-interpolate": {
|
| 670 |
+
"version": "3.0.1",
|
| 671 |
+
"resolved": "https://registry.npmjs.org/d3-interpolate/-/d3-interpolate-3.0.1.tgz",
|
| 672 |
+
"integrity": "sha512-3bYs1rOD33uo8aqJfKP3JWPAibgw8Zm2+L9vBKEHJ2Rg+viTR7o5Mmv5mZcieN+FRYaAOWX5SJATX6k1PWz72g==",
|
| 673 |
+
"license": "ISC",
|
| 674 |
+
"dependencies": {
|
| 675 |
+
"d3-color": "1 - 3"
|
| 676 |
+
},
|
| 677 |
+
"engines": {
|
| 678 |
+
"node": ">=12"
|
| 679 |
+
}
|
| 680 |
+
},
|
| 681 |
+
"node_modules/d3-path": {
|
| 682 |
+
"version": "3.1.0",
|
| 683 |
+
"resolved": "https://registry.npmjs.org/d3-path/-/d3-path-3.1.0.tgz",
|
| 684 |
+
"integrity": "sha512-p3KP5HCf/bvjBSSKuXid6Zqijx7wIfNW+J/maPs+iwR35at5JCbLUT0LzF1cnjbCHWhqzQTIN2Jpe8pRebIEFQ==",
|
| 685 |
+
"license": "ISC",
|
| 686 |
+
"engines": {
|
| 687 |
+
"node": ">=12"
|
| 688 |
+
}
|
| 689 |
+
},
|
| 690 |
+
"node_modules/d3-scale": {
|
| 691 |
+
"version": "4.0.2",
|
| 692 |
+
"resolved": "https://registry.npmjs.org/d3-scale/-/d3-scale-4.0.2.tgz",
|
| 693 |
+
"integrity": "sha512-GZW464g1SH7ag3Y7hXjf8RoUuAFIqklOAq3MRl4OaWabTFJY9PN/E1YklhXLh+OQ3fM9yS2nOkCoS+WLZ6kvxQ==",
|
| 694 |
+
"license": "ISC",
|
| 695 |
+
"dependencies": {
|
| 696 |
+
"d3-array": "2.10.0 - 3",
|
| 697 |
+
"d3-format": "1 - 3",
|
| 698 |
+
"d3-interpolate": "1.2.0 - 3",
|
| 699 |
+
"d3-time": "2.1.1 - 3",
|
| 700 |
+
"d3-time-format": "2 - 4"
|
| 701 |
+
},
|
| 702 |
+
"engines": {
|
| 703 |
+
"node": ">=12"
|
| 704 |
+
}
|
| 705 |
+
},
|
| 706 |
+
"node_modules/d3-shape": {
|
| 707 |
+
"version": "3.2.0",
|
| 708 |
+
"resolved": "https://registry.npmjs.org/d3-shape/-/d3-shape-3.2.0.tgz",
|
| 709 |
+
"integrity": "sha512-SaLBuwGm3MOViRq2ABk3eLoxwZELpH6zhl3FbAoJ7Vm1gofKx6El1Ib5z23NUEhF9AsGl7y+dzLe5Cw2AArGTA==",
|
| 710 |
+
"license": "ISC",
|
| 711 |
+
"dependencies": {
|
| 712 |
+
"d3-path": "^3.1.0"
|
| 713 |
+
},
|
| 714 |
+
"engines": {
|
| 715 |
+
"node": ">=12"
|
| 716 |
+
}
|
| 717 |
+
},
|
| 718 |
+
"node_modules/d3-time": {
|
| 719 |
+
"version": "3.1.0",
|
| 720 |
+
"resolved": "https://registry.npmjs.org/d3-time/-/d3-time-3.1.0.tgz",
|
| 721 |
+
"integrity": "sha512-VqKjzBLejbSMT4IgbmVgDjpkYrNWUYJnbCGo874u7MMKIWsILRX+OpX/gTk8MqjpT1A/c6HY2dCA77ZN0lkQ2Q==",
|
| 722 |
+
"license": "ISC",
|
| 723 |
+
"dependencies": {
|
| 724 |
+
"d3-array": "2 - 3"
|
| 725 |
+
},
|
| 726 |
+
"engines": {
|
| 727 |
+
"node": ">=12"
|
| 728 |
+
}
|
| 729 |
+
},
|
| 730 |
+
"node_modules/d3-time-format": {
|
| 731 |
+
"version": "4.1.0",
|
| 732 |
+
"resolved": "https://registry.npmjs.org/d3-time-format/-/d3-time-format-4.1.0.tgz",
|
| 733 |
+
"integrity": "sha512-dJxPBlzC7NugB2PDLwo9Q8JiTR3M3e4/XANkreKSUxF8vvXKqm1Yfq4Q5dl8budlunRVlUUaDUgFt7eA8D6NLg==",
|
| 734 |
+
"license": "ISC",
|
| 735 |
+
"dependencies": {
|
| 736 |
+
"d3-time": "1 - 3"
|
| 737 |
+
},
|
| 738 |
+
"engines": {
|
| 739 |
+
"node": ">=12"
|
| 740 |
+
}
|
| 741 |
+
},
|
| 742 |
+
"node_modules/d3-timer": {
|
| 743 |
+
"version": "3.0.1",
|
| 744 |
+
"resolved": "https://registry.npmjs.org/d3-timer/-/d3-timer-3.0.1.tgz",
|
| 745 |
+
"integrity": "sha512-ndfJ/JxxMd3nw31uyKoY2naivF+r29V+Lc0svZxe1JvvIRmi8hUsrMvdOwgS1o6uBHmiz91geQ0ylPP0aj1VUA==",
|
| 746 |
+
"license": "ISC",
|
| 747 |
+
"engines": {
|
| 748 |
+
"node": ">=12"
|
| 749 |
+
}
|
| 750 |
+
},
|
| 751 |
+
"node_modules/date-fns": {
|
| 752 |
+
"version": "4.1.0",
|
| 753 |
+
"resolved": "https://registry.npmjs.org/date-fns/-/date-fns-4.1.0.tgz",
|
| 754 |
+
"integrity": "sha512-Ukq0owbQXxa/U3EGtsdVBkR1w7KOQ5gIBqdH2hkvknzZPYvBxb/aa6E8L7tmjFtkwZBu3UXBbjIgPo/Ez4xaNg==",
|
| 755 |
+
"license": "MIT",
|
| 756 |
+
"funding": {
|
| 757 |
+
"type": "github",
|
| 758 |
+
"url": "https://github.com/sponsors/kossnocorp"
|
| 759 |
+
}
|
| 760 |
+
},
|
| 761 |
+
"node_modules/decimal.js-light": {
|
| 762 |
+
"version": "2.5.1",
|
| 763 |
+
"resolved": "https://registry.npmjs.org/decimal.js-light/-/decimal.js-light-2.5.1.tgz",
|
| 764 |
+
"integrity": "sha512-qIMFpTMZmny+MMIitAB6D7iVPEorVw6YQRWkvarTkT4tBeSLLiHzcwj6q0MmYSFCiVpiqPJTJEYIrpcPzVEIvg==",
|
| 765 |
+
"license": "MIT"
|
| 766 |
+
},
|
| 767 |
+
"node_modules/didyoumean": {
|
| 768 |
+
"version": "1.2.2",
|
| 769 |
+
"resolved": "https://registry.npmjs.org/didyoumean/-/didyoumean-1.2.2.tgz",
|
| 770 |
+
"integrity": "sha512-gxtyfqMg7GKyhQmb056K7M3xszy/myH8w+B4RT+QXBQsvAOdc3XymqDDPHx1BgPgsdAA5SIifona89YtRATDzw==",
|
| 771 |
+
"license": "Apache-2.0"
|
| 772 |
+
},
|
| 773 |
+
"node_modules/dlv": {
|
| 774 |
+
"version": "1.1.3",
|
| 775 |
+
"resolved": "https://registry.npmjs.org/dlv/-/dlv-1.1.3.tgz",
|
| 776 |
+
"integrity": "sha512-+HlytyjlPKnIG8XuRG8WvmBP8xs8P71y+SKKS6ZXWoEgLuePxtDoUEiH7WkdePWrQ5JBpE6aoVqfZfJUQkjXwA==",
|
| 777 |
+
"license": "MIT"
|
| 778 |
+
},
|
| 779 |
+
"node_modules/dom-helpers": {
|
| 780 |
+
"version": "5.2.1",
|
| 781 |
+
"resolved": "https://registry.npmjs.org/dom-helpers/-/dom-helpers-5.2.1.tgz",
|
| 782 |
+
"integrity": "sha512-nRCa7CK3VTrM2NmGkIy4cbK7IZlgBE/PYMn55rrXefr5xXDP0LdtfPnblFDoVdcAfslJ7or6iqAUnx0CCGIWQA==",
|
| 783 |
+
"license": "MIT",
|
| 784 |
+
"dependencies": {
|
| 785 |
+
"@babel/runtime": "^7.8.7",
|
| 786 |
+
"csstype": "^3.0.2"
|
| 787 |
+
}
|
| 788 |
+
},
|
| 789 |
+
"node_modules/electron-to-chromium": {
|
| 790 |
+
"version": "1.5.348",
|
| 791 |
+
"resolved": "https://registry.npmjs.org/electron-to-chromium/-/electron-to-chromium-1.5.348.tgz",
|
| 792 |
+
"integrity": "sha512-QC2X59nRlycQQMc4ZXjSVBX+tSgJfgRtcrYHbIZLgOV2dCvefoQGegLR7lLXKgpPpSuVmJU19LMzGrSa2C7k3Q==",
|
| 793 |
+
"license": "ISC"
|
| 794 |
+
},
|
| 795 |
+
"node_modules/es-errors": {
|
| 796 |
+
"version": "1.3.0",
|
| 797 |
+
"resolved": "https://registry.npmjs.org/es-errors/-/es-errors-1.3.0.tgz",
|
| 798 |
+
"integrity": "sha512-Zf5H2Kxt2xjTvbJvP2ZWLEICxA6j+hAmMzIlypy4xcBg1vKVnx89Wy0GbS+kf5cwCVFFzdCFh2XSCFNULS6csw==",
|
| 799 |
+
"license": "MIT",
|
| 800 |
+
"engines": {
|
| 801 |
+
"node": ">= 0.4"
|
| 802 |
+
}
|
| 803 |
+
},
|
| 804 |
+
"node_modules/escalade": {
|
| 805 |
+
"version": "3.2.0",
|
| 806 |
+
"resolved": "https://registry.npmjs.org/escalade/-/escalade-3.2.0.tgz",
|
| 807 |
+
"integrity": "sha512-WUj2qlxaQtO4g6Pq5c29GTcWGDyd8itL8zTlipgECz3JesAiiOKotd8JU6otB3PACgG6xkJUyVhboMS+bje/jA==",
|
| 808 |
+
"license": "MIT",
|
| 809 |
+
"engines": {
|
| 810 |
+
"node": ">=6"
|
| 811 |
+
}
|
| 812 |
+
},
|
| 813 |
+
"node_modules/eventemitter3": {
|
| 814 |
+
"version": "4.0.7",
|
| 815 |
+
"resolved": "https://registry.npmjs.org/eventemitter3/-/eventemitter3-4.0.7.tgz",
|
| 816 |
+
"integrity": "sha512-8guHBZCwKnFhYdHr2ysuRWErTwhoN2X8XELRlrRwpmfeY2jjuUN4taQMsULKUVo1K4DvZl+0pgfyoysHxvmvEw==",
|
| 817 |
+
"license": "MIT"
|
| 818 |
+
},
|
| 819 |
+
"node_modules/fast-equals": {
|
| 820 |
+
"version": "5.4.0",
|
| 821 |
+
"resolved": "https://registry.npmjs.org/fast-equals/-/fast-equals-5.4.0.tgz",
|
| 822 |
+
"integrity": "sha512-jt2DW/aNFNwke7AUd+Z+e6pz39KO5rzdbbFCg2sGafS4mk13MI7Z8O5z9cADNn5lhGODIgLwug6TZO2ctf7kcw==",
|
| 823 |
+
"license": "MIT",
|
| 824 |
+
"engines": {
|
| 825 |
+
"node": ">=6.0.0"
|
| 826 |
+
}
|
| 827 |
+
},
|
| 828 |
+
"node_modules/fast-glob": {
|
| 829 |
+
"version": "3.3.3",
|
| 830 |
+
"resolved": "https://registry.npmjs.org/fast-glob/-/fast-glob-3.3.3.tgz",
|
| 831 |
+
"integrity": "sha512-7MptL8U0cqcFdzIzwOTHoilX9x5BrNqye7Z/LuC7kCMRio1EMSyqRK3BEAUD7sXRq4iT4AzTVuZdhgQ2TCvYLg==",
|
| 832 |
+
"license": "MIT",
|
| 833 |
+
"dependencies": {
|
| 834 |
+
"@nodelib/fs.stat": "^2.0.2",
|
| 835 |
+
"@nodelib/fs.walk": "^1.2.3",
|
| 836 |
+
"glob-parent": "^5.1.2",
|
| 837 |
+
"merge2": "^1.3.0",
|
| 838 |
+
"micromatch": "^4.0.8"
|
| 839 |
+
},
|
| 840 |
+
"engines": {
|
| 841 |
+
"node": ">=8.6.0"
|
| 842 |
+
}
|
| 843 |
+
},
|
| 844 |
+
"node_modules/fast-glob/node_modules/glob-parent": {
|
| 845 |
+
"version": "5.1.2",
|
| 846 |
+
"resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-5.1.2.tgz",
|
| 847 |
+
"integrity": "sha512-AOIgSQCepiJYwP3ARnGx+5VnTu2HBYdzbGP45eLw1vr3zB3vZLeyed1sC9hnbcOc9/SrMyM5RPQrkGz4aS9Zow==",
|
| 848 |
+
"license": "ISC",
|
| 849 |
+
"dependencies": {
|
| 850 |
+
"is-glob": "^4.0.1"
|
| 851 |
+
},
|
| 852 |
+
"engines": {
|
| 853 |
+
"node": ">= 6"
|
| 854 |
+
}
|
| 855 |
+
},
|
| 856 |
+
"node_modules/fastq": {
|
| 857 |
+
"version": "1.20.1",
|
| 858 |
+
"resolved": "https://registry.npmjs.org/fastq/-/fastq-1.20.1.tgz",
|
| 859 |
+
"integrity": "sha512-GGToxJ/w1x32s/D2EKND7kTil4n8OVk/9mycTc4VDza13lOvpUZTGX3mFSCtV9ksdGBVzvsyAVLM6mHFThxXxw==",
|
| 860 |
+
"license": "ISC",
|
| 861 |
+
"dependencies": {
|
| 862 |
+
"reusify": "^1.0.4"
|
| 863 |
+
}
|
| 864 |
+
},
|
| 865 |
+
"node_modules/fill-range": {
|
| 866 |
+
"version": "7.1.1",
|
| 867 |
+
"resolved": "https://registry.npmjs.org/fill-range/-/fill-range-7.1.1.tgz",
|
| 868 |
+
"integrity": "sha512-YsGpe3WHLK8ZYi4tWDg2Jy3ebRz2rXowDxnld4bkQB00cc/1Zw9AWnC0i9ztDJitivtQvaI9KaLyKrc+hBW0yg==",
|
| 869 |
+
"license": "MIT",
|
| 870 |
+
"dependencies": {
|
| 871 |
+
"to-regex-range": "^5.0.1"
|
| 872 |
+
},
|
| 873 |
+
"engines": {
|
| 874 |
+
"node": ">=8"
|
| 875 |
+
}
|
| 876 |
+
},
|
| 877 |
+
"node_modules/fraction.js": {
|
| 878 |
+
"version": "5.3.4",
|
| 879 |
+
"resolved": "https://registry.npmjs.org/fraction.js/-/fraction.js-5.3.4.tgz",
|
| 880 |
+
"integrity": "sha512-1X1NTtiJphryn/uLQz3whtY6jK3fTqoE3ohKs0tT+Ujr1W59oopxmoEh7Lu5p6vBaPbgoM0bzveAW4Qi5RyWDQ==",
|
| 881 |
+
"license": "MIT",
|
| 882 |
+
"engines": {
|
| 883 |
+
"node": "*"
|
| 884 |
+
},
|
| 885 |
+
"funding": {
|
| 886 |
+
"type": "github",
|
| 887 |
+
"url": "https://github.com/sponsors/rawify"
|
| 888 |
+
}
|
| 889 |
+
},
|
| 890 |
+
"node_modules/fsevents": {
|
| 891 |
+
"version": "2.3.3",
|
| 892 |
+
"resolved": "https://registry.npmjs.org/fsevents/-/fsevents-2.3.3.tgz",
|
| 893 |
+
"integrity": "sha512-5xoDfX+fL7faATnagmWPpbFtwh/R77WmMMqqHGS65C3vvB0YHrgF+B1YmZ3441tMj5n63k0212XNoJwzlhffQw==",
|
| 894 |
+
"hasInstallScript": true,
|
| 895 |
+
"license": "MIT",
|
| 896 |
+
"optional": true,
|
| 897 |
+
"os": [
|
| 898 |
+
"darwin"
|
| 899 |
+
],
|
| 900 |
+
"engines": {
|
| 901 |
+
"node": "^8.16.0 || ^10.6.0 || >=11.0.0"
|
| 902 |
+
}
|
| 903 |
+
},
|
| 904 |
+
"node_modules/function-bind": {
|
| 905 |
+
"version": "1.1.2",
|
| 906 |
+
"resolved": "https://registry.npmjs.org/function-bind/-/function-bind-1.1.2.tgz",
|
| 907 |
+
"integrity": "sha512-7XHNxH7qX9xG5mIwxkhumTox/MIRNcOgDrxWsMt2pAr23WHp6MrRlN7FBSFpCpr+oVO0F744iUgR82nJMfG2SA==",
|
| 908 |
+
"license": "MIT",
|
| 909 |
+
"funding": {
|
| 910 |
+
"url": "https://github.com/sponsors/ljharb"
|
| 911 |
+
}
|
| 912 |
+
},
|
| 913 |
+
"node_modules/glob-parent": {
|
| 914 |
+
"version": "6.0.2",
|
| 915 |
+
"resolved": "https://registry.npmjs.org/glob-parent/-/glob-parent-6.0.2.tgz",
|
| 916 |
+
"integrity": "sha512-XxwI8EOhVQgWp6iDL+3b0r86f4d6AX6zSU55HfB4ydCEuXLXc5FcYeOu+nnGftS4TEju/11rt4KJPTMgbfmv4A==",
|
| 917 |
+
"license": "ISC",
|
| 918 |
+
"dependencies": {
|
| 919 |
+
"is-glob": "^4.0.3"
|
| 920 |
+
},
|
| 921 |
+
"engines": {
|
| 922 |
+
"node": ">=10.13.0"
|
| 923 |
+
}
|
| 924 |
+
},
|
| 925 |
+
"node_modules/graceful-fs": {
|
| 926 |
+
"version": "4.2.11",
|
| 927 |
+
"resolved": "https://registry.npmjs.org/graceful-fs/-/graceful-fs-4.2.11.tgz",
|
| 928 |
+
"integrity": "sha512-RbJ5/jmFcNNCcDV5o9eTnBLJ/HszWV0P73bc+Ff4nS/rJj+YaS6IGyiOL0VoBYX+l1Wrl3k63h/KrH+nhJ0XvQ==",
|
| 929 |
+
"license": "ISC"
|
| 930 |
+
},
|
| 931 |
+
"node_modules/hasown": {
|
| 932 |
+
"version": "2.0.3",
|
| 933 |
+
"resolved": "https://registry.npmjs.org/hasown/-/hasown-2.0.3.tgz",
|
| 934 |
+
"integrity": "sha512-ej4AhfhfL2Q2zpMmLo7U1Uv9+PyhIZpgQLGT1F9miIGmiCJIoCgSmczFdrc97mWT4kVY72KA+WnnhJ5pghSvSg==",
|
| 935 |
+
"license": "MIT",
|
| 936 |
+
"dependencies": {
|
| 937 |
+
"function-bind": "^1.1.2"
|
| 938 |
+
},
|
| 939 |
+
"engines": {
|
| 940 |
+
"node": ">= 0.4"
|
| 941 |
+
}
|
| 942 |
+
},
|
| 943 |
+
"node_modules/internmap": {
|
| 944 |
+
"version": "2.0.3",
|
| 945 |
+
"resolved": "https://registry.npmjs.org/internmap/-/internmap-2.0.3.tgz",
|
| 946 |
+
"integrity": "sha512-5Hh7Y1wQbvY5ooGgPbDaL5iYLAPzMTUrjMulskHLH6wnv/A+1q5rgEaiuqEjB+oxGXIVZs1FF+R/KPN3ZSQYYg==",
|
| 947 |
+
"license": "ISC",
|
| 948 |
+
"engines": {
|
| 949 |
+
"node": ">=12"
|
| 950 |
+
}
|
| 951 |
+
},
|
| 952 |
+
"node_modules/is-binary-path": {
|
| 953 |
+
"version": "2.1.0",
|
| 954 |
+
"resolved": "https://registry.npmjs.org/is-binary-path/-/is-binary-path-2.1.0.tgz",
|
| 955 |
+
"integrity": "sha512-ZMERYes6pDydyuGidse7OsHxtbI7WVeUEozgR/g7rd0xUimYNlvZRE/K2MgZTjWy725IfelLeVcEM97mmtRGXw==",
|
| 956 |
+
"license": "MIT",
|
| 957 |
+
"dependencies": {
|
| 958 |
+
"binary-extensions": "^2.0.0"
|
| 959 |
+
},
|
| 960 |
+
"engines": {
|
| 961 |
+
"node": ">=8"
|
| 962 |
+
}
|
| 963 |
+
},
|
| 964 |
+
"node_modules/is-core-module": {
|
| 965 |
+
"version": "2.16.1",
|
| 966 |
+
"resolved": "https://registry.npmjs.org/is-core-module/-/is-core-module-2.16.1.tgz",
|
| 967 |
+
"integrity": "sha512-UfoeMA6fIJ8wTYFEUjelnaGI67v6+N7qXJEvQuIGa99l4xsCruSYOVSQ0uPANn4dAzm8lkYPaKLrrijLq7x23w==",
|
| 968 |
+
"license": "MIT",
|
| 969 |
+
"dependencies": {
|
| 970 |
+
"hasown": "^2.0.2"
|
| 971 |
+
},
|
| 972 |
+
"engines": {
|
| 973 |
+
"node": ">= 0.4"
|
| 974 |
+
},
|
| 975 |
+
"funding": {
|
| 976 |
+
"url": "https://github.com/sponsors/ljharb"
|
| 977 |
+
}
|
| 978 |
+
},
|
| 979 |
+
"node_modules/is-extglob": {
|
| 980 |
+
"version": "2.1.1",
|
| 981 |
+
"resolved": "https://registry.npmjs.org/is-extglob/-/is-extglob-2.1.1.tgz",
|
| 982 |
+
"integrity": "sha512-SbKbANkN603Vi4jEZv49LeVJMn4yGwsbzZworEoyEiutsN3nJYdbO36zfhGJ6QEDpOZIFkDtnq5JRxmvl3jsoQ==",
|
| 983 |
+
"license": "MIT",
|
| 984 |
+
"engines": {
|
| 985 |
+
"node": ">=0.10.0"
|
| 986 |
+
}
|
| 987 |
+
},
|
| 988 |
+
"node_modules/is-glob": {
|
| 989 |
+
"version": "4.0.3",
|
| 990 |
+
"resolved": "https://registry.npmjs.org/is-glob/-/is-glob-4.0.3.tgz",
|
| 991 |
+
"integrity": "sha512-xelSayHH36ZgE7ZWhli7pW34hNbNl8Ojv5KVmkJD4hBdD3th8Tfk9vYasLM+mXWOZhFkgZfxhLSnrwRr4elSSg==",
|
| 992 |
+
"license": "MIT",
|
| 993 |
+
"dependencies": {
|
| 994 |
+
"is-extglob": "^2.1.1"
|
| 995 |
+
},
|
| 996 |
+
"engines": {
|
| 997 |
+
"node": ">=0.10.0"
|
| 998 |
+
}
|
| 999 |
+
},
|
| 1000 |
+
"node_modules/is-number": {
|
| 1001 |
+
"version": "7.0.0",
|
| 1002 |
+
"resolved": "https://registry.npmjs.org/is-number/-/is-number-7.0.0.tgz",
|
| 1003 |
+
"integrity": "sha512-41Cifkg6e8TylSpdtTpeLVMqvSBEVzTttHvERD741+pnZ8ANv0004MRL43QKPDlK9cGvNp6NZWZUBlbGXYxxng==",
|
| 1004 |
+
"license": "MIT",
|
| 1005 |
+
"engines": {
|
| 1006 |
+
"node": ">=0.12.0"
|
| 1007 |
+
}
|
| 1008 |
+
},
|
| 1009 |
+
"node_modules/jiti": {
|
| 1010 |
+
"version": "1.21.7",
|
| 1011 |
+
"resolved": "https://registry.npmjs.org/jiti/-/jiti-1.21.7.tgz",
|
| 1012 |
+
"integrity": "sha512-/imKNG4EbWNrVjoNC/1H5/9GFy+tqjGBHCaSsN+P2RnPqjsLmv6UD3Ej+Kj8nBWaRAwyk7kK5ZUc+OEatnTR3A==",
|
| 1013 |
+
"license": "MIT",
|
| 1014 |
+
"bin": {
|
| 1015 |
+
"jiti": "bin/jiti.js"
|
| 1016 |
+
}
|
| 1017 |
+
},
|
| 1018 |
+
"node_modules/js-tokens": {
|
| 1019 |
+
"version": "4.0.0",
|
| 1020 |
+
"resolved": "https://registry.npmjs.org/js-tokens/-/js-tokens-4.0.0.tgz",
|
| 1021 |
+
"integrity": "sha512-RdJUflcE3cUzKiMqQgsCu06FPu9UdIJO0beYbPhHN4k6apgJtifcoCtT9bcxOpYBtpD2kCM6Sbzg4CausW/PKQ==",
|
| 1022 |
+
"license": "MIT"
|
| 1023 |
+
},
|
| 1024 |
+
"node_modules/lilconfig": {
|
| 1025 |
+
"version": "3.1.3",
|
| 1026 |
+
"resolved": "https://registry.npmjs.org/lilconfig/-/lilconfig-3.1.3.tgz",
|
| 1027 |
+
"integrity": "sha512-/vlFKAoH5Cgt3Ie+JLhRbwOsCQePABiU3tJ1egGvyQ+33R/vcwM2Zl2QR/LzjsBeItPt3oSVXapn+m4nQDvpzw==",
|
| 1028 |
+
"license": "MIT",
|
| 1029 |
+
"engines": {
|
| 1030 |
+
"node": ">=14"
|
| 1031 |
+
},
|
| 1032 |
+
"funding": {
|
| 1033 |
+
"url": "https://github.com/sponsors/antonk52"
|
| 1034 |
+
}
|
| 1035 |
+
},
|
| 1036 |
+
"node_modules/lines-and-columns": {
|
| 1037 |
+
"version": "1.2.4",
|
| 1038 |
+
"resolved": "https://registry.npmjs.org/lines-and-columns/-/lines-and-columns-1.2.4.tgz",
|
| 1039 |
+
"integrity": "sha512-7ylylesZQ/PV29jhEDl3Ufjo6ZX7gCqJr5F7PKrqc93v7fzSymt1BpwEU8nAUXs8qzzvqhbjhK5QZg6Mt/HkBg==",
|
| 1040 |
+
"license": "MIT"
|
| 1041 |
+
},
|
| 1042 |
+
"node_modules/lodash": {
|
| 1043 |
+
"version": "4.18.1",
|
| 1044 |
+
"resolved": "https://registry.npmjs.org/lodash/-/lodash-4.18.1.tgz",
|
| 1045 |
+
"integrity": "sha512-dMInicTPVE8d1e5otfwmmjlxkZoUpiVLwyeTdUsi/Caj/gfzzblBcCE5sRHV/AsjuCmxWrte2TNGSYuCeCq+0Q==",
|
| 1046 |
+
"license": "MIT"
|
| 1047 |
+
},
|
| 1048 |
+
"node_modules/loose-envify": {
|
| 1049 |
+
"version": "1.4.0",
|
| 1050 |
+
"resolved": "https://registry.npmjs.org/loose-envify/-/loose-envify-1.4.0.tgz",
|
| 1051 |
+
"integrity": "sha512-lyuxPGr/Wfhrlem2CL/UcnUc1zcqKAImBDzukY7Y5F/yQiNdko6+fRLevlw1HgMySw7f611UIY408EtxRSoK3Q==",
|
| 1052 |
+
"license": "MIT",
|
| 1053 |
+
"dependencies": {
|
| 1054 |
+
"js-tokens": "^3.0.0 || ^4.0.0"
|
| 1055 |
+
},
|
| 1056 |
+
"bin": {
|
| 1057 |
+
"loose-envify": "cli.js"
|
| 1058 |
+
}
|
| 1059 |
+
},
|
| 1060 |
+
"node_modules/lucide-react": {
|
| 1061 |
+
"version": "1.14.0",
|
| 1062 |
+
"resolved": "https://registry.npmjs.org/lucide-react/-/lucide-react-1.14.0.tgz",
|
| 1063 |
+
"integrity": "sha512-+1mdWcfSJVUsaTIjN9zoezmUhfXo5l0vP7ekBMPo3jcS/aIkxHnXqAPsByszMZx/Y8oQBRJxJx5xg+RH3urzxA==",
|
| 1064 |
+
"license": "ISC",
|
| 1065 |
+
"peerDependencies": {
|
| 1066 |
+
"react": "^16.5.1 || ^17.0.0 || ^18.0.0 || ^19.0.0"
|
| 1067 |
+
}
|
| 1068 |
+
},
|
| 1069 |
+
"node_modules/merge2": {
|
| 1070 |
+
"version": "1.4.1",
|
| 1071 |
+
"resolved": "https://registry.npmjs.org/merge2/-/merge2-1.4.1.tgz",
|
| 1072 |
+
"integrity": "sha512-8q7VEgMJW4J8tcfVPy8g09NcQwZdbwFEqhe/WZkoIzjn/3TGDwtOCYtXGxA3O8tPzpczCCDgv+P2P5y00ZJOOg==",
|
| 1073 |
+
"license": "MIT",
|
| 1074 |
+
"engines": {
|
| 1075 |
+
"node": ">= 8"
|
| 1076 |
+
}
|
| 1077 |
+
},
|
| 1078 |
+
"node_modules/micromatch": {
|
| 1079 |
+
"version": "4.0.8",
|
| 1080 |
+
"resolved": "https://registry.npmjs.org/micromatch/-/micromatch-4.0.8.tgz",
|
| 1081 |
+
"integrity": "sha512-PXwfBhYu0hBCPw8Dn0E+WDYb7af3dSLVWKi3HGv84IdF4TyFoC0ysxFd0Goxw7nSv4T/PzEJQxsYsEiFCKo2BA==",
|
| 1082 |
+
"license": "MIT",
|
| 1083 |
+
"dependencies": {
|
| 1084 |
+
"braces": "^3.0.3",
|
| 1085 |
+
"picomatch": "^2.3.1"
|
| 1086 |
+
},
|
| 1087 |
+
"engines": {
|
| 1088 |
+
"node": ">=8.6"
|
| 1089 |
+
}
|
| 1090 |
+
},
|
| 1091 |
+
"node_modules/mz": {
|
| 1092 |
+
"version": "2.7.0",
|
| 1093 |
+
"resolved": "https://registry.npmjs.org/mz/-/mz-2.7.0.tgz",
|
| 1094 |
+
"integrity": "sha512-z81GNO7nnYMEhrGh9LeymoE4+Yr0Wn5McHIZMK5cfQCl+NDX08sCZgUc9/6MHni9IWuFLm1Z3HTCXu2z9fN62Q==",
|
| 1095 |
+
"license": "MIT",
|
| 1096 |
+
"dependencies": {
|
| 1097 |
+
"any-promise": "^1.0.0",
|
| 1098 |
+
"object-assign": "^4.0.1",
|
| 1099 |
+
"thenify-all": "^1.0.0"
|
| 1100 |
+
}
|
| 1101 |
+
},
|
| 1102 |
+
"node_modules/nanoid": {
|
| 1103 |
+
"version": "3.3.12",
|
| 1104 |
+
"resolved": "https://registry.npmjs.org/nanoid/-/nanoid-3.3.12.tgz",
|
| 1105 |
+
"integrity": "sha512-ZB9RH/39qpq5Vu6Y+NmUaFhQR6pp+M2Xt76XBnEwDaGcVAqhlvxrl3B2bKS5D3NH3QR76v3aSrKaF/Kiy7lEtQ==",
|
| 1106 |
+
"funding": [
|
| 1107 |
+
{
|
| 1108 |
+
"type": "github",
|
| 1109 |
+
"url": "https://github.com/sponsors/ai"
|
| 1110 |
+
}
|
| 1111 |
+
],
|
| 1112 |
+
"license": "MIT",
|
| 1113 |
+
"bin": {
|
| 1114 |
+
"nanoid": "bin/nanoid.cjs"
|
| 1115 |
+
},
|
| 1116 |
+
"engines": {
|
| 1117 |
+
"node": "^10 || ^12 || ^13.7 || ^14 || >=15.0.1"
|
| 1118 |
+
}
|
| 1119 |
+
},
|
| 1120 |
+
"node_modules/next": {
|
| 1121 |
+
"version": "14.2.35",
|
| 1122 |
+
"resolved": "https://registry.npmjs.org/next/-/next-14.2.35.tgz",
|
| 1123 |
+
"integrity": "sha512-KhYd2Hjt/O1/1aZVX3dCwGXM1QmOV4eNM2UTacK5gipDdPN/oHHK/4oVGy7X8GMfPMsUTUEmGlsy0EY1YGAkig==",
|
| 1124 |
+
"license": "MIT",
|
| 1125 |
+
"dependencies": {
|
| 1126 |
+
"@next/env": "14.2.35",
|
| 1127 |
+
"@swc/helpers": "0.5.5",
|
| 1128 |
+
"busboy": "1.6.0",
|
| 1129 |
+
"caniuse-lite": "^1.0.30001579",
|
| 1130 |
+
"graceful-fs": "^4.2.11",
|
| 1131 |
+
"postcss": "8.4.31",
|
| 1132 |
+
"styled-jsx": "5.1.1"
|
| 1133 |
+
},
|
| 1134 |
+
"bin": {
|
| 1135 |
+
"next": "dist/bin/next"
|
| 1136 |
+
},
|
| 1137 |
+
"engines": {
|
| 1138 |
+
"node": ">=18.17.0"
|
| 1139 |
+
},
|
| 1140 |
+
"optionalDependencies": {
|
| 1141 |
+
"@next/swc-darwin-arm64": "14.2.33",
|
| 1142 |
+
"@next/swc-darwin-x64": "14.2.33",
|
| 1143 |
+
"@next/swc-linux-arm64-gnu": "14.2.33",
|
| 1144 |
+
"@next/swc-linux-arm64-musl": "14.2.33",
|
| 1145 |
+
"@next/swc-linux-x64-gnu": "14.2.33",
|
| 1146 |
+
"@next/swc-linux-x64-musl": "14.2.33",
|
| 1147 |
+
"@next/swc-win32-arm64-msvc": "14.2.33",
|
| 1148 |
+
"@next/swc-win32-ia32-msvc": "14.2.33",
|
| 1149 |
+
"@next/swc-win32-x64-msvc": "14.2.33"
|
| 1150 |
+
},
|
| 1151 |
+
"peerDependencies": {
|
| 1152 |
+
"@opentelemetry/api": "^1.1.0",
|
| 1153 |
+
"@playwright/test": "^1.41.2",
|
| 1154 |
+
"react": "^18.2.0",
|
| 1155 |
+
"react-dom": "^18.2.0",
|
| 1156 |
+
"sass": "^1.3.0"
|
| 1157 |
+
},
|
| 1158 |
+
"peerDependenciesMeta": {
|
| 1159 |
+
"@opentelemetry/api": {
|
| 1160 |
+
"optional": true
|
| 1161 |
+
},
|
| 1162 |
+
"@playwright/test": {
|
| 1163 |
+
"optional": true
|
| 1164 |
+
},
|
| 1165 |
+
"sass": {
|
| 1166 |
+
"optional": true
|
| 1167 |
+
}
|
| 1168 |
+
}
|
| 1169 |
+
},
|
| 1170 |
+
"node_modules/next/node_modules/postcss": {
|
| 1171 |
+
"version": "8.4.31",
|
| 1172 |
+
"resolved": "https://registry.npmjs.org/postcss/-/postcss-8.4.31.tgz",
|
| 1173 |
+
"integrity": "sha512-PS08Iboia9mts/2ygV3eLpY5ghnUcfLV/EXTOW1E2qYxJKGGBUtNjN76FYHnMs36RmARn41bC0AZmn+rR0OVpQ==",
|
| 1174 |
+
"funding": [
|
| 1175 |
+
{
|
| 1176 |
+
"type": "opencollective",
|
| 1177 |
+
"url": "https://opencollective.com/postcss/"
|
| 1178 |
+
},
|
| 1179 |
+
{
|
| 1180 |
+
"type": "tidelift",
|
| 1181 |
+
"url": "https://tidelift.com/funding/github/npm/postcss"
|
| 1182 |
+
},
|
| 1183 |
+
{
|
| 1184 |
+
"type": "github",
|
| 1185 |
+
"url": "https://github.com/sponsors/ai"
|
| 1186 |
+
}
|
| 1187 |
+
],
|
| 1188 |
+
"license": "MIT",
|
| 1189 |
+
"dependencies": {
|
| 1190 |
+
"nanoid": "^3.3.6",
|
| 1191 |
+
"picocolors": "^1.0.0",
|
| 1192 |
+
"source-map-js": "^1.0.2"
|
| 1193 |
+
},
|
| 1194 |
+
"engines": {
|
| 1195 |
+
"node": "^10 || ^12 || >=14"
|
| 1196 |
+
}
|
| 1197 |
+
},
|
| 1198 |
+
"node_modules/node-releases": {
|
| 1199 |
+
"version": "2.0.38",
|
| 1200 |
+
"resolved": "https://registry.npmjs.org/node-releases/-/node-releases-2.0.38.tgz",
|
| 1201 |
+
"integrity": "sha512-3qT/88Y3FbH/Kx4szpQQ4HzUbVrHPKTLVpVocKiLfoYvw9XSGOX2FmD2d6DrXbVYyAQTF2HeF6My8jmzx7/CRw==",
|
| 1202 |
+
"license": "MIT"
|
| 1203 |
+
},
|
| 1204 |
+
"node_modules/normalize-path": {
|
| 1205 |
+
"version": "3.0.0",
|
| 1206 |
+
"resolved": "https://registry.npmjs.org/normalize-path/-/normalize-path-3.0.0.tgz",
|
| 1207 |
+
"integrity": "sha512-6eZs5Ls3WtCisHWp9S2GUy8dqkpGi4BVSz3GaqiE6ezub0512ESztXUwUB6C6IKbQkY2Pnb/mD4WYojCRwcwLA==",
|
| 1208 |
+
"license": "MIT",
|
| 1209 |
+
"engines": {
|
| 1210 |
+
"node": ">=0.10.0"
|
| 1211 |
+
}
|
| 1212 |
+
},
|
| 1213 |
+
"node_modules/object-assign": {
|
| 1214 |
+
"version": "4.1.1",
|
| 1215 |
+
"resolved": "https://registry.npmjs.org/object-assign/-/object-assign-4.1.1.tgz",
|
| 1216 |
+
"integrity": "sha512-rJgTQnkUnH1sFw8yT6VSU3zD3sWmu6sZhIseY8VX+GRu3P6F7Fu+JNDoXfklElbLJSnc3FUQHVe4cU5hj+BcUg==",
|
| 1217 |
+
"license": "MIT",
|
| 1218 |
+
"engines": {
|
| 1219 |
+
"node": ">=0.10.0"
|
| 1220 |
+
}
|
| 1221 |
+
},
|
| 1222 |
+
"node_modules/object-hash": {
|
| 1223 |
+
"version": "3.0.0",
|
| 1224 |
+
"resolved": "https://registry.npmjs.org/object-hash/-/object-hash-3.0.0.tgz",
|
| 1225 |
+
"integrity": "sha512-RSn9F68PjH9HqtltsSnqYC1XXoWe9Bju5+213R98cNGttag9q9yAOTzdbsqvIa7aNm5WffBZFpWYr2aWrklWAw==",
|
| 1226 |
+
"license": "MIT",
|
| 1227 |
+
"engines": {
|
| 1228 |
+
"node": ">= 6"
|
| 1229 |
+
}
|
| 1230 |
+
},
|
| 1231 |
+
"node_modules/path-parse": {
|
| 1232 |
+
"version": "1.0.7",
|
| 1233 |
+
"resolved": "https://registry.npmjs.org/path-parse/-/path-parse-1.0.7.tgz",
|
| 1234 |
+
"integrity": "sha512-LDJzPVEEEPR+y48z93A0Ed0yXb8pAByGWo/k5YYdYgpY2/2EsOsksJrq7lOHxryrVOn1ejG6oAp8ahvOIQD8sw==",
|
| 1235 |
+
"license": "MIT"
|
| 1236 |
+
},
|
| 1237 |
+
"node_modules/picocolors": {
|
| 1238 |
+
"version": "1.1.1",
|
| 1239 |
+
"resolved": "https://registry.npmjs.org/picocolors/-/picocolors-1.1.1.tgz",
|
| 1240 |
+
"integrity": "sha512-xceH2snhtb5M9liqDsmEw56le376mTZkEX/jEb/RxNFyegNul7eNslCXP9FDj/Lcu0X8KEyMceP2ntpaHrDEVA==",
|
| 1241 |
+
"license": "ISC"
|
| 1242 |
+
},
|
| 1243 |
+
"node_modules/picomatch": {
|
| 1244 |
+
"version": "2.3.2",
|
| 1245 |
+
"resolved": "https://registry.npmjs.org/picomatch/-/picomatch-2.3.2.tgz",
|
| 1246 |
+
"integrity": "sha512-V7+vQEJ06Z+c5tSye8S+nHUfI51xoXIXjHQ99cQtKUkQqqO1kO/KCJUfZXuB47h/YBlDhah2H3hdUGXn8ie0oA==",
|
| 1247 |
+
"license": "MIT",
|
| 1248 |
+
"engines": {
|
| 1249 |
+
"node": ">=8.6"
|
| 1250 |
+
},
|
| 1251 |
+
"funding": {
|
| 1252 |
+
"url": "https://github.com/sponsors/jonschlinkert"
|
| 1253 |
+
}
|
| 1254 |
+
},
|
| 1255 |
+
"node_modules/pify": {
|
| 1256 |
+
"version": "2.3.0",
|
| 1257 |
+
"resolved": "https://registry.npmjs.org/pify/-/pify-2.3.0.tgz",
|
| 1258 |
+
"integrity": "sha512-udgsAY+fTnvv7kI7aaxbqwWNb0AHiB0qBO89PZKPkoTmGOgdbrHDKD+0B2X4uTfJ/FT1R09r9gTsjUjNJotuog==",
|
| 1259 |
+
"license": "MIT",
|
| 1260 |
+
"engines": {
|
| 1261 |
+
"node": ">=0.10.0"
|
| 1262 |
+
}
|
| 1263 |
+
},
|
| 1264 |
+
"node_modules/pirates": {
|
| 1265 |
+
"version": "4.0.7",
|
| 1266 |
+
"resolved": "https://registry.npmjs.org/pirates/-/pirates-4.0.7.tgz",
|
| 1267 |
+
"integrity": "sha512-TfySrs/5nm8fQJDcBDuUng3VOUKsd7S+zqvbOTiGXHfxX4wK31ard+hoNuvkicM/2YFzlpDgABOevKSsB4G/FA==",
|
| 1268 |
+
"license": "MIT",
|
| 1269 |
+
"engines": {
|
| 1270 |
+
"node": ">= 6"
|
| 1271 |
+
}
|
| 1272 |
+
},
|
| 1273 |
+
"node_modules/postcss": {
|
| 1274 |
+
"version": "8.5.13",
|
| 1275 |
+
"resolved": "https://registry.npmjs.org/postcss/-/postcss-8.5.13.tgz",
|
| 1276 |
+
"integrity": "sha512-qif0+jGGZoLWdHey3UFHHWP0H7Gbmsk8T5VEqyYFbWqPr1XqvLGBbk/sl8V5exGmcYJklJOhOQq1pV9IcsiFag==",
|
| 1277 |
+
"funding": [
|
| 1278 |
+
{
|
| 1279 |
+
"type": "opencollective",
|
| 1280 |
+
"url": "https://opencollective.com/postcss/"
|
| 1281 |
+
},
|
| 1282 |
+
{
|
| 1283 |
+
"type": "tidelift",
|
| 1284 |
+
"url": "https://tidelift.com/funding/github/npm/postcss"
|
| 1285 |
+
},
|
| 1286 |
+
{
|
| 1287 |
+
"type": "github",
|
| 1288 |
+
"url": "https://github.com/sponsors/ai"
|
| 1289 |
+
}
|
| 1290 |
+
],
|
| 1291 |
+
"license": "MIT",
|
| 1292 |
+
"dependencies": {
|
| 1293 |
+
"nanoid": "^3.3.11",
|
| 1294 |
+
"picocolors": "^1.1.1",
|
| 1295 |
+
"source-map-js": "^1.2.1"
|
| 1296 |
+
},
|
| 1297 |
+
"engines": {
|
| 1298 |
+
"node": "^10 || ^12 || >=14"
|
| 1299 |
+
}
|
| 1300 |
+
},
|
| 1301 |
+
"node_modules/postcss-import": {
|
| 1302 |
+
"version": "15.1.0",
|
| 1303 |
+
"resolved": "https://registry.npmjs.org/postcss-import/-/postcss-import-15.1.0.tgz",
|
| 1304 |
+
"integrity": "sha512-hpr+J05B2FVYUAXHeK1YyI267J/dDDhMU6B6civm8hSY1jYJnBXxzKDKDswzJmtLHryrjhnDjqqp/49t8FALew==",
|
| 1305 |
+
"license": "MIT",
|
| 1306 |
+
"dependencies": {
|
| 1307 |
+
"postcss-value-parser": "^4.0.0",
|
| 1308 |
+
"read-cache": "^1.0.0",
|
| 1309 |
+
"resolve": "^1.1.7"
|
| 1310 |
+
},
|
| 1311 |
+
"engines": {
|
| 1312 |
+
"node": ">=14.0.0"
|
| 1313 |
+
},
|
| 1314 |
+
"peerDependencies": {
|
| 1315 |
+
"postcss": "^8.0.0"
|
| 1316 |
+
}
|
| 1317 |
+
},
|
| 1318 |
+
"node_modules/postcss-js": {
|
| 1319 |
+
"version": "4.1.0",
|
| 1320 |
+
"resolved": "https://registry.npmjs.org/postcss-js/-/postcss-js-4.1.0.tgz",
|
| 1321 |
+
"integrity": "sha512-oIAOTqgIo7q2EOwbhb8UalYePMvYoIeRY2YKntdpFQXNosSu3vLrniGgmH9OKs/qAkfoj5oB3le/7mINW1LCfw==",
|
| 1322 |
+
"funding": [
|
| 1323 |
+
{
|
| 1324 |
+
"type": "opencollective",
|
| 1325 |
+
"url": "https://opencollective.com/postcss/"
|
| 1326 |
+
},
|
| 1327 |
+
{
|
| 1328 |
+
"type": "github",
|
| 1329 |
+
"url": "https://github.com/sponsors/ai"
|
| 1330 |
+
}
|
| 1331 |
+
],
|
| 1332 |
+
"license": "MIT",
|
| 1333 |
+
"dependencies": {
|
| 1334 |
+
"camelcase-css": "^2.0.1"
|
| 1335 |
+
},
|
| 1336 |
+
"engines": {
|
| 1337 |
+
"node": "^12 || ^14 || >= 16"
|
| 1338 |
+
},
|
| 1339 |
+
"peerDependencies": {
|
| 1340 |
+
"postcss": "^8.4.21"
|
| 1341 |
+
}
|
| 1342 |
+
},
|
| 1343 |
+
"node_modules/postcss-load-config": {
|
| 1344 |
+
"version": "6.0.1",
|
| 1345 |
+
"resolved": "https://registry.npmjs.org/postcss-load-config/-/postcss-load-config-6.0.1.tgz",
|
| 1346 |
+
"integrity": "sha512-oPtTM4oerL+UXmx+93ytZVN82RrlY/wPUV8IeDxFrzIjXOLF1pN+EmKPLbubvKHT2HC20xXsCAH2Z+CKV6Oz/g==",
|
| 1347 |
+
"funding": [
|
| 1348 |
+
{
|
| 1349 |
+
"type": "opencollective",
|
| 1350 |
+
"url": "https://opencollective.com/postcss/"
|
| 1351 |
+
},
|
| 1352 |
+
{
|
| 1353 |
+
"type": "github",
|
| 1354 |
+
"url": "https://github.com/sponsors/ai"
|
| 1355 |
+
}
|
| 1356 |
+
],
|
| 1357 |
+
"license": "MIT",
|
| 1358 |
+
"dependencies": {
|
| 1359 |
+
"lilconfig": "^3.1.1"
|
| 1360 |
+
},
|
| 1361 |
+
"engines": {
|
| 1362 |
+
"node": ">= 18"
|
| 1363 |
+
},
|
| 1364 |
+
"peerDependencies": {
|
| 1365 |
+
"jiti": ">=1.21.0",
|
| 1366 |
+
"postcss": ">=8.0.9",
|
| 1367 |
+
"tsx": "^4.8.1",
|
| 1368 |
+
"yaml": "^2.4.2"
|
| 1369 |
+
},
|
| 1370 |
+
"peerDependenciesMeta": {
|
| 1371 |
+
"jiti": {
|
| 1372 |
+
"optional": true
|
| 1373 |
+
},
|
| 1374 |
+
"postcss": {
|
| 1375 |
+
"optional": true
|
| 1376 |
+
},
|
| 1377 |
+
"tsx": {
|
| 1378 |
+
"optional": true
|
| 1379 |
+
},
|
| 1380 |
+
"yaml": {
|
| 1381 |
+
"optional": true
|
| 1382 |
+
}
|
| 1383 |
+
}
|
| 1384 |
+
},
|
| 1385 |
+
"node_modules/postcss-nested": {
|
| 1386 |
+
"version": "6.2.0",
|
| 1387 |
+
"resolved": "https://registry.npmjs.org/postcss-nested/-/postcss-nested-6.2.0.tgz",
|
| 1388 |
+
"integrity": "sha512-HQbt28KulC5AJzG+cZtj9kvKB93CFCdLvog1WFLf1D+xmMvPGlBstkpTEZfK5+AN9hfJocyBFCNiqyS48bpgzQ==",
|
| 1389 |
+
"funding": [
|
| 1390 |
+
{
|
| 1391 |
+
"type": "opencollective",
|
| 1392 |
+
"url": "https://opencollective.com/postcss/"
|
| 1393 |
+
},
|
| 1394 |
+
{
|
| 1395 |
+
"type": "github",
|
| 1396 |
+
"url": "https://github.com/sponsors/ai"
|
| 1397 |
+
}
|
| 1398 |
+
],
|
| 1399 |
+
"license": "MIT",
|
| 1400 |
+
"dependencies": {
|
| 1401 |
+
"postcss-selector-parser": "^6.1.1"
|
| 1402 |
+
},
|
| 1403 |
+
"engines": {
|
| 1404 |
+
"node": ">=12.0"
|
| 1405 |
+
},
|
| 1406 |
+
"peerDependencies": {
|
| 1407 |
+
"postcss": "^8.2.14"
|
| 1408 |
+
}
|
| 1409 |
+
},
|
| 1410 |
+
"node_modules/postcss-selector-parser": {
|
| 1411 |
+
"version": "6.1.2",
|
| 1412 |
+
"resolved": "https://registry.npmjs.org/postcss-selector-parser/-/postcss-selector-parser-6.1.2.tgz",
|
| 1413 |
+
"integrity": "sha512-Q8qQfPiZ+THO/3ZrOrO0cJJKfpYCagtMUkXbnEfmgUjwXg6z/WBeOyS9APBBPCTSiDV+s4SwQGu8yFsiMRIudg==",
|
| 1414 |
+
"license": "MIT",
|
| 1415 |
+
"dependencies": {
|
| 1416 |
+
"cssesc": "^3.0.0",
|
| 1417 |
+
"util-deprecate": "^1.0.2"
|
| 1418 |
+
},
|
| 1419 |
+
"engines": {
|
| 1420 |
+
"node": ">=4"
|
| 1421 |
+
}
|
| 1422 |
+
},
|
| 1423 |
+
"node_modules/postcss-value-parser": {
|
| 1424 |
+
"version": "4.2.0",
|
| 1425 |
+
"resolved": "https://registry.npmjs.org/postcss-value-parser/-/postcss-value-parser-4.2.0.tgz",
|
| 1426 |
+
"integrity": "sha512-1NNCs6uurfkVbeXG4S8JFT9t19m45ICnif8zWLd5oPSZ50QnwMfK+H3jv408d4jw/7Bttv5axS5IiHoLaVNHeQ==",
|
| 1427 |
+
"license": "MIT"
|
| 1428 |
+
},
|
| 1429 |
+
"node_modules/prop-types": {
|
| 1430 |
+
"version": "15.8.1",
|
| 1431 |
+
"resolved": "https://registry.npmjs.org/prop-types/-/prop-types-15.8.1.tgz",
|
| 1432 |
+
"integrity": "sha512-oj87CgZICdulUohogVAR7AjlC0327U4el4L6eAvOqCeudMDVU0NThNaV+b9Df4dXgSP1gXMTnPdhfe/2qDH5cg==",
|
| 1433 |
+
"license": "MIT",
|
| 1434 |
+
"dependencies": {
|
| 1435 |
+
"loose-envify": "^1.4.0",
|
| 1436 |
+
"object-assign": "^4.1.1",
|
| 1437 |
+
"react-is": "^16.13.1"
|
| 1438 |
+
}
|
| 1439 |
+
},
|
| 1440 |
+
"node_modules/prop-types/node_modules/react-is": {
|
| 1441 |
+
"version": "16.13.1",
|
| 1442 |
+
"resolved": "https://registry.npmjs.org/react-is/-/react-is-16.13.1.tgz",
|
| 1443 |
+
"integrity": "sha512-24e6ynE2H+OKt4kqsOvNd8kBpV65zoxbA4BVsEOB3ARVWQki/DHzaUoC5KuON/BiccDaCCTZBuOcfZs70kR8bQ==",
|
| 1444 |
+
"license": "MIT"
|
| 1445 |
+
},
|
| 1446 |
+
"node_modules/queue-microtask": {
|
| 1447 |
+
"version": "1.2.3",
|
| 1448 |
+
"resolved": "https://registry.npmjs.org/queue-microtask/-/queue-microtask-1.2.3.tgz",
|
| 1449 |
+
"integrity": "sha512-NuaNSa6flKT5JaSYQzJok04JzTL1CA6aGhv5rfLW3PgqA+M2ChpZQnAC8h8i4ZFkBS8X5RqkDBHA7r4hej3K9A==",
|
| 1450 |
+
"funding": [
|
| 1451 |
+
{
|
| 1452 |
+
"type": "github",
|
| 1453 |
+
"url": "https://github.com/sponsors/feross"
|
| 1454 |
+
},
|
| 1455 |
+
{
|
| 1456 |
+
"type": "patreon",
|
| 1457 |
+
"url": "https://www.patreon.com/feross"
|
| 1458 |
+
},
|
| 1459 |
+
{
|
| 1460 |
+
"type": "consulting",
|
| 1461 |
+
"url": "https://feross.org/support"
|
| 1462 |
+
}
|
| 1463 |
+
],
|
| 1464 |
+
"license": "MIT"
|
| 1465 |
+
},
|
| 1466 |
+
"node_modules/react": {
|
| 1467 |
+
"version": "18.3.1",
|
| 1468 |
+
"resolved": "https://registry.npmjs.org/react/-/react-18.3.1.tgz",
|
| 1469 |
+
"integrity": "sha512-wS+hAgJShR0KhEvPJArfuPVN1+Hz1t0Y6n5jLrGQbkb4urgPE/0Rve+1kMB1v/oWgHgm4WIcV+i7F2pTVj+2iQ==",
|
| 1470 |
+
"license": "MIT",
|
| 1471 |
+
"dependencies": {
|
| 1472 |
+
"loose-envify": "^1.1.0"
|
| 1473 |
+
},
|
| 1474 |
+
"engines": {
|
| 1475 |
+
"node": ">=0.10.0"
|
| 1476 |
+
}
|
| 1477 |
+
},
|
| 1478 |
+
"node_modules/react-dom": {
|
| 1479 |
+
"version": "18.3.1",
|
| 1480 |
+
"resolved": "https://registry.npmjs.org/react-dom/-/react-dom-18.3.1.tgz",
|
| 1481 |
+
"integrity": "sha512-5m4nQKp+rZRb09LNH59GM4BxTh9251/ylbKIbpe7TpGxfJ+9kv6BLkLBXIjjspbgbnIBNqlI23tRnTWT0snUIw==",
|
| 1482 |
+
"license": "MIT",
|
| 1483 |
+
"dependencies": {
|
| 1484 |
+
"loose-envify": "^1.1.0",
|
| 1485 |
+
"scheduler": "^0.23.2"
|
| 1486 |
+
},
|
| 1487 |
+
"peerDependencies": {
|
| 1488 |
+
"react": "^18.3.1"
|
| 1489 |
+
}
|
| 1490 |
+
},
|
| 1491 |
+
"node_modules/react-is": {
|
| 1492 |
+
"version": "18.3.1",
|
| 1493 |
+
"resolved": "https://registry.npmjs.org/react-is/-/react-is-18.3.1.tgz",
|
| 1494 |
+
"integrity": "sha512-/LLMVyas0ljjAtoYiPqYiL8VWXzUUdThrmU5+n20DZv+a+ClRoevUzw5JxU+Ieh5/c87ytoTBV9G1FiKfNJdmg==",
|
| 1495 |
+
"license": "MIT"
|
| 1496 |
+
},
|
| 1497 |
+
"node_modules/react-smooth": {
|
| 1498 |
+
"version": "4.0.4",
|
| 1499 |
+
"resolved": "https://registry.npmjs.org/react-smooth/-/react-smooth-4.0.4.tgz",
|
| 1500 |
+
"integrity": "sha512-gnGKTpYwqL0Iii09gHobNolvX4Kiq4PKx6eWBCYYix+8cdw+cGo3do906l1NBPKkSWx1DghC1dlWG9L2uGd61Q==",
|
| 1501 |
+
"license": "MIT",
|
| 1502 |
+
"dependencies": {
|
| 1503 |
+
"fast-equals": "^5.0.1",
|
| 1504 |
+
"prop-types": "^15.8.1",
|
| 1505 |
+
"react-transition-group": "^4.4.5"
|
| 1506 |
+
},
|
| 1507 |
+
"peerDependencies": {
|
| 1508 |
+
"react": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0",
|
| 1509 |
+
"react-dom": "^16.8.0 || ^17.0.0 || ^18.0.0 || ^19.0.0"
|
| 1510 |
+
}
|
| 1511 |
+
},
|
| 1512 |
+
"node_modules/react-transition-group": {
|
| 1513 |
+
"version": "4.4.5",
|
| 1514 |
+
"resolved": "https://registry.npmjs.org/react-transition-group/-/react-transition-group-4.4.5.tgz",
|
| 1515 |
+
"integrity": "sha512-pZcd1MCJoiKiBR2NRxeCRg13uCXbydPnmB4EOeRrY7480qNWO8IIgQG6zlDkm6uRMsURXPuKq0GWtiM59a5Q6g==",
|
| 1516 |
+
"license": "BSD-3-Clause",
|
| 1517 |
+
"dependencies": {
|
| 1518 |
+
"@babel/runtime": "^7.5.5",
|
| 1519 |
+
"dom-helpers": "^5.0.1",
|
| 1520 |
+
"loose-envify": "^1.4.0",
|
| 1521 |
+
"prop-types": "^15.6.2"
|
| 1522 |
+
},
|
| 1523 |
+
"peerDependencies": {
|
| 1524 |
+
"react": ">=16.6.0",
|
| 1525 |
+
"react-dom": ">=16.6.0"
|
| 1526 |
+
}
|
| 1527 |
+
},
|
| 1528 |
+
"node_modules/read-cache": {
|
| 1529 |
+
"version": "1.0.0",
|
| 1530 |
+
"resolved": "https://registry.npmjs.org/read-cache/-/read-cache-1.0.0.tgz",
|
| 1531 |
+
"integrity": "sha512-Owdv/Ft7IjOgm/i0xvNDZ1LrRANRfew4b2prF3OWMQLxLfu3bS8FVhCsrSCMK4lR56Y9ya+AThoTpDCTxCmpRA==",
|
| 1532 |
+
"license": "MIT",
|
| 1533 |
+
"dependencies": {
|
| 1534 |
+
"pify": "^2.3.0"
|
| 1535 |
+
}
|
| 1536 |
+
},
|
| 1537 |
+
"node_modules/readdirp": {
|
| 1538 |
+
"version": "3.6.0",
|
| 1539 |
+
"resolved": "https://registry.npmjs.org/readdirp/-/readdirp-3.6.0.tgz",
|
| 1540 |
+
"integrity": "sha512-hOS089on8RduqdbhvQ5Z37A0ESjsqz6qnRcffsMU3495FuTdqSm+7bhJ29JvIOsBDEEnan5DPu9t3To9VRlMzA==",
|
| 1541 |
+
"license": "MIT",
|
| 1542 |
+
"dependencies": {
|
| 1543 |
+
"picomatch": "^2.2.1"
|
| 1544 |
+
},
|
| 1545 |
+
"engines": {
|
| 1546 |
+
"node": ">=8.10.0"
|
| 1547 |
+
}
|
| 1548 |
+
},
|
| 1549 |
+
"node_modules/recharts": {
|
| 1550 |
+
"version": "2.15.4",
|
| 1551 |
+
"resolved": "https://registry.npmjs.org/recharts/-/recharts-2.15.4.tgz",
|
| 1552 |
+
"integrity": "sha512-UT/q6fwS3c1dHbXv2uFgYJ9BMFHu3fwnd7AYZaEQhXuYQ4hgsxLvsUXzGdKeZrW5xopzDCvuA2N41WJ88I7zIw==",
|
| 1553 |
+
"license": "MIT",
|
| 1554 |
+
"dependencies": {
|
| 1555 |
+
"clsx": "^2.0.0",
|
| 1556 |
+
"eventemitter3": "^4.0.1",
|
| 1557 |
+
"lodash": "^4.17.21",
|
| 1558 |
+
"react-is": "^18.3.1",
|
| 1559 |
+
"react-smooth": "^4.0.4",
|
| 1560 |
+
"recharts-scale": "^0.4.4",
|
| 1561 |
+
"tiny-invariant": "^1.3.1",
|
| 1562 |
+
"victory-vendor": "^36.6.8"
|
| 1563 |
+
},
|
| 1564 |
+
"engines": {
|
| 1565 |
+
"node": ">=14"
|
| 1566 |
+
},
|
| 1567 |
+
"peerDependencies": {
|
| 1568 |
+
"react": "^16.0.0 || ^17.0.0 || ^18.0.0 || ^19.0.0",
|
| 1569 |
+
"react-dom": "^16.0.0 || ^17.0.0 || ^18.0.0 || ^19.0.0"
|
| 1570 |
+
}
|
| 1571 |
+
},
|
| 1572 |
+
"node_modules/recharts-scale": {
|
| 1573 |
+
"version": "0.4.5",
|
| 1574 |
+
"resolved": "https://registry.npmjs.org/recharts-scale/-/recharts-scale-0.4.5.tgz",
|
| 1575 |
+
"integrity": "sha512-kivNFO+0OcUNu7jQquLXAxz1FIwZj8nrj+YkOKc5694NbjCvcT6aSZiIzNzd2Kul4o4rTto8QVR9lMNtxD4G1w==",
|
| 1576 |
+
"license": "MIT",
|
| 1577 |
+
"dependencies": {
|
| 1578 |
+
"decimal.js-light": "^2.4.1"
|
| 1579 |
+
}
|
| 1580 |
+
},
|
| 1581 |
+
"node_modules/resolve": {
|
| 1582 |
+
"version": "1.22.12",
|
| 1583 |
+
"resolved": "https://registry.npmjs.org/resolve/-/resolve-1.22.12.tgz",
|
| 1584 |
+
"integrity": "sha512-TyeJ1zif53BPfHootBGwPRYT1RUt6oGWsaQr8UyZW/eAm9bKoijtvruSDEmZHm92CwS9nj7/fWttqPCgzep8CA==",
|
| 1585 |
+
"license": "MIT",
|
| 1586 |
+
"dependencies": {
|
| 1587 |
+
"es-errors": "^1.3.0",
|
| 1588 |
+
"is-core-module": "^2.16.1",
|
| 1589 |
+
"path-parse": "^1.0.7",
|
| 1590 |
+
"supports-preserve-symlinks-flag": "^1.0.0"
|
| 1591 |
+
},
|
| 1592 |
+
"bin": {
|
| 1593 |
+
"resolve": "bin/resolve"
|
| 1594 |
+
},
|
| 1595 |
+
"engines": {
|
| 1596 |
+
"node": ">= 0.4"
|
| 1597 |
+
},
|
| 1598 |
+
"funding": {
|
| 1599 |
+
"url": "https://github.com/sponsors/ljharb"
|
| 1600 |
+
}
|
| 1601 |
+
},
|
| 1602 |
+
"node_modules/reusify": {
|
| 1603 |
+
"version": "1.1.0",
|
| 1604 |
+
"resolved": "https://registry.npmjs.org/reusify/-/reusify-1.1.0.tgz",
|
| 1605 |
+
"integrity": "sha512-g6QUff04oZpHs0eG5p83rFLhHeV00ug/Yf9nZM6fLeUrPguBTkTQOdpAWWspMh55TZfVQDPaN3NQJfbVRAxdIw==",
|
| 1606 |
+
"license": "MIT",
|
| 1607 |
+
"engines": {
|
| 1608 |
+
"iojs": ">=1.0.0",
|
| 1609 |
+
"node": ">=0.10.0"
|
| 1610 |
+
}
|
| 1611 |
+
},
|
| 1612 |
+
"node_modules/run-parallel": {
|
| 1613 |
+
"version": "1.2.0",
|
| 1614 |
+
"resolved": "https://registry.npmjs.org/run-parallel/-/run-parallel-1.2.0.tgz",
|
| 1615 |
+
"integrity": "sha512-5l4VyZR86LZ/lDxZTR6jqL8AFE2S0IFLMP26AbjsLVADxHdhB/c0GUsH+y39UfCi3dzz8OlQuPmnaJOMoDHQBA==",
|
| 1616 |
+
"funding": [
|
| 1617 |
+
{
|
| 1618 |
+
"type": "github",
|
| 1619 |
+
"url": "https://github.com/sponsors/feross"
|
| 1620 |
+
},
|
| 1621 |
+
{
|
| 1622 |
+
"type": "patreon",
|
| 1623 |
+
"url": "https://www.patreon.com/feross"
|
| 1624 |
+
},
|
| 1625 |
+
{
|
| 1626 |
+
"type": "consulting",
|
| 1627 |
+
"url": "https://feross.org/support"
|
| 1628 |
+
}
|
| 1629 |
+
],
|
| 1630 |
+
"license": "MIT",
|
| 1631 |
+
"dependencies": {
|
| 1632 |
+
"queue-microtask": "^1.2.2"
|
| 1633 |
+
}
|
| 1634 |
+
},
|
| 1635 |
+
"node_modules/scheduler": {
|
| 1636 |
+
"version": "0.23.2",
|
| 1637 |
+
"resolved": "https://registry.npmjs.org/scheduler/-/scheduler-0.23.2.tgz",
|
| 1638 |
+
"integrity": "sha512-UOShsPwz7NrMUqhR6t0hWjFduvOzbtv7toDH1/hIrfRNIDBnnBWd0CwJTGvTpngVlmwGCdP9/Zl/tVrDqcuYzQ==",
|
| 1639 |
+
"license": "MIT",
|
| 1640 |
+
"dependencies": {
|
| 1641 |
+
"loose-envify": "^1.1.0"
|
| 1642 |
+
}
|
| 1643 |
+
},
|
| 1644 |
+
"node_modules/source-map-js": {
|
| 1645 |
+
"version": "1.2.1",
|
| 1646 |
+
"resolved": "https://registry.npmjs.org/source-map-js/-/source-map-js-1.2.1.tgz",
|
| 1647 |
+
"integrity": "sha512-UXWMKhLOwVKb728IUtQPXxfYU+usdybtUrK/8uGE8CQMvrhOpwvzDBwj0QhSL7MQc7vIsISBG8VQ8+IDQxpfQA==",
|
| 1648 |
+
"license": "BSD-3-Clause",
|
| 1649 |
+
"engines": {
|
| 1650 |
+
"node": ">=0.10.0"
|
| 1651 |
+
}
|
| 1652 |
+
},
|
| 1653 |
+
"node_modules/streamsearch": {
|
| 1654 |
+
"version": "1.1.0",
|
| 1655 |
+
"resolved": "https://registry.npmjs.org/streamsearch/-/streamsearch-1.1.0.tgz",
|
| 1656 |
+
"integrity": "sha512-Mcc5wHehp9aXz1ax6bZUyY5afg9u2rv5cqQI3mRrYkGC8rW2hM02jWuwjtL++LS5qinSyhj2QfLyNsuc+VsExg==",
|
| 1657 |
+
"engines": {
|
| 1658 |
+
"node": ">=10.0.0"
|
| 1659 |
+
}
|
| 1660 |
+
},
|
| 1661 |
+
"node_modules/styled-jsx": {
|
| 1662 |
+
"version": "5.1.1",
|
| 1663 |
+
"resolved": "https://registry.npmjs.org/styled-jsx/-/styled-jsx-5.1.1.tgz",
|
| 1664 |
+
"integrity": "sha512-pW7uC1l4mBZ8ugbiZrcIsiIvVx1UmTfw7UkC3Um2tmfUq9Bhk8IiyEIPl6F8agHgjzku6j0xQEZbfA5uSgSaCw==",
|
| 1665 |
+
"license": "MIT",
|
| 1666 |
+
"dependencies": {
|
| 1667 |
+
"client-only": "0.0.1"
|
| 1668 |
+
},
|
| 1669 |
+
"engines": {
|
| 1670 |
+
"node": ">= 12.0.0"
|
| 1671 |
+
},
|
| 1672 |
+
"peerDependencies": {
|
| 1673 |
+
"react": ">= 16.8.0 || 17.x.x || ^18.0.0-0"
|
| 1674 |
+
},
|
| 1675 |
+
"peerDependenciesMeta": {
|
| 1676 |
+
"@babel/core": {
|
| 1677 |
+
"optional": true
|
| 1678 |
+
},
|
| 1679 |
+
"babel-plugin-macros": {
|
| 1680 |
+
"optional": true
|
| 1681 |
+
}
|
| 1682 |
+
}
|
| 1683 |
+
},
|
| 1684 |
+
"node_modules/sucrase": {
|
| 1685 |
+
"version": "3.35.1",
|
| 1686 |
+
"resolved": "https://registry.npmjs.org/sucrase/-/sucrase-3.35.1.tgz",
|
| 1687 |
+
"integrity": "sha512-DhuTmvZWux4H1UOnWMB3sk0sbaCVOoQZjv8u1rDoTV0HTdGem9hkAZtl4JZy8P2z4Bg0nT+YMeOFyVr4zcG5Tw==",
|
| 1688 |
+
"license": "MIT",
|
| 1689 |
+
"dependencies": {
|
| 1690 |
+
"@jridgewell/gen-mapping": "^0.3.2",
|
| 1691 |
+
"commander": "^4.0.0",
|
| 1692 |
+
"lines-and-columns": "^1.1.6",
|
| 1693 |
+
"mz": "^2.7.0",
|
| 1694 |
+
"pirates": "^4.0.1",
|
| 1695 |
+
"tinyglobby": "^0.2.11",
|
| 1696 |
+
"ts-interface-checker": "^0.1.9"
|
| 1697 |
+
},
|
| 1698 |
+
"bin": {
|
| 1699 |
+
"sucrase": "bin/sucrase",
|
| 1700 |
+
"sucrase-node": "bin/sucrase-node"
|
| 1701 |
+
},
|
| 1702 |
+
"engines": {
|
| 1703 |
+
"node": ">=16 || 14 >=14.17"
|
| 1704 |
+
}
|
| 1705 |
+
},
|
| 1706 |
+
"node_modules/supports-preserve-symlinks-flag": {
|
| 1707 |
+
"version": "1.0.0",
|
| 1708 |
+
"resolved": "https://registry.npmjs.org/supports-preserve-symlinks-flag/-/supports-preserve-symlinks-flag-1.0.0.tgz",
|
| 1709 |
+
"integrity": "sha512-ot0WnXS9fgdkgIcePe6RHNk1WA8+muPa6cSjeR3V8K27q9BB1rTE3R1p7Hv0z1ZyAc8s6Vvv8DIyWf681MAt0w==",
|
| 1710 |
+
"license": "MIT",
|
| 1711 |
+
"engines": {
|
| 1712 |
+
"node": ">= 0.4"
|
| 1713 |
+
},
|
| 1714 |
+
"funding": {
|
| 1715 |
+
"url": "https://github.com/sponsors/ljharb"
|
| 1716 |
+
}
|
| 1717 |
+
},
|
| 1718 |
+
"node_modules/tailwind-merge": {
|
| 1719 |
+
"version": "2.6.1",
|
| 1720 |
+
"resolved": "https://registry.npmjs.org/tailwind-merge/-/tailwind-merge-2.6.1.tgz",
|
| 1721 |
+
"integrity": "sha512-Oo6tHdpZsGpkKG88HJ8RR1rg/RdnEkQEfMoEk2x1XRI3F1AxeU+ijRXpiVUF4UbLfcxxRGw6TbUINKYdWVsQTQ==",
|
| 1722 |
+
"license": "MIT",
|
| 1723 |
+
"funding": {
|
| 1724 |
+
"type": "github",
|
| 1725 |
+
"url": "https://github.com/sponsors/dcastil"
|
| 1726 |
+
}
|
| 1727 |
+
},
|
| 1728 |
+
"node_modules/tailwindcss": {
|
| 1729 |
+
"version": "3.4.19",
|
| 1730 |
+
"resolved": "https://registry.npmjs.org/tailwindcss/-/tailwindcss-3.4.19.tgz",
|
| 1731 |
+
"integrity": "sha512-3ofp+LL8E+pK/JuPLPggVAIaEuhvIz4qNcf3nA1Xn2o/7fb7s/TYpHhwGDv1ZU3PkBluUVaF8PyCHcm48cKLWQ==",
|
| 1732 |
+
"license": "MIT",
|
| 1733 |
+
"dependencies": {
|
| 1734 |
+
"@alloc/quick-lru": "^5.2.0",
|
| 1735 |
+
"arg": "^5.0.2",
|
| 1736 |
+
"chokidar": "^3.6.0",
|
| 1737 |
+
"didyoumean": "^1.2.2",
|
| 1738 |
+
"dlv": "^1.1.3",
|
| 1739 |
+
"fast-glob": "^3.3.2",
|
| 1740 |
+
"glob-parent": "^6.0.2",
|
| 1741 |
+
"is-glob": "^4.0.3",
|
| 1742 |
+
"jiti": "^1.21.7",
|
| 1743 |
+
"lilconfig": "^3.1.3",
|
| 1744 |
+
"micromatch": "^4.0.8",
|
| 1745 |
+
"normalize-path": "^3.0.0",
|
| 1746 |
+
"object-hash": "^3.0.0",
|
| 1747 |
+
"picocolors": "^1.1.1",
|
| 1748 |
+
"postcss": "^8.4.47",
|
| 1749 |
+
"postcss-import": "^15.1.0",
|
| 1750 |
+
"postcss-js": "^4.0.1",
|
| 1751 |
+
"postcss-load-config": "^4.0.2 || ^5.0 || ^6.0",
|
| 1752 |
+
"postcss-nested": "^6.2.0",
|
| 1753 |
+
"postcss-selector-parser": "^6.1.2",
|
| 1754 |
+
"resolve": "^1.22.8",
|
| 1755 |
+
"sucrase": "^3.35.0"
|
| 1756 |
+
},
|
| 1757 |
+
"bin": {
|
| 1758 |
+
"tailwind": "lib/cli.js",
|
| 1759 |
+
"tailwindcss": "lib/cli.js"
|
| 1760 |
+
},
|
| 1761 |
+
"engines": {
|
| 1762 |
+
"node": ">=14.0.0"
|
| 1763 |
+
}
|
| 1764 |
+
},
|
| 1765 |
+
"node_modules/thenify": {
|
| 1766 |
+
"version": "3.3.1",
|
| 1767 |
+
"resolved": "https://registry.npmjs.org/thenify/-/thenify-3.3.1.tgz",
|
| 1768 |
+
"integrity": "sha512-RVZSIV5IG10Hk3enotrhvz0T9em6cyHBLkH/YAZuKqd8hRkKhSfCGIcP2KUY0EPxndzANBmNllzWPwak+bheSw==",
|
| 1769 |
+
"license": "MIT",
|
| 1770 |
+
"dependencies": {
|
| 1771 |
+
"any-promise": "^1.0.0"
|
| 1772 |
+
}
|
| 1773 |
+
},
|
| 1774 |
+
"node_modules/thenify-all": {
|
| 1775 |
+
"version": "1.6.0",
|
| 1776 |
+
"resolved": "https://registry.npmjs.org/thenify-all/-/thenify-all-1.6.0.tgz",
|
| 1777 |
+
"integrity": "sha512-RNxQH/qI8/t3thXJDwcstUO4zeqo64+Uy/+sNVRBx4Xn2OX+OZ9oP+iJnNFqplFra2ZUVeKCSa2oVWi3T4uVmA==",
|
| 1778 |
+
"license": "MIT",
|
| 1779 |
+
"dependencies": {
|
| 1780 |
+
"thenify": ">= 3.1.0 < 4"
|
| 1781 |
+
},
|
| 1782 |
+
"engines": {
|
| 1783 |
+
"node": ">=0.8"
|
| 1784 |
+
}
|
| 1785 |
+
},
|
| 1786 |
+
"node_modules/tiny-invariant": {
|
| 1787 |
+
"version": "1.3.3",
|
| 1788 |
+
"resolved": "https://registry.npmjs.org/tiny-invariant/-/tiny-invariant-1.3.3.tgz",
|
| 1789 |
+
"integrity": "sha512-+FbBPE1o9QAYvviau/qC5SE3caw21q3xkvWKBtja5vgqOWIHHJ3ioaq1VPfn/Szqctz2bU/oYeKd9/z5BL+PVg==",
|
| 1790 |
+
"license": "MIT"
|
| 1791 |
+
},
|
| 1792 |
+
"node_modules/tinyglobby": {
|
| 1793 |
+
"version": "0.2.16",
|
| 1794 |
+
"resolved": "https://registry.npmjs.org/tinyglobby/-/tinyglobby-0.2.16.tgz",
|
| 1795 |
+
"integrity": "sha512-pn99VhoACYR8nFHhxqix+uvsbXineAasWm5ojXoN8xEwK5Kd3/TrhNn1wByuD52UxWRLy8pu+kRMniEi6Eq9Zg==",
|
| 1796 |
+
"license": "MIT",
|
| 1797 |
+
"dependencies": {
|
| 1798 |
+
"fdir": "^6.5.0",
|
| 1799 |
+
"picomatch": "^4.0.4"
|
| 1800 |
+
},
|
| 1801 |
+
"engines": {
|
| 1802 |
+
"node": ">=12.0.0"
|
| 1803 |
+
},
|
| 1804 |
+
"funding": {
|
| 1805 |
+
"url": "https://github.com/sponsors/SuperchupuDev"
|
| 1806 |
+
}
|
| 1807 |
+
},
|
| 1808 |
+
"node_modules/tinyglobby/node_modules/fdir": {
|
| 1809 |
+
"version": "6.5.0",
|
| 1810 |
+
"resolved": "https://registry.npmjs.org/fdir/-/fdir-6.5.0.tgz",
|
| 1811 |
+
"integrity": "sha512-tIbYtZbucOs0BRGqPJkshJUYdL+SDH7dVM8gjy+ERp3WAUjLEFJE+02kanyHtwjWOnwrKYBiwAmM0p4kLJAnXg==",
|
| 1812 |
+
"license": "MIT",
|
| 1813 |
+
"engines": {
|
| 1814 |
+
"node": ">=12.0.0"
|
| 1815 |
+
},
|
| 1816 |
+
"peerDependencies": {
|
| 1817 |
+
"picomatch": "^3 || ^4"
|
| 1818 |
+
},
|
| 1819 |
+
"peerDependenciesMeta": {
|
| 1820 |
+
"picomatch": {
|
| 1821 |
+
"optional": true
|
| 1822 |
+
}
|
| 1823 |
+
}
|
| 1824 |
+
},
|
| 1825 |
+
"node_modules/tinyglobby/node_modules/picomatch": {
|
| 1826 |
+
"version": "4.0.4",
|
| 1827 |
+
"resolved": "https://registry.npmjs.org/picomatch/-/picomatch-4.0.4.tgz",
|
| 1828 |
+
"integrity": "sha512-QP88BAKvMam/3NxH6vj2o21R6MjxZUAd6nlwAS/pnGvN9IVLocLHxGYIzFhg6fUQ+5th6P4dv4eW9jX3DSIj7A==",
|
| 1829 |
+
"license": "MIT",
|
| 1830 |
+
"engines": {
|
| 1831 |
+
"node": ">=12"
|
| 1832 |
+
},
|
| 1833 |
+
"funding": {
|
| 1834 |
+
"url": "https://github.com/sponsors/jonschlinkert"
|
| 1835 |
+
}
|
| 1836 |
+
},
|
| 1837 |
+
"node_modules/to-regex-range": {
|
| 1838 |
+
"version": "5.0.1",
|
| 1839 |
+
"resolved": "https://registry.npmjs.org/to-regex-range/-/to-regex-range-5.0.1.tgz",
|
| 1840 |
+
"integrity": "sha512-65P7iz6X5yEr1cwcgvQxbbIw7Uk3gOy5dIdtZ4rDveLqhrdJP+Li/Hx6tyK0NEb+2GCyneCMJiGqrADCSNk8sQ==",
|
| 1841 |
+
"license": "MIT",
|
| 1842 |
+
"dependencies": {
|
| 1843 |
+
"is-number": "^7.0.0"
|
| 1844 |
+
},
|
| 1845 |
+
"engines": {
|
| 1846 |
+
"node": ">=8.0"
|
| 1847 |
+
}
|
| 1848 |
+
},
|
| 1849 |
+
"node_modules/ts-interface-checker": {
|
| 1850 |
+
"version": "0.1.13",
|
| 1851 |
+
"resolved": "https://registry.npmjs.org/ts-interface-checker/-/ts-interface-checker-0.1.13.tgz",
|
| 1852 |
+
"integrity": "sha512-Y/arvbn+rrz3JCKl9C4kVNfTfSm2/mEp5FSz5EsZSANGPSlQrpRI5M4PKF+mJnE52jOO90PnPSc3Ur3bTQw0gA==",
|
| 1853 |
+
"license": "Apache-2.0"
|
| 1854 |
+
},
|
| 1855 |
+
"node_modules/tslib": {
|
| 1856 |
+
"version": "2.8.1",
|
| 1857 |
+
"resolved": "https://registry.npmjs.org/tslib/-/tslib-2.8.1.tgz",
|
| 1858 |
+
"integrity": "sha512-oJFu94HQb+KVduSUQL7wnpmqnfmLsOA/nAh6b6EH0wCEoK0/mPeXU6c3wKDV83MkOuHPRHtSXKKU99IBazS/2w==",
|
| 1859 |
+
"license": "0BSD"
|
| 1860 |
+
},
|
| 1861 |
+
"node_modules/typescript": {
|
| 1862 |
+
"version": "5.9.3",
|
| 1863 |
+
"resolved": "https://registry.npmjs.org/typescript/-/typescript-5.9.3.tgz",
|
| 1864 |
+
"integrity": "sha512-jl1vZzPDinLr9eUt3J/t7V6FgNEw9QjvBPdysz9KfQDD41fQrC2Y4vKQdiaUpFT4bXlb1RHhLpp8wtm6M5TgSw==",
|
| 1865 |
+
"license": "Apache-2.0",
|
| 1866 |
+
"bin": {
|
| 1867 |
+
"tsc": "bin/tsc",
|
| 1868 |
+
"tsserver": "bin/tsserver"
|
| 1869 |
+
},
|
| 1870 |
+
"engines": {
|
| 1871 |
+
"node": ">=14.17"
|
| 1872 |
+
}
|
| 1873 |
+
},
|
| 1874 |
+
"node_modules/undici-types": {
|
| 1875 |
+
"version": "6.21.0",
|
| 1876 |
+
"resolved": "https://registry.npmjs.org/undici-types/-/undici-types-6.21.0.tgz",
|
| 1877 |
+
"integrity": "sha512-iwDZqg0QAGrg9Rav5H4n0M64c3mkR59cJ6wQp+7C4nI0gsmExaedaYLNO44eT4AtBBwjbTiGPMlt2Md0T9H9JQ==",
|
| 1878 |
+
"license": "MIT"
|
| 1879 |
+
},
|
| 1880 |
+
"node_modules/update-browserslist-db": {
|
| 1881 |
+
"version": "1.2.3",
|
| 1882 |
+
"resolved": "https://registry.npmjs.org/update-browserslist-db/-/update-browserslist-db-1.2.3.tgz",
|
| 1883 |
+
"integrity": "sha512-Js0m9cx+qOgDxo0eMiFGEueWztz+d4+M3rGlmKPT+T4IS/jP4ylw3Nwpu6cpTTP8R1MAC1kF4VbdLt3ARf209w==",
|
| 1884 |
+
"funding": [
|
| 1885 |
+
{
|
| 1886 |
+
"type": "opencollective",
|
| 1887 |
+
"url": "https://opencollective.com/browserslist"
|
| 1888 |
+
},
|
| 1889 |
+
{
|
| 1890 |
+
"type": "tidelift",
|
| 1891 |
+
"url": "https://tidelift.com/funding/github/npm/browserslist"
|
| 1892 |
+
},
|
| 1893 |
+
{
|
| 1894 |
+
"type": "github",
|
| 1895 |
+
"url": "https://github.com/sponsors/ai"
|
| 1896 |
+
}
|
| 1897 |
+
],
|
| 1898 |
+
"license": "MIT",
|
| 1899 |
+
"dependencies": {
|
| 1900 |
+
"escalade": "^3.2.0",
|
| 1901 |
+
"picocolors": "^1.1.1"
|
| 1902 |
+
},
|
| 1903 |
+
"bin": {
|
| 1904 |
+
"update-browserslist-db": "cli.js"
|
| 1905 |
+
},
|
| 1906 |
+
"peerDependencies": {
|
| 1907 |
+
"browserslist": ">= 4.21.0"
|
| 1908 |
+
}
|
| 1909 |
+
},
|
| 1910 |
+
"node_modules/util-deprecate": {
|
| 1911 |
+
"version": "1.0.2",
|
| 1912 |
+
"resolved": "https://registry.npmjs.org/util-deprecate/-/util-deprecate-1.0.2.tgz",
|
| 1913 |
+
"integrity": "sha512-EPD5q1uXyFxJpCrLnCc1nHnq3gOa6DZBocAIiI2TaSCA7VCJ1UJDMagCzIkXNsUYfD1daK//LTEQ8xiIbrHtcw==",
|
| 1914 |
+
"license": "MIT"
|
| 1915 |
+
},
|
| 1916 |
+
"node_modules/victory-vendor": {
|
| 1917 |
+
"version": "36.9.2",
|
| 1918 |
+
"resolved": "https://registry.npmjs.org/victory-vendor/-/victory-vendor-36.9.2.tgz",
|
| 1919 |
+
"integrity": "sha512-PnpQQMuxlwYdocC8fIJqVXvkeViHYzotI+NJrCuav0ZYFoq912ZHBk3mCeuj+5/VpodOjPe1z0Fk2ihgzlXqjQ==",
|
| 1920 |
+
"license": "MIT AND ISC",
|
| 1921 |
+
"dependencies": {
|
| 1922 |
+
"@types/d3-array": "^3.0.3",
|
| 1923 |
+
"@types/d3-ease": "^3.0.0",
|
| 1924 |
+
"@types/d3-interpolate": "^3.0.1",
|
| 1925 |
+
"@types/d3-scale": "^4.0.2",
|
| 1926 |
+
"@types/d3-shape": "^3.1.0",
|
| 1927 |
+
"@types/d3-time": "^3.0.0",
|
| 1928 |
+
"@types/d3-timer": "^3.0.0",
|
| 1929 |
+
"d3-array": "^3.1.6",
|
| 1930 |
+
"d3-ease": "^3.0.1",
|
| 1931 |
+
"d3-interpolate": "^3.0.1",
|
| 1932 |
+
"d3-scale": "^4.0.2",
|
| 1933 |
+
"d3-shape": "^3.1.0",
|
| 1934 |
+
"d3-time": "^3.0.0",
|
| 1935 |
+
"d3-timer": "^3.0.1"
|
| 1936 |
+
}
|
| 1937 |
+
}
|
| 1938 |
+
}
|
| 1939 |
+
}
|
|
@@ -1,63 +1,140 @@
|
|
| 1 |
'use client'
|
| 2 |
import { cn, fmtNum } from '@/lib/utils'
|
| 3 |
import { stats, agentMsgs } from '@/lib/mock-data'
|
| 4 |
-
import { Brain, Zap, Shield, Eye, Play, Pause, Bot, ArrowRight, Cpu, RefreshCw, Target, Activity, AlertTriangle } from 'lucide-react'
|
| 5 |
-
import { useState, useEffect } from 'react'
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 6 |
|
| 7 |
export default function AgentPage() {
|
| 8 |
const [on, setOn] = useState(true)
|
| 9 |
-
const [tab, setTab] = useState<'feed'|'config'>('feed')
|
| 10 |
-
const [feed, setFeed] = useState<
|
|
|
|
|
|
|
| 11 |
|
| 12 |
useEffect(() => {
|
| 13 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
| 14 |
setFeed(init)
|
| 15 |
if (!on) return
|
| 16 |
let c = 8
|
| 17 |
const iv = setInterval(() => {
|
| 18 |
const t = agentMsgs[c++ % agentMsgs.length]
|
| 19 |
-
setFeed(p => [{text:t, time: new Date().toLocaleTimeString('en-US',{hour12:false})}, ...p].slice(0,
|
| 20 |
}, 3000)
|
| 21 |
return () => clearInterval(iv)
|
| 22 |
}, [on])
|
| 23 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 24 |
return (
|
| 25 |
<div className="p-6 space-y-6">
|
| 26 |
<div className="flex items-center justify-between">
|
| 27 |
<div className="flex items-center gap-4">
|
| 28 |
-
<div className={cn('w-12 h-12 rounded-xl flex items-center justify-center', on?'bg-trading-up/10 animate-pulse-glow':'bg-surface-elevated')}><Brain className={cn('w-6 h-6', on?'text-trading-up':'text-muted')}/></div>
|
| 29 |
<div><h1 className="text-display-sm text-[#eaecef]">AI Agent</h1><p className="text-body-md text-muted mt-0.5">Autonomous churn detection & retention engine</p></div>
|
| 30 |
</div>
|
| 31 |
<div className="flex items-center gap-3">
|
| 32 |
-
|
| 33 |
-
<div className=
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 34 |
</div>
|
| 35 |
-
<button onClick={() => setOn(!on)} className={cn('flex items-center gap-2 px-5 py-2.5 rounded-md text-button font-semibold transition', on?'bg-trading-down/10 text-trading-down border border-trading-down/30 hover:bg-trading-down/20':'bg-brand-yellow text-ink hover:bg-brand-yellow-active')}>
|
| 36 |
-
{on?<Pause className="w-4 h-4"/>:<Play className="w-4 h-4"/>}{on?'Pause
|
| 37 |
</button>
|
| 38 |
</div>
|
| 39 |
</div>
|
| 40 |
|
| 41 |
<div className="grid grid-cols-2 lg:grid-cols-4 gap-4">
|
| 42 |
-
{[{i:Activity,l:'Actions Today',v:fmtNum(stats.agentActionsToday),c:'text-brand-yellow'},{i:Eye,l:'Wallets Scanned',v:fmtNum(stats.activeWallets),c:'text-info'},{i:Shield,l:'Wallets Saved',v:fmtNum(stats.walletsSaved),c:'text-trading-up'},{i:Target,l:'Churn Prevented',v:fmtNum(stats.walletsAtRisk),c:'text-brand-yellow'}].map(s=>{const I=s.i;return(
|
| 43 |
-
<div key={s.l} className="rounded-xl bg-surface-card border border-hairline-dark p-4"><div className="flex items-center gap-2 mb-2"><I className={cn('w-4 h-4',s.c)}/><span className="text-caption text-muted">{s.l}</span></div><p className={cn('font-mono text-title-lg tabular-nums',s.c==='text-info'?'text-[#eaecef]':s.c)}>{s.v}</p></div>
|
| 44 |
)})}
|
| 45 |
</div>
|
| 46 |
|
| 47 |
<div className="flex items-center gap-1 p-1 rounded-lg bg-surface-card border border-hairline-dark w-fit">
|
| 48 |
-
{(['feed','config'] as const).map(t => <button key={t} onClick={()=>setTab(t)} className={cn('px-5 py-2 rounded-md text-button transition capitalize', tab===t?'bg-brand-yellow text-ink font-semibold':'text-muted hover:text-[#eaecef]')}>{t==='feed'?'Live Feed':'Configuration'}</button>)}
|
| 49 |
</div>
|
| 50 |
|
| 51 |
{tab === 'feed' && (
|
| 52 |
<div className="rounded-xl bg-surface-card border border-hairline-dark overflow-hidden">
|
| 53 |
<div className="p-4 border-b border-hairline-dark flex items-center justify-between">
|
| 54 |
-
<div className="flex items-center gap-2"><Bot className="w-4 h-4 text-brand-yellow"/><span className="text-title-sm">Real-Time Agent Feed</span>{on && <div className="w-2 h-2 rounded-full bg-trading-up animate-pulse"/>}</div>
|
| 55 |
-
<
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 56 |
</div>
|
| 57 |
-
<div className="max-h-[500px] overflow-y-auto divide-y divide-hairline-dark/50">{feed.map((m,i) => (
|
| 58 |
-
<div key={`${m.time}-${i}`} className={cn('flex items-start gap-4 px-5 py-3 transition-colors', i===0 && on && 'animate-slide-up bg-brand-yellow/5')}>
|
| 59 |
<span className="font-mono text-caption text-muted tabular-nums whitespace-nowrap mt-0.5">{m.time}</span>
|
| 60 |
-
<span className=
|
|
|
|
| 61 |
</div>
|
| 62 |
))}</div>
|
| 63 |
</div>
|
|
@@ -66,19 +143,32 @@ export default function AgentPage() {
|
|
| 66 |
{tab === 'config' && (
|
| 67 |
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
|
| 68 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-5">
|
| 69 |
-
<h3 className="text-title-sm mb-4 flex items-center gap-2"><AlertTriangle className="w-4 h-4 text-brand-yellow"/>Detection Thresholds</h3>
|
| 70 |
-
<div className="space-y-3">{[{l:'critical',d:10,v:90},{l:'high',d:7,v:60},{l:'medium',d:5,v:30}].map(t => (
|
| 71 |
<div key={t.l} className="p-3 rounded-lg bg-surface-elevated border border-hairline-dark/50">
|
| 72 |
-
<span className={cn('text-caption font-semibold uppercase', t.l==='critical'?'text-trading-down':t.l==='high'?'text-[#ff9500]':'text-brand-yellow')}>{t.l}</span>
|
| 73 |
-
<div className="grid grid-cols-2 gap-3 mt-2"><div><span className="text-caption text-muted">Inactive Days</span><p className="font-mono text-num-md text-[#eaecef]">{'
|
| 74 |
</div>
|
| 75 |
))}</div>
|
| 76 |
</div>
|
| 77 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-5">
|
| 78 |
-
<h3 className="text-title-sm mb-4 flex items-center gap-2"><Cpu className="w-4 h-4 text-brand-yellow"/>Agent Configuration</h3>
|
| 79 |
-
<div className="space-y-0">{[
|
| 80 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 81 |
))}</div>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 82 |
</div>
|
| 83 |
</div>
|
| 84 |
)}
|
|
@@ -86,12 +176,12 @@ export default function AgentPage() {
|
|
| 86 |
<div className="rounded-xl bg-brand-yellow/5 border border-brand-yellow/20 p-6">
|
| 87 |
<h3 className="text-title-sm text-brand-yellow mb-3">How FlowState AI Agent Works</h3>
|
| 88 |
<div className="grid grid-cols-5 gap-3 items-center">
|
| 89 |
-
{[{i:Eye,l:'Monitor',d:'Helius webhooks scan Solana txns'},{i:Brain,l:'Detect',d:'AI scores churn risk per wallet'},{i:Zap,l:'Decide',d:'Select optimal incentive type'},{i:Bot,l:'Execute',d:'Fire events via Torque
|
| 90 |
<div key={s.l} className="text-center">
|
| 91 |
-
<div className="w-10 h-10 rounded-lg bg-surface-card border border-brand-yellow/30 flex items-center justify-center mx-auto mb-2"><s.i className="w-5 h-5 text-brand-yellow"/></div>
|
| 92 |
<p className="text-caption text-brand-yellow font-semibold">{s.l}</p>
|
| 93 |
<p className="text-[10px] text-muted mt-0.5">{s.d}</p>
|
| 94 |
-
{i < 4 && <ArrowRight className="w-4 h-4 text-brand-yellow/30 mx-auto mt-2 hidden lg:block"/>}
|
| 95 |
</div>
|
| 96 |
))}
|
| 97 |
</div>
|
|
|
|
| 1 |
'use client'
|
| 2 |
import { cn, fmtNum } from '@/lib/utils'
|
| 3 |
import { stats, agentMsgs } from '@/lib/mock-data'
|
| 4 |
+
import { Brain, Zap, Shield, Eye, Play, Pause, Bot, ArrowRight, Cpu, RefreshCw, Target, Activity, AlertTriangle, Scan } from 'lucide-react'
|
| 5 |
+
import { useState, useEffect, useCallback } from 'react'
|
| 6 |
+
|
| 7 |
+
type FeedMsg = { text: string; time: string; real?: boolean }
|
| 8 |
+
|
| 9 |
+
type ScanDetection = {
|
| 10 |
+
wallet: string; risk: string; score: number; eventType: string;
|
| 11 |
+
eventSent: boolean; eventId?: string; error?: string
|
| 12 |
+
}
|
| 13 |
+
|
| 14 |
+
type ScanResult = {
|
| 15 |
+
detections: ScanDetection[]; count: number; configured: boolean; timestamp: string
|
| 16 |
+
}
|
| 17 |
+
|
| 18 |
+
const RISK_ICON: Record<string, string> = {
|
| 19 |
+
critical: '🚨', high: '⚠️', medium: '📊',
|
| 20 |
+
}
|
| 21 |
|
| 22 |
export default function AgentPage() {
|
| 23 |
const [on, setOn] = useState(true)
|
| 24 |
+
const [tab, setTab] = useState<'feed' | 'config'>('feed')
|
| 25 |
+
const [feed, setFeed] = useState<FeedMsg[]>([])
|
| 26 |
+
const [scanning, setScanning] = useState(false)
|
| 27 |
+
const [configured, setConfigured] = useState<boolean | null>(null)
|
| 28 |
|
| 29 |
useEffect(() => {
|
| 30 |
+
fetch('/api/agent/scan').then(r => r.json()).then(d => setConfigured(d.configured)).catch(() => setConfigured(false))
|
| 31 |
+
}, [])
|
| 32 |
+
|
| 33 |
+
useEffect(() => {
|
| 34 |
+
const init = agentMsgs.slice(0, 8).map((t, i) => ({ text: t, time: new Date(Date.now() - i * 120000).toLocaleTimeString('en-US', { hour12: false }) }))
|
| 35 |
setFeed(init)
|
| 36 |
if (!on) return
|
| 37 |
let c = 8
|
| 38 |
const iv = setInterval(() => {
|
| 39 |
const t = agentMsgs[c++ % agentMsgs.length]
|
| 40 |
+
setFeed(p => [{ text: t, time: new Date().toLocaleTimeString('en-US', { hour12: false }) }, ...p].slice(0, 50))
|
| 41 |
}, 3000)
|
| 42 |
return () => clearInterval(iv)
|
| 43 |
}, [on])
|
| 44 |
|
| 45 |
+
const triggerScan = useCallback(async () => {
|
| 46 |
+
setScanning(true)
|
| 47 |
+
const now = () => new Date().toLocaleTimeString('en-US', { hour12: false })
|
| 48 |
+
try {
|
| 49 |
+
setFeed(p => [{ text: '🔍 Triggering real wallet scan via Torque API...', time: now(), real: true }, ...p].slice(0, 50))
|
| 50 |
+
const res = await fetch('/api/agent/scan', { method: 'POST' })
|
| 51 |
+
const data: ScanResult = await res.json()
|
| 52 |
+
|
| 53 |
+
const msgs: FeedMsg[] = []
|
| 54 |
+
|
| 55 |
+
if (!data.configured) {
|
| 56 |
+
msgs.push({ text: '⚠️ TORQUE_API_KEY not set — add it to .env to fire live events', time: now(), real: true })
|
| 57 |
+
msgs.push({ text: `📋 Scan ran locally: ${data.count} at-risk wallets scored (no events sent)`, time: now(), real: true })
|
| 58 |
+
} else {
|
| 59 |
+
msgs.push({ text: `✅ Scan complete — ${data.count} at-risk wallets detected`, time: now(), real: true })
|
| 60 |
+
}
|
| 61 |
+
|
| 62 |
+
for (const d of data.detections.slice(0, 6)) {
|
| 63 |
+
const icon = RISK_ICON[d.risk] || '📊'
|
| 64 |
+
if (d.eventSent) {
|
| 65 |
+
msgs.push({ text: `${icon} ${d.risk.toUpperCase()}: ${d.wallet.slice(0, 8)}... → ${d.eventType} sent [${d.eventId}]`, time: now(), real: true })
|
| 66 |
+
} else if (data.configured) {
|
| 67 |
+
msgs.push({ text: `❌ ${d.risk.toUpperCase()}: ${d.wallet.slice(0, 8)}... → ${d.error}`, time: now(), real: true })
|
| 68 |
+
} else {
|
| 69 |
+
msgs.push({ text: `${icon} ${d.risk.toUpperCase()}: ${d.wallet.slice(0, 8)}... score=${d.score} → ${d.eventType}`, time: now(), real: true })
|
| 70 |
+
}
|
| 71 |
+
}
|
| 72 |
+
|
| 73 |
+
setFeed(p => [...msgs.reverse(), ...p].slice(0, 50))
|
| 74 |
+
setConfigured(data.configured)
|
| 75 |
+
} catch (e) {
|
| 76 |
+
setFeed(p => [{ text: '❌ Scan failed: ' + String(e), time: now(), real: true }, ...p].slice(0, 50))
|
| 77 |
+
} finally {
|
| 78 |
+
setScanning(false)
|
| 79 |
+
}
|
| 80 |
+
}, [])
|
| 81 |
+
|
| 82 |
return (
|
| 83 |
<div className="p-6 space-y-6">
|
| 84 |
<div className="flex items-center justify-between">
|
| 85 |
<div className="flex items-center gap-4">
|
| 86 |
+
<div className={cn('w-12 h-12 rounded-xl flex items-center justify-center', on ? 'bg-trading-up/10 animate-pulse-glow' : 'bg-surface-elevated')}><Brain className={cn('w-6 h-6', on ? 'text-trading-up' : 'text-muted')} /></div>
|
| 87 |
<div><h1 className="text-display-sm text-[#eaecef]">AI Agent</h1><p className="text-body-md text-muted mt-0.5">Autonomous churn detection & retention engine</p></div>
|
| 88 |
</div>
|
| 89 |
<div className="flex items-center gap-3">
|
| 90 |
+
{configured !== null && (
|
| 91 |
+
<div className={cn('px-3 py-1.5 rounded-lg border text-caption font-semibold', configured ? 'bg-trading-up/10 border-trading-up/30 text-trading-up' : 'bg-trading-down/10 border-trading-down/30 text-trading-down')}>
|
| 92 |
+
{configured ? '⚡ Torque Connected' : '⚠️ API Key Missing'}
|
| 93 |
+
</div>
|
| 94 |
+
)}
|
| 95 |
+
<button
|
| 96 |
+
onClick={triggerScan}
|
| 97 |
+
disabled={scanning}
|
| 98 |
+
className="flex items-center gap-2 px-4 py-2.5 rounded-md text-button font-semibold bg-brand-yellow/10 text-brand-yellow border border-brand-yellow/30 hover:bg-brand-yellow/20 transition disabled:opacity-50"
|
| 99 |
+
>
|
| 100 |
+
<Scan className={cn('w-4 h-4', scanning && 'animate-spin')} />
|
| 101 |
+
{scanning ? 'Scanning...' : 'Scan Now'}
|
| 102 |
+
</button>
|
| 103 |
+
<div className={cn('px-4 py-2 rounded-lg border', on ? 'bg-trading-up/10 border-trading-up/30 text-trading-up' : 'bg-surface-card border-hairline-dark text-muted')}>
|
| 104 |
+
<div className="flex items-center gap-2"><div className={cn('w-2.5 h-2.5 rounded-full', on ? 'bg-trading-up animate-pulse' : 'bg-muted')} /><span className="text-button font-semibold">{on ? 'RUNNING' : 'PAUSED'}</span></div>
|
| 105 |
</div>
|
| 106 |
+
<button onClick={() => setOn(!on)} className={cn('flex items-center gap-2 px-5 py-2.5 rounded-md text-button font-semibold transition', on ? 'bg-trading-down/10 text-trading-down border border-trading-down/30 hover:bg-trading-down/20' : 'bg-brand-yellow text-ink hover:bg-brand-yellow-active')}>
|
| 107 |
+
{on ? <Pause className="w-4 h-4" /> : <Play className="w-4 h-4" />}{on ? 'Pause' : 'Start'}
|
| 108 |
</button>
|
| 109 |
</div>
|
| 110 |
</div>
|
| 111 |
|
| 112 |
<div className="grid grid-cols-2 lg:grid-cols-4 gap-4">
|
| 113 |
+
{[{ i: Activity, l: 'Actions Today', v: fmtNum(stats.agentActionsToday), c: 'text-brand-yellow' }, { i: Eye, l: 'Wallets Scanned', v: fmtNum(stats.activeWallets), c: 'text-info' }, { i: Shield, l: 'Wallets Saved', v: fmtNum(stats.walletsSaved), c: 'text-trading-up' }, { i: Target, l: 'Churn Prevented', v: fmtNum(stats.walletsAtRisk), c: 'text-brand-yellow' }].map(s => { const I = s.i; return (
|
| 114 |
+
<div key={s.l} className="rounded-xl bg-surface-card border border-hairline-dark p-4"><div className="flex items-center gap-2 mb-2"><I className={cn('w-4 h-4', s.c)} /><span className="text-caption text-muted">{s.l}</span></div><p className={cn('font-mono text-title-lg tabular-nums', s.c === 'text-info' ? 'text-[#eaecef]' : s.c)}>{s.v}</p></div>
|
| 115 |
)})}
|
| 116 |
</div>
|
| 117 |
|
| 118 |
<div className="flex items-center gap-1 p-1 rounded-lg bg-surface-card border border-hairline-dark w-fit">
|
| 119 |
+
{(['feed', 'config'] as const).map(t => <button key={t} onClick={() => setTab(t)} className={cn('px-5 py-2 rounded-md text-button transition capitalize', tab === t ? 'bg-brand-yellow text-ink font-semibold' : 'text-muted hover:text-[#eaecef]')}>{t === 'feed' ? 'Live Feed' : 'Configuration'}</button>)}
|
| 120 |
</div>
|
| 121 |
|
| 122 |
{tab === 'feed' && (
|
| 123 |
<div className="rounded-xl bg-surface-card border border-hairline-dark overflow-hidden">
|
| 124 |
<div className="p-4 border-b border-hairline-dark flex items-center justify-between">
|
| 125 |
+
<div className="flex items-center gap-2"><Bot className="w-4 h-4 text-brand-yellow" /><span className="text-title-sm">Real-Time Agent Feed</span>{on && <div className="w-2 h-2 rounded-full bg-trading-up animate-pulse" />}</div>
|
| 126 |
+
<div className="flex items-center gap-3">
|
| 127 |
+
<span className="text-caption text-muted">{feed.length} messages</span>
|
| 128 |
+
{!configured && configured !== null && (
|
| 129 |
+
<span className="text-caption text-trading-down">Set TORQUE_API_KEY to fire live events</span>
|
| 130 |
+
)}
|
| 131 |
+
</div>
|
| 132 |
</div>
|
| 133 |
+
<div className="max-h-[500px] overflow-y-auto divide-y divide-hairline-dark/50">{feed.map((m, i) => (
|
| 134 |
+
<div key={`${m.time}-${i}`} className={cn('flex items-start gap-4 px-5 py-3 transition-colors', i === 0 && on && 'animate-slide-up bg-brand-yellow/5', m.real && 'bg-trading-up/5 border-l-2 border-trading-up/40')}>
|
| 135 |
<span className="font-mono text-caption text-muted tabular-nums whitespace-nowrap mt-0.5">{m.time}</span>
|
| 136 |
+
<span className={cn('text-body-sm', m.real ? 'text-[#eaecef] font-medium' : 'text-[#eaecef]')}>{m.text}</span>
|
| 137 |
+
{m.real && <span className="ml-auto text-[10px] text-trading-up font-semibold uppercase tracking-wider whitespace-nowrap">LIVE</span>}
|
| 138 |
</div>
|
| 139 |
))}</div>
|
| 140 |
</div>
|
|
|
|
| 143 |
{tab === 'config' && (
|
| 144 |
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
|
| 145 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-5">
|
| 146 |
+
<h3 className="text-title-sm mb-4 flex items-center gap-2"><AlertTriangle className="w-4 h-4 text-brand-yellow" />Detection Thresholds</h3>
|
| 147 |
+
<div className="space-y-3">{[{ l: 'critical', d: 10, v: 90 }, { l: 'high', d: 7, v: 60 }, { l: 'medium', d: 5, v: 30 }].map(t => (
|
| 148 |
<div key={t.l} className="p-3 rounded-lg bg-surface-elevated border border-hairline-dark/50">
|
| 149 |
+
<span className={cn('text-caption font-semibold uppercase', t.l === 'critical' ? 'text-trading-down' : t.l === 'high' ? 'text-[#ff9500]' : 'text-brand-yellow')}>{t.l}</span>
|
| 150 |
+
<div className="grid grid-cols-2 gap-3 mt-2"><div><span className="text-caption text-muted">Inactive Days</span><p className="font-mono text-num-md text-[#eaecef]">{'>='} {t.d}</p></div><div><span className="text-caption text-muted">Volume Drop</span><p className="font-mono text-num-md text-[#eaecef]">{'>='} {t.v}%</p></div></div>
|
| 151 |
</div>
|
| 152 |
))}</div>
|
| 153 |
</div>
|
| 154 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-5">
|
| 155 |
+
<h3 className="text-title-sm mb-4 flex items-center gap-2"><Cpu className="w-4 h-4 text-brand-yellow" />Agent Configuration</h3>
|
| 156 |
+
<div className="space-y-0">{[
|
| 157 |
+
['Scan Interval', '30s'],
|
| 158 |
+
['Monitored Protocols', '6'],
|
| 159 |
+
['Torque API', configured === null ? 'Checking...' : configured ? 'Connected' : 'Not Configured'],
|
| 160 |
+
['Sybil Filter', 'Enabled'],
|
| 161 |
+
].map(([k, v]) => (
|
| 162 |
+
<div key={k} className="flex items-center justify-between py-2 border-b border-hairline-dark/50">
|
| 163 |
+
<span className="text-body-sm text-muted">{k}</span>
|
| 164 |
+
<span className={cn('font-mono text-num-sm', v === 'Connected' || v === 'Enabled' ? 'text-trading-up font-semibold' : v === 'Not Configured' ? 'text-trading-down font-semibold' : 'text-[#eaecef]')}>{v}</span>
|
| 165 |
+
</div>
|
| 166 |
))}</div>
|
| 167 |
+
<div className="mt-4 p-3 rounded-lg bg-surface-elevated border border-brand-yellow/20">
|
| 168 |
+
<p className="text-caption text-muted mb-1">To enable live events:</p>
|
| 169 |
+
<p className="text-caption text-muted mb-0.5">Get token: platform.torque.so/connect-mcp</p>
|
| 170 |
+
<code className="text-caption text-brand-yellow font-mono">TORQUE_API_TOKEN=your-token</code>
|
| 171 |
+
</div>
|
| 172 |
</div>
|
| 173 |
</div>
|
| 174 |
)}
|
|
|
|
| 176 |
<div className="rounded-xl bg-brand-yellow/5 border border-brand-yellow/20 p-6">
|
| 177 |
<h3 className="text-title-sm text-brand-yellow mb-3">How FlowState AI Agent Works</h3>
|
| 178 |
<div className="grid grid-cols-5 gap-3 items-center">
|
| 179 |
+
{[{ i: Eye, l: 'Monitor', d: 'Helius webhooks scan Solana txns' }, { i: Brain, l: 'Detect', d: 'AI scores churn risk per wallet' }, { i: Zap, l: 'Decide', d: 'Select optimal incentive type' }, { i: Bot, l: 'Execute', d: 'Fire events via Torque API' }, { i: RefreshCw, l: 'Learn', d: 'Track outcomes, improve model' }].map((s, i) => (
|
| 180 |
<div key={s.l} className="text-center">
|
| 181 |
+
<div className="w-10 h-10 rounded-lg bg-surface-card border border-brand-yellow/30 flex items-center justify-center mx-auto mb-2"><s.i className="w-5 h-5 text-brand-yellow" /></div>
|
| 182 |
<p className="text-caption text-brand-yellow font-semibold">{s.l}</p>
|
| 183 |
<p className="text-[10px] text-muted mt-0.5">{s.d}</p>
|
| 184 |
+
{i < 4 && <ArrowRight className="w-4 h-4 text-brand-yellow/30 mx-auto mt-2 hidden lg:block" />}
|
| 185 |
</div>
|
| 186 |
))}
|
| 187 |
</div>
|
|
@@ -1,9 +1,10 @@
|
|
| 1 |
'use client'
|
| 2 |
-
import { Plus, Bot, Users, Activity, DollarSign,
|
| 3 |
import { cn, fmtNum, fmtUsd } from '@/lib/utils'
|
| 4 |
import { campaigns } from '@/lib/mock-data'
|
| 5 |
import { CampaignBadge } from '@/components/ui/CampaignBadge'
|
| 6 |
import { BarChart, Bar, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer } from 'recharts'
|
|
|
|
| 7 |
import type { CampaignType } from '@/lib/types'
|
| 8 |
|
| 9 |
const perfData = [
|
|
@@ -19,29 +20,75 @@ const Tip = ({ active, payload, label }: any) => {
|
|
| 19 |
}
|
| 20 |
const sc: Record<string, string> = { active:'bg-trading-up/10 text-trading-up', ended:'bg-brand-yellow/10 text-brand-yellow', draft:'bg-muted/10 text-muted', distributed:'bg-brand-turquoise/10 text-brand-turquoise' }
|
| 21 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 22 |
export default function CampaignsPage() {
|
| 23 |
const total = campaigns.reduce((s,c) => s+c.budget, 0)
|
| 24 |
const dist = campaigns.reduce((s,c) => s+c.rewardsDistributed, 0)
|
| 25 |
const parts = campaigns.reduce((s,c) => s+c.participantCount, 0)
|
| 26 |
const ai = campaigns.filter(c => c.createdBy === 'ai-agent').length
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 27 |
return (
|
| 28 |
<div className="p-6 space-y-6">
|
| 29 |
<div className="flex items-center justify-between">
|
| 30 |
<div><h1 className="text-display-sm text-[#eaecef]">Campaigns</h1><p className="text-body-md text-muted mt-1">Manage Torque campaigns — leaderboards, rebates, raffles & gifts</p></div>
|
| 31 |
-
<button className="flex items-center gap-2 px-5 py-2.5 rounded-md bg-brand-yellow text-ink text-button font-semibold hover:bg-brand-yellow-active transition"><Plus className="w-4 h-4"/>Create Campaign</button>
|
| 32 |
</div>
|
|
|
|
| 33 |
<div className="grid grid-cols-4 gap-4">
|
| 34 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-4"><span className="text-caption text-muted">Total Budget</span><p className="font-mono text-title-lg text-[#eaecef] mt-1 tabular-nums">{fmtUsd(total)}</p></div>
|
| 35 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-4"><span className="text-caption text-muted">Distributed</span><p className="font-mono text-title-lg text-trading-up mt-1 tabular-nums">{fmtUsd(dist)}</p></div>
|
| 36 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-4"><span className="text-caption text-muted">Participants</span><p className="font-mono text-title-lg text-[#eaecef] mt-1 tabular-nums">{fmtNum(parts)}</p></div>
|
| 37 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-4"><div className="flex items-center gap-1.5"><Bot className="w-3.5 h-3.5 text-brand-yellow"/><span className="text-caption text-muted">AI Created</span></div><p className="font-mono text-title-lg text-brand-yellow mt-1 tabular-nums">{ai}/{campaigns.length}</p></div>
|
| 38 |
</div>
|
|
|
|
| 39 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-5">
|
| 40 |
<h3 className="text-title-sm mb-4">Campaign Performance</h3>
|
| 41 |
<ResponsiveContainer width="100%" height={250}>
|
| 42 |
<BarChart data={perfData}><CartesianGrid strokeDasharray="3 3" stroke="#2b3139"/><XAxis dataKey="name" tick={{fill:'#707a8a',fontSize:11}} axisLine={{stroke:'#2b3139'}}/><YAxis tick={{fill:'#707a8a',fontSize:11}} axisLine={{stroke:'#2b3139'}}/><Tooltip content={<Tip/>}/><Bar dataKey="p" fill="#FCD535" radius={[4,4,0,0]} name="Participants"/><Bar dataKey="e" fill="#0ecb81" radius={[4,4,0,0]} name="Events"/></BarChart>
|
| 43 |
</ResponsiveContainer>
|
| 44 |
</div>
|
|
|
|
| 45 |
<div className="grid grid-cols-1 lg:grid-cols-2 gap-4">
|
| 46 |
{campaigns.map(c => (
|
| 47 |
<div key={c.id} className="rounded-xl bg-surface-card border border-hairline-dark p-5 hover:border-brand-yellow/30 transition-all group cursor-pointer">
|
|
@@ -60,6 +107,73 @@ export default function CampaignsPage() {
|
|
| 60 |
</div>
|
| 61 |
))}
|
| 62 |
</div>
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 63 |
</div>
|
| 64 |
)
|
| 65 |
}
|
|
|
|
| 1 |
'use client'
|
| 2 |
+
import { Plus, Bot, Users, Activity, DollarSign, X, Loader2, CheckCircle2, AlertCircle } from 'lucide-react'
|
| 3 |
import { cn, fmtNum, fmtUsd } from '@/lib/utils'
|
| 4 |
import { campaigns } from '@/lib/mock-data'
|
| 5 |
import { CampaignBadge } from '@/components/ui/CampaignBadge'
|
| 6 |
import { BarChart, Bar, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer } from 'recharts'
|
| 7 |
+
import { useState, useCallback } from 'react'
|
| 8 |
import type { CampaignType } from '@/lib/types'
|
| 9 |
|
| 10 |
const perfData = [
|
|
|
|
| 20 |
}
|
| 21 |
const sc: Record<string, string> = { active:'bg-trading-up/10 text-trading-up', ended:'bg-brand-yellow/10 text-brand-yellow', draft:'bg-muted/10 text-muted', distributed:'bg-brand-turquoise/10 text-brand-turquoise' }
|
| 22 |
|
| 23 |
+
const CAMPAIGN_TYPES = [
|
| 24 |
+
{ value: 'LEADERBOARD', label: 'Leaderboard', desc: 'Rank users by onchain activity' },
|
| 25 |
+
{ value: 'RAFFLE', label: 'Raffle', desc: 'Random reward from eligible pool' },
|
| 26 |
+
{ value: 'BOUNTY', label: 'Gift / Bounty', desc: 'Guaranteed reward for action' },
|
| 27 |
+
{ value: 'TRIAL', label: 'Rebate / Trial', desc: 'Cashback for qualifying activity' },
|
| 28 |
+
]
|
| 29 |
+
|
| 30 |
+
const EVENT_OPTIONS = [
|
| 31 |
+
'churn_risk_high', 'churn_risk_medium', 'comeback_detected',
|
| 32 |
+
'streak_maintained', 'volume_milestone', 'inactivity_detected', 'referral_from_saved',
|
| 33 |
+
]
|
| 34 |
+
|
| 35 |
+
interface CreateResult { success: boolean; campaignId?: string; error?: string }
|
| 36 |
+
|
| 37 |
export default function CampaignsPage() {
|
| 38 |
const total = campaigns.reduce((s,c) => s+c.budget, 0)
|
| 39 |
const dist = campaigns.reduce((s,c) => s+c.rewardsDistributed, 0)
|
| 40 |
const parts = campaigns.reduce((s,c) => s+c.participantCount, 0)
|
| 41 |
const ai = campaigns.filter(c => c.createdBy === 'ai-agent').length
|
| 42 |
+
|
| 43 |
+
const [open, setOpen] = useState(false)
|
| 44 |
+
const [creating, setCreating] = useState(false)
|
| 45 |
+
const [result, setResult] = useState<CreateResult | null>(null)
|
| 46 |
+
const [form, setForm] = useState({ type: 'LEADERBOARD', name: '', description: '', budget: '', eventName: 'churn_risk_high' })
|
| 47 |
+
|
| 48 |
+
const handleCreate = useCallback(async () => {
|
| 49 |
+
if (!form.name.trim() || !form.budget) return
|
| 50 |
+
setCreating(true)
|
| 51 |
+
setResult(null)
|
| 52 |
+
try {
|
| 53 |
+
const res = await fetch('/api/torque/campaigns', {
|
| 54 |
+
method: 'POST',
|
| 55 |
+
headers: { 'Content-Type': 'application/json' },
|
| 56 |
+
body: JSON.stringify({ ...form, budget: parseFloat(form.budget) }),
|
| 57 |
+
})
|
| 58 |
+
const data = await res.json()
|
| 59 |
+
if (data.success) {
|
| 60 |
+
setResult({ success: true, campaignId: data.campaignId })
|
| 61 |
+
} else {
|
| 62 |
+
setResult({ success: false, error: data.error || 'Campaign creation failed' })
|
| 63 |
+
}
|
| 64 |
+
} catch (e: any) {
|
| 65 |
+
setResult({ success: false, error: e.message })
|
| 66 |
+
} finally {
|
| 67 |
+
setCreating(false)
|
| 68 |
+
}
|
| 69 |
+
}, [form])
|
| 70 |
+
|
| 71 |
return (
|
| 72 |
<div className="p-6 space-y-6">
|
| 73 |
<div className="flex items-center justify-between">
|
| 74 |
<div><h1 className="text-display-sm text-[#eaecef]">Campaigns</h1><p className="text-body-md text-muted mt-1">Manage Torque campaigns — leaderboards, rebates, raffles & gifts</p></div>
|
| 75 |
+
<button onClick={() => { setOpen(true); setResult(null) }} className="flex items-center gap-2 px-5 py-2.5 rounded-md bg-brand-yellow text-ink text-button font-semibold hover:bg-brand-yellow-active transition"><Plus className="w-4 h-4"/>Create Campaign</button>
|
| 76 |
</div>
|
| 77 |
+
|
| 78 |
<div className="grid grid-cols-4 gap-4">
|
| 79 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-4"><span className="text-caption text-muted">Total Budget</span><p className="font-mono text-title-lg text-[#eaecef] mt-1 tabular-nums">{fmtUsd(total)}</p></div>
|
| 80 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-4"><span className="text-caption text-muted">Distributed</span><p className="font-mono text-title-lg text-trading-up mt-1 tabular-nums">{fmtUsd(dist)}</p></div>
|
| 81 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-4"><span className="text-caption text-muted">Participants</span><p className="font-mono text-title-lg text-[#eaecef] mt-1 tabular-nums">{fmtNum(parts)}</p></div>
|
| 82 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-4"><div className="flex items-center gap-1.5"><Bot className="w-3.5 h-3.5 text-brand-yellow"/><span className="text-caption text-muted">AI Created</span></div><p className="font-mono text-title-lg text-brand-yellow mt-1 tabular-nums">{ai}/{campaigns.length}</p></div>
|
| 83 |
</div>
|
| 84 |
+
|
| 85 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-5">
|
| 86 |
<h3 className="text-title-sm mb-4">Campaign Performance</h3>
|
| 87 |
<ResponsiveContainer width="100%" height={250}>
|
| 88 |
<BarChart data={perfData}><CartesianGrid strokeDasharray="3 3" stroke="#2b3139"/><XAxis dataKey="name" tick={{fill:'#707a8a',fontSize:11}} axisLine={{stroke:'#2b3139'}}/><YAxis tick={{fill:'#707a8a',fontSize:11}} axisLine={{stroke:'#2b3139'}}/><Tooltip content={<Tip/>}/><Bar dataKey="p" fill="#FCD535" radius={[4,4,0,0]} name="Participants"/><Bar dataKey="e" fill="#0ecb81" radius={[4,4,0,0]} name="Events"/></BarChart>
|
| 89 |
</ResponsiveContainer>
|
| 90 |
</div>
|
| 91 |
+
|
| 92 |
<div className="grid grid-cols-1 lg:grid-cols-2 gap-4">
|
| 93 |
{campaigns.map(c => (
|
| 94 |
<div key={c.id} className="rounded-xl bg-surface-card border border-hairline-dark p-5 hover:border-brand-yellow/30 transition-all group cursor-pointer">
|
|
|
|
| 107 |
</div>
|
| 108 |
))}
|
| 109 |
</div>
|
| 110 |
+
|
| 111 |
+
{/* Create Campaign Modal */}
|
| 112 |
+
{open && (
|
| 113 |
+
<div className="fixed inset-0 z-50 flex items-center justify-center bg-black/60 backdrop-blur-sm">
|
| 114 |
+
<div className="w-full max-w-lg bg-surface-card border border-hairline-dark rounded-2xl shadow-2xl p-6">
|
| 115 |
+
<div className="flex items-center justify-between mb-5">
|
| 116 |
+
<h2 className="text-title-md text-[#eaecef]">New Torque Campaign</h2>
|
| 117 |
+
<button onClick={() => setOpen(false)} className="p-1.5 rounded-lg hover:bg-surface-elevated text-muted hover:text-[#eaecef] transition"><X className="w-4 h-4"/></button>
|
| 118 |
+
</div>
|
| 119 |
+
|
| 120 |
+
{result ? (
|
| 121 |
+
<div className={cn('rounded-xl p-5 flex flex-col items-center gap-3 text-center', result.success ? 'bg-trading-up/10 border border-trading-up/30' : 'bg-trading-down/10 border border-trading-down/30')}>
|
| 122 |
+
{result.success ? <CheckCircle2 className="w-8 h-8 text-trading-up" /> : <AlertCircle className="w-8 h-8 text-trading-down" />}
|
| 123 |
+
<p className="text-title-sm text-[#eaecef]">{result.success ? 'Campaign Created!' : 'Creation Failed'}</p>
|
| 124 |
+
{result.campaignId && <p className="font-mono text-caption text-muted">ID: {result.campaignId}</p>}
|
| 125 |
+
{result.error && <p className="text-body-sm text-trading-down">{result.error}</p>}
|
| 126 |
+
<button onClick={() => { setResult(null); if (result.success) setOpen(false) }} className="mt-2 px-4 py-2 rounded-md bg-brand-yellow text-ink text-button font-semibold hover:bg-brand-yellow-active transition">
|
| 127 |
+
{result.success ? 'Done' : 'Try Again'}
|
| 128 |
+
</button>
|
| 129 |
+
</div>
|
| 130 |
+
) : (
|
| 131 |
+
<div className="space-y-4">
|
| 132 |
+
<div>
|
| 133 |
+
<label className="text-caption text-muted mb-1.5 block">Campaign Type</label>
|
| 134 |
+
<div className="grid grid-cols-2 gap-2">
|
| 135 |
+
{CAMPAIGN_TYPES.map(t => (
|
| 136 |
+
<button key={t.value} onClick={() => setForm(f => ({...f, type: t.value}))}
|
| 137 |
+
className={cn('p-3 rounded-xl border text-left transition-all', form.type === t.value ? 'border-brand-yellow/50 bg-brand-yellow/5' : 'border-hairline-dark hover:border-brand-yellow/20')}>
|
| 138 |
+
<p className="text-body-sm font-semibold text-[#eaecef]">{t.label}</p>
|
| 139 |
+
<p className="text-caption text-muted mt-0.5">{t.desc}</p>
|
| 140 |
+
</button>
|
| 141 |
+
))}
|
| 142 |
+
</div>
|
| 143 |
+
</div>
|
| 144 |
+
<div>
|
| 145 |
+
<label className="text-caption text-muted mb-1.5 block">Campaign Name</label>
|
| 146 |
+
<input value={form.name} onChange={e => setForm(f => ({...f, name: e.target.value}))} placeholder="Anti-Churn Gift Drop Q2" className="w-full h-9 px-3 rounded-lg bg-surface-elevated border border-hairline-dark text-body-sm text-[#eaecef] placeholder:text-muted focus:outline-none focus:ring-1 focus:ring-brand-yellow/50"/>
|
| 147 |
+
</div>
|
| 148 |
+
<div>
|
| 149 |
+
<label className="text-caption text-muted mb-1.5 block">Description (optional)</label>
|
| 150 |
+
<input value={form.description} onChange={e => setForm(f => ({...f, description: e.target.value}))} placeholder="Targets wallets with churn_risk_high event" className="w-full h-9 px-3 rounded-lg bg-surface-elevated border border-hairline-dark text-body-sm text-[#eaecef] placeholder:text-muted focus:outline-none focus:ring-1 focus:ring-brand-yellow/50"/>
|
| 151 |
+
</div>
|
| 152 |
+
<div className="grid grid-cols-2 gap-3">
|
| 153 |
+
<div>
|
| 154 |
+
<label className="text-caption text-muted mb-1.5 block">Budget (USDC)</label>
|
| 155 |
+
<input type="number" value={form.budget} onChange={e => setForm(f => ({...f, budget: e.target.value}))} placeholder="1000" className="w-full h-9 px-3 rounded-lg bg-surface-elevated border border-hairline-dark text-body-sm text-[#eaecef] placeholder:text-muted focus:outline-none focus:ring-1 focus:ring-brand-yellow/50"/>
|
| 156 |
+
</div>
|
| 157 |
+
<div>
|
| 158 |
+
<label className="text-caption text-muted mb-1.5 block">Trigger Event</label>
|
| 159 |
+
<select value={form.eventName} onChange={e => setForm(f => ({...f, eventName: e.target.value}))} className="w-full h-9 px-3 rounded-lg bg-surface-elevated border border-hairline-dark text-body-sm text-[#eaecef] focus:outline-none focus:ring-1 focus:ring-brand-yellow/50">
|
| 160 |
+
{EVENT_OPTIONS.map(e => <option key={e} value={e}>{e}</option>)}
|
| 161 |
+
</select>
|
| 162 |
+
</div>
|
| 163 |
+
</div>
|
| 164 |
+
<div className="flex gap-3 pt-1">
|
| 165 |
+
<button onClick={() => setOpen(false)} className="flex-1 py-2.5 rounded-md border border-hairline-dark text-body-sm text-muted hover:text-[#eaecef] hover:border-brand-yellow/20 transition">Cancel</button>
|
| 166 |
+
<button onClick={handleCreate} disabled={creating || !form.name.trim() || !form.budget}
|
| 167 |
+
className="flex-1 py-2.5 rounded-md bg-brand-yellow text-ink text-button font-semibold hover:bg-brand-yellow-active transition disabled:opacity-50 flex items-center justify-center gap-2">
|
| 168 |
+
{creating && <Loader2 className="w-4 h-4 animate-spin"/>}
|
| 169 |
+
{creating ? 'Creating...' : 'Create on Torque'}
|
| 170 |
+
</button>
|
| 171 |
+
</div>
|
| 172 |
+
</div>
|
| 173 |
+
)}
|
| 174 |
+
</div>
|
| 175 |
+
</div>
|
| 176 |
+
)}
|
| 177 |
</div>
|
| 178 |
)
|
| 179 |
}
|
|
@@ -1,14 +1,17 @@
|
|
| 1 |
import { Sidebar } from '@/components/layout/Sidebar'
|
| 2 |
import { Topbar } from '@/components/layout/Topbar'
|
|
|
|
| 3 |
|
| 4 |
export default function DashboardLayout({ children }: { children: React.ReactNode }) {
|
| 5 |
return (
|
| 6 |
-
<
|
| 7 |
-
<
|
| 8 |
-
|
| 9 |
-
<
|
| 10 |
-
|
|
|
|
|
|
|
| 11 |
</div>
|
| 12 |
-
</
|
| 13 |
)
|
| 14 |
}
|
|
|
|
| 1 |
import { Sidebar } from '@/components/layout/Sidebar'
|
| 2 |
import { Topbar } from '@/components/layout/Topbar'
|
| 3 |
+
import { ToastProvider } from '@/components/ui/Toast'
|
| 4 |
|
| 5 |
export default function DashboardLayout({ children }: { children: React.ReactNode }) {
|
| 6 |
return (
|
| 7 |
+
<ToastProvider>
|
| 8 |
+
<div className="flex h-screen overflow-hidden bg-canvas-dark">
|
| 9 |
+
<Sidebar />
|
| 10 |
+
<div className="flex flex-col flex-1 min-w-0">
|
| 11 |
+
<Topbar />
|
| 12 |
+
<main className="flex-1 overflow-y-auto">{children}</main>
|
| 13 |
+
</div>
|
| 14 |
</div>
|
| 15 |
+
</ToastProvider>
|
| 16 |
)
|
| 17 |
}
|
|
@@ -1,13 +1,15 @@
|
|
| 1 |
'use client'
|
| 2 |
|
| 3 |
import { AreaChart, Area, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer, PieChart, Pie, Cell } from 'recharts'
|
| 4 |
-
import { Users, ShieldAlert, HeartPulse, TrendingUp, Bot, Flame, ArrowUpRight, ArrowDownRight } from 'lucide-react'
|
| 5 |
import { StatCard } from '@/components/ui/StatCard'
|
| 6 |
import { CampaignBadge } from '@/components/ui/CampaignBadge'
|
| 7 |
import { AgentFeed } from '@/components/ui/AgentFeed'
|
|
|
|
| 8 |
import { cn, fmtNum, fmtUsd } from '@/lib/utils'
|
| 9 |
import { stats, events, campaigns, retentionData, churnData, protocols } from '@/lib/mock-data'
|
| 10 |
import type { CampaignType } from '@/lib/types'
|
|
|
|
| 11 |
|
| 12 |
const Tip = ({ active, payload, label }: any) => {
|
| 13 |
if (!active || !payload?.length) return null
|
|
@@ -23,13 +25,151 @@ const riskDist = [
|
|
| 23 |
{ name: 'Critical', value: 5, color: '#f6465d' },
|
| 24 |
]
|
| 25 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 26 |
export default function DashboardPage() {
|
| 27 |
const active = campaigns.filter(c => c.status === 'active')
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 28 |
return (
|
| 29 |
<div className="p-6 space-y-6">
|
| 30 |
<div className="flex items-center justify-between">
|
| 31 |
-
<div>
|
| 32 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 33 |
</div>
|
| 34 |
|
| 35 |
<div className="grid grid-cols-2 lg:grid-cols-4 gap-4">
|
|
@@ -39,6 +179,42 @@ export default function DashboardPage() {
|
|
| 39 |
<StatCard title="ROI" value={stats.roi + '%'} change={15.2} changeLabel="vs last week" icon={TrendingUp} variant="yellow" />
|
| 40 |
</div>
|
| 41 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 42 |
<AgentFeed />
|
| 43 |
|
| 44 |
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
|
|
@@ -48,9 +224,10 @@ export default function DashboardPage() {
|
|
| 48 |
<div className="flex items-center gap-1"><ArrowUpRight className="w-4 h-4 text-trading-up" /><span className="font-mono text-num-sm text-trading-up">+9.6%</span></div>
|
| 49 |
</div>
|
| 50 |
<ResponsiveContainer width="100%" height={220}>
|
| 51 |
-
<AreaChart data={retentionData}>
|
| 52 |
-
<
|
| 53 |
-
<
|
|
|
|
| 54 |
</AreaChart>
|
| 55 |
</ResponsiveContainer>
|
| 56 |
</div>
|
|
@@ -60,9 +237,10 @@ export default function DashboardPage() {
|
|
| 60 |
<div className="flex items-center gap-1"><ArrowDownRight className="w-4 h-4 text-trading-up" /><span className="font-mono text-num-sm text-trading-up">-4.3%</span></div>
|
| 61 |
</div>
|
| 62 |
<ResponsiveContainer width="100%" height={220}>
|
| 63 |
-
<AreaChart data={churnData}>
|
| 64 |
-
<
|
| 65 |
-
<
|
|
|
|
| 66 |
</AreaChart>
|
| 67 |
</ResponsiveContainer>
|
| 68 |
</div>
|
|
@@ -72,17 +250,17 @@ export default function DashboardPage() {
|
|
| 72 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-5">
|
| 73 |
<h3 className="text-title-sm mb-4">Risk Distribution</h3>
|
| 74 |
<div className="flex items-center gap-6">
|
| 75 |
-
<ResponsiveContainer width={140} height={140}><PieChart><Pie data={riskDist} innerRadius={40} outerRadius={65} paddingAngle={3} dataKey="value">{riskDist.map((e,i) => <Cell key={i} fill={e.color}/>)}</Pie></PieChart></ResponsiveContainer>
|
| 76 |
-
<div className="space-y-2">{riskDist.map(r => (<div key={r.name} className="flex items-center gap-2"><div className="w-2.5 h-2.5 rounded-full" style={{backgroundColor:r.color}}/><span className="text-caption text-muted">{r.name}</span><span className="font-mono text-caption text-[#eaecef] ml-auto">{r.value}%</span></div>))}</div>
|
| 77 |
</div>
|
| 78 |
</div>
|
| 79 |
|
| 80 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-5">
|
| 81 |
<h3 className="text-title-sm mb-4">Recent Events</h3>
|
| 82 |
-
<div className="space-y-3">{events.slice(0,5).map(e => (
|
| 83 |
<div key={e.id} className="flex items-start gap-3 pb-3 border-b border-hairline-dark/50 last:border-0">
|
| 84 |
<div className={cn('w-2 h-2 rounded-full mt-1.5 flex-shrink-0', e.resolved ? 'bg-trading-up' : 'bg-trading-down animate-pulse')} />
|
| 85 |
-
<div className="min-w-0"><p className="text-body-sm text-[#eaecef] truncate">{e.eventType.replace(/_/g,' ').toUpperCase()}</p><p className="text-caption text-muted font-mono">{e.wallet}</p></div>
|
| 86 |
<span className="text-caption text-muted ml-auto whitespace-nowrap">{e.timestamp}</span>
|
| 87 |
</div>
|
| 88 |
))}</div>
|
|
@@ -90,7 +268,7 @@ export default function DashboardPage() {
|
|
| 90 |
|
| 91 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-5">
|
| 92 |
<h3 className="text-title-sm mb-4">Active Campaigns</h3>
|
| 93 |
-
<div className="space-y-3">{active.slice(0,4).map(c => (
|
| 94 |
<div key={c.id} className="flex items-center gap-3 pb-3 border-b border-hairline-dark/50 last:border-0">
|
| 95 |
<CampaignBadge type={c.type as CampaignType} />
|
| 96 |
<div className="min-w-0 flex-1"><p className="text-body-sm text-[#eaecef] truncate">{c.name}</p><p className="text-caption text-muted font-mono">{fmtNum(c.participantCount)} users</p></div>
|
|
@@ -111,12 +289,12 @@ export default function DashboardPage() {
|
|
| 111 |
<th className="text-right text-caption text-muted uppercase tracking-wider px-4 py-3">Avg Streak</th>
|
| 112 |
</tr></thead><tbody>{protocols.map(p => (
|
| 113 |
<tr key={p.protocol} className="border-b border-hairline-dark/50 hover:bg-surface-hover transition-colors">
|
| 114 |
-
<td className="px-4 py-3"><div className="flex items-center gap-2"><div className="w-3 h-3 rounded-full" style={{backgroundColor:p.color}}/><span className="text-body-md font-medium">{p.protocol}</span></div></td>
|
| 115 |
<td className="text-right px-4 py-3 font-mono text-num-md tabular-nums">{fmtUsd(p.volume)}</td>
|
| 116 |
<td className="text-right px-4 py-3 font-mono text-num-md tabular-nums">{fmtNum(p.users)}</td>
|
| 117 |
<td className="text-right px-4 py-3"><span className={cn('font-mono text-num-sm', p.churnRate <= 4 ? 'text-trading-up' : 'text-trading-down')}>{p.churnRate}%</span></td>
|
| 118 |
<td className="text-right px-4 py-3"><span className={cn('font-mono text-num-sm', p.retentionRate >= 70 ? 'text-trading-up' : 'text-brand-yellow')}>{p.retentionRate}%</span></td>
|
| 119 |
-
<td className="text-right px-4 py-3"><div className="flex items-center justify-end gap-1"><Flame className="w-3.5 h-3.5 text-brand-yellow"/><span className="font-mono text-num-sm">{p.avgStreak}d</span></div></td>
|
| 120 |
</tr>
|
| 121 |
))}</tbody></table>
|
| 122 |
</div>
|
|
|
|
| 1 |
'use client'
|
| 2 |
|
| 3 |
import { AreaChart, Area, XAxis, YAxis, CartesianGrid, Tooltip, ResponsiveContainer, PieChart, Pie, Cell } from 'recharts'
|
| 4 |
+
import { Users, ShieldAlert, HeartPulse, TrendingUp, Bot, Flame, ArrowUpRight, ArrowDownRight, Scan, Zap, ExternalLink, Timer, ToggleLeft, ToggleRight } from 'lucide-react'
|
| 5 |
import { StatCard } from '@/components/ui/StatCard'
|
| 6 |
import { CampaignBadge } from '@/components/ui/CampaignBadge'
|
| 7 |
import { AgentFeed } from '@/components/ui/AgentFeed'
|
| 8 |
+
import { useToast } from '@/components/ui/Toast'
|
| 9 |
import { cn, fmtNum, fmtUsd } from '@/lib/utils'
|
| 10 |
import { stats, events, campaigns, retentionData, churnData, protocols } from '@/lib/mock-data'
|
| 11 |
import type { CampaignType } from '@/lib/types'
|
| 12 |
+
import { useState, useEffect, useCallback, useRef } from 'react'
|
| 13 |
|
| 14 |
const Tip = ({ active, payload, label }: any) => {
|
| 15 |
if (!active || !payload?.length) return null
|
|
|
|
| 25 |
{ name: 'Critical', value: 5, color: '#f6465d' },
|
| 26 |
]
|
| 27 |
|
| 28 |
+
interface LiveEvent {
|
| 29 |
+
ingestionId: string; wallet: string; eventName: string
|
| 30 |
+
risk?: string; score?: number; firedAt: string; source: string
|
| 31 |
+
}
|
| 32 |
+
|
| 33 |
+
const RISK_COLOR: Record<string, string> = {
|
| 34 |
+
critical: '#f6465d', high: '#ff9500', medium: '#FCD535', low: '#0ecb81', safe: '#2dbdb6'
|
| 35 |
+
}
|
| 36 |
+
|
| 37 |
+
const AUTO_SCAN_INTERVAL = 30
|
| 38 |
+
|
| 39 |
export default function DashboardPage() {
|
| 40 |
const active = campaigns.filter(c => c.status === 'active')
|
| 41 |
+
const { fire: toast } = useToast()
|
| 42 |
+
const [scanning, setScanning] = useState(false)
|
| 43 |
+
const [autoScan, setAutoScan] = useState(false)
|
| 44 |
+
const [countdown, setCountdown] = useState(AUTO_SCAN_INTERVAL)
|
| 45 |
+
const [liveEvents, setLiveEvents] = useState<LiveEvent[]>([])
|
| 46 |
+
const [sessionCount, setSessionCount] = useState(0)
|
| 47 |
+
const [torqueStatus, setTorqueStatus] = useState<'connected' | 'unconfigured' | null>(null)
|
| 48 |
+
const prevCountRef = useRef(0)
|
| 49 |
+
|
| 50 |
+
useEffect(() => {
|
| 51 |
+
fetch('/api/torque/status').then(r => r.json()).then(d => {
|
| 52 |
+
setTorqueStatus(d.configured ? 'connected' : 'unconfigured')
|
| 53 |
+
setSessionCount(d.sessionEvents || 0)
|
| 54 |
+
}).catch(() => setTorqueStatus('unconfigured'))
|
| 55 |
+
}, [])
|
| 56 |
+
|
| 57 |
+
useEffect(() => {
|
| 58 |
+
const poll = () => {
|
| 59 |
+
fetch('/api/torque/events/recent?limit=8').then(r => r.json()).then(d => {
|
| 60 |
+
const newEvents: LiveEvent[] = d.events || []
|
| 61 |
+
const newCount: number = d.total || 0
|
| 62 |
+
if (newCount > prevCountRef.current && prevCountRef.current > 0) {
|
| 63 |
+
const latest = newEvents[0]
|
| 64 |
+
if (latest) {
|
| 65 |
+
toast({
|
| 66 |
+
type: 'event',
|
| 67 |
+
title: latest.eventName.replace(/_/g, ' ').toUpperCase(),
|
| 68 |
+
body: `${latest.wallet.slice(0, 8)}...${latest.wallet.slice(-4)} · ${latest.ingestionId.slice(0, 8)}`,
|
| 69 |
+
})
|
| 70 |
+
}
|
| 71 |
+
}
|
| 72 |
+
prevCountRef.current = newCount
|
| 73 |
+
setLiveEvents(newEvents)
|
| 74 |
+
setSessionCount(newCount)
|
| 75 |
+
}).catch(() => {})
|
| 76 |
+
}
|
| 77 |
+
poll()
|
| 78 |
+
const iv = setInterval(poll, 5000)
|
| 79 |
+
return () => clearInterval(iv)
|
| 80 |
+
}, [toast])
|
| 81 |
+
|
| 82 |
+
const triggerScan = useCallback(async () => {
|
| 83 |
+
if (scanning) return
|
| 84 |
+
setScanning(true)
|
| 85 |
+
try {
|
| 86 |
+
const res = await fetch('/api/agent/scan', { method: 'POST' })
|
| 87 |
+
const data = await res.json()
|
| 88 |
+
if (data.configured) {
|
| 89 |
+
toast({ type: 'success', title: `Scan complete — ${data.count} at-risk wallets`, body: `${data.detections?.filter((d: any) => d.eventSent).length || 0} events fired to Torque` })
|
| 90 |
+
} else {
|
| 91 |
+
toast({ type: 'error', title: 'Torque not configured', body: 'Set TORQUE_INGEST_KEY in .env.local' })
|
| 92 |
+
}
|
| 93 |
+
const d = await fetch('/api/torque/events/recent?limit=8').then(r => r.json())
|
| 94 |
+
setLiveEvents(d.events || [])
|
| 95 |
+
setSessionCount(d.total || 0)
|
| 96 |
+
} catch {
|
| 97 |
+
toast({ type: 'error', title: 'Scan failed', body: 'Check server logs' })
|
| 98 |
+
} finally {
|
| 99 |
+
setScanning(false)
|
| 100 |
+
setCountdown(AUTO_SCAN_INTERVAL)
|
| 101 |
+
}
|
| 102 |
+
}, [scanning, toast])
|
| 103 |
+
|
| 104 |
+
// Auto-scan countdown + trigger
|
| 105 |
+
useEffect(() => {
|
| 106 |
+
if (!autoScan) { setCountdown(AUTO_SCAN_INTERVAL); return }
|
| 107 |
+
const tick = setInterval(() => {
|
| 108 |
+
setCountdown(c => {
|
| 109 |
+
if (c <= 1) { triggerScan(); return AUTO_SCAN_INTERVAL }
|
| 110 |
+
return c - 1
|
| 111 |
+
})
|
| 112 |
+
}, 1000)
|
| 113 |
+
return () => clearInterval(tick)
|
| 114 |
+
}, [autoScan, triggerScan])
|
| 115 |
+
|
| 116 |
+
// Keyboard shortcut: S = scan
|
| 117 |
+
useEffect(() => {
|
| 118 |
+
const handler = (e: KeyboardEvent) => {
|
| 119 |
+
if (e.key === 's' && !e.metaKey && !e.ctrlKey && e.target === document.body) triggerScan()
|
| 120 |
+
}
|
| 121 |
+
document.addEventListener('keydown', handler)
|
| 122 |
+
return () => document.removeEventListener('keydown', handler)
|
| 123 |
+
}, [triggerScan])
|
| 124 |
+
|
| 125 |
return (
|
| 126 |
<div className="p-6 space-y-6">
|
| 127 |
<div className="flex items-center justify-between">
|
| 128 |
+
<div>
|
| 129 |
+
<h1 className="text-display-sm text-[#eaecef]">Dashboard</h1>
|
| 130 |
+
<p className="text-body-md text-muted mt-1">Real-time churn detection and autonomous retention</p>
|
| 131 |
+
</div>
|
| 132 |
+
<div className="flex items-center gap-3">
|
| 133 |
+
{torqueStatus !== null && (
|
| 134 |
+
<div className={cn('flex items-center gap-2 px-3 py-1.5 rounded-lg border text-caption font-semibold',
|
| 135 |
+
torqueStatus === 'connected'
|
| 136 |
+
? 'bg-trading-up/10 border-trading-up/30 text-trading-up'
|
| 137 |
+
: 'bg-trading-down/10 border-trading-down/30 text-trading-down')}>
|
| 138 |
+
<div className={cn('w-2 h-2 rounded-full', torqueStatus === 'connected' ? 'bg-trading-up animate-pulse' : 'bg-trading-down')} />
|
| 139 |
+
{torqueStatus === 'connected' ? 'Torque Connected' : 'Torque Unconfigured'}
|
| 140 |
+
</div>
|
| 141 |
+
)}
|
| 142 |
+
{sessionCount > 0 && (
|
| 143 |
+
<div className="flex items-center gap-2 px-3 py-1.5 rounded-lg border border-brand-yellow/30 bg-brand-yellow/5">
|
| 144 |
+
<Zap className="w-3.5 h-3.5 text-brand-yellow" />
|
| 145 |
+
<span className="text-caption text-brand-yellow font-semibold tabular-nums">{sessionCount} live events</span>
|
| 146 |
+
</div>
|
| 147 |
+
)}
|
| 148 |
+
{/* Auto-scan toggle */}
|
| 149 |
+
<button
|
| 150 |
+
onClick={() => setAutoScan(v => !v)}
|
| 151 |
+
className={cn('flex items-center gap-2 px-3 py-1.5 rounded-lg border text-caption font-semibold transition-all',
|
| 152 |
+
autoScan ? 'bg-brand-turquoise/10 border-brand-turquoise/30 text-brand-turquoise' : 'bg-surface-card border-hairline-dark text-muted hover:text-[#eaecef]')}
|
| 153 |
+
>
|
| 154 |
+
{autoScan ? <ToggleRight className="w-4 h-4" /> : <ToggleLeft className="w-4 h-4" />}
|
| 155 |
+
Auto
|
| 156 |
+
{autoScan && (
|
| 157 |
+
<span className="flex items-center gap-1">
|
| 158 |
+
<Timer className="w-3 h-3" />
|
| 159 |
+
<span className="font-mono tabular-nums">{countdown}s</span>
|
| 160 |
+
</span>
|
| 161 |
+
)}
|
| 162 |
+
</button>
|
| 163 |
+
<button
|
| 164 |
+
onClick={triggerScan}
|
| 165 |
+
disabled={scanning}
|
| 166 |
+
className="flex items-center gap-2 px-4 py-2 rounded-md bg-brand-yellow text-ink text-button font-semibold hover:bg-brand-yellow-active transition disabled:opacity-60"
|
| 167 |
+
title="Press S to scan"
|
| 168 |
+
>
|
| 169 |
+
<Scan className={cn('w-4 h-4', scanning && 'animate-spin')} />
|
| 170 |
+
{scanning ? 'Scanning...' : 'Scan Now'}
|
| 171 |
+
</button>
|
| 172 |
+
</div>
|
| 173 |
</div>
|
| 174 |
|
| 175 |
<div className="grid grid-cols-2 lg:grid-cols-4 gap-4">
|
|
|
|
| 179 |
<StatCard title="ROI" value={stats.roi + '%'} change={15.2} changeLabel="vs last week" icon={TrendingUp} variant="yellow" />
|
| 180 |
</div>
|
| 181 |
|
| 182 |
+
{/* Live Torque Events Strip */}
|
| 183 |
+
{liveEvents.length > 0 && (
|
| 184 |
+
<div className="rounded-xl bg-surface-card border border-trading-up/20 overflow-hidden">
|
| 185 |
+
<div className="flex items-center gap-3 px-5 py-3 border-b border-trading-up/10 bg-trading-up/5">
|
| 186 |
+
<div className="w-2 h-2 rounded-full bg-trading-up animate-pulse" />
|
| 187 |
+
<span className="text-caption text-trading-up font-semibold uppercase tracking-wider">Live Torque Events</span>
|
| 188 |
+
<span className="text-caption text-muted">{sessionCount} fired this session</span>
|
| 189 |
+
<a href="https://platform.torque.so" target="_blank" rel="noreferrer" className="ml-auto flex items-center gap-1 text-caption text-trading-up/70 hover:text-trading-up transition">
|
| 190 |
+
<ExternalLink className="w-3 h-3" />View on Torque
|
| 191 |
+
</a>
|
| 192 |
+
</div>
|
| 193 |
+
<div className="divide-y divide-hairline-dark/30">
|
| 194 |
+
{liveEvents.slice(0, 5).map((e, i) => (
|
| 195 |
+
<div key={e.ingestionId} className={cn('flex items-center gap-4 px-5 py-2.5 group', i === 0 && 'bg-trading-up/5')}>
|
| 196 |
+
<div className="w-2 h-2 rounded-full flex-shrink-0" style={{ backgroundColor: RISK_COLOR[e.risk || 'safe'] || '#0ecb81' }} />
|
| 197 |
+
<code className="font-mono text-caption text-brand-yellow">{e.eventName}</code>
|
| 198 |
+
<span className="font-mono text-caption text-muted">{e.wallet.slice(0, 8)}...{e.wallet.slice(-4)}</span>
|
| 199 |
+
{e.score !== undefined && <span className="font-mono text-caption text-muted">score={e.score}</span>}
|
| 200 |
+
<span className="ml-auto font-mono text-[10px] text-muted/60 group-hover:text-muted transition">{e.ingestionId.slice(0, 12)}...</span>
|
| 201 |
+
<span className="text-[10px] text-trading-up font-semibold uppercase">LIVE</span>
|
| 202 |
+
</div>
|
| 203 |
+
))}
|
| 204 |
+
</div>
|
| 205 |
+
</div>
|
| 206 |
+
)}
|
| 207 |
+
|
| 208 |
+
{/* Auto-scan active banner */}
|
| 209 |
+
{autoScan && (
|
| 210 |
+
<div className="rounded-xl bg-brand-turquoise/5 border border-brand-turquoise/20 px-5 py-3 flex items-center gap-3">
|
| 211 |
+
<div className="w-2 h-2 rounded-full bg-brand-turquoise animate-pulse" />
|
| 212 |
+
<span className="text-caption text-brand-turquoise font-semibold">Auto-scan active</span>
|
| 213 |
+
<span className="text-caption text-muted">Next scan in <span className="font-mono text-brand-turquoise">{countdown}s</span> — firing real Torque events for at-risk wallets</span>
|
| 214 |
+
<button onClick={() => setAutoScan(false)} className="ml-auto text-caption text-muted hover:text-trading-down transition">Stop</button>
|
| 215 |
+
</div>
|
| 216 |
+
)}
|
| 217 |
+
|
| 218 |
<AgentFeed />
|
| 219 |
|
| 220 |
<div className="grid grid-cols-1 lg:grid-cols-2 gap-6">
|
|
|
|
| 224 |
<div className="flex items-center gap-1"><ArrowUpRight className="w-4 h-4 text-trading-up" /><span className="font-mono text-num-sm text-trading-up">+9.6%</span></div>
|
| 225 |
</div>
|
| 226 |
<ResponsiveContainer width="100%" height={220}>
|
| 227 |
+
<AreaChart data={retentionData}>
|
| 228 |
+
<defs><linearGradient id="rg" x1="0" y1="0" x2="0" y2="1"><stop offset="5%" stopColor="#0ecb81" stopOpacity={0.3} /><stop offset="95%" stopColor="#0ecb81" stopOpacity={0} /></linearGradient></defs>
|
| 229 |
+
<CartesianGrid strokeDasharray="3 3" stroke="#2b3139" /><XAxis dataKey="date" tick={{ fill: '#707a8a', fontSize: 11 }} axisLine={{ stroke: '#2b3139' }} /><YAxis tick={{ fill: '#707a8a', fontSize: 11 }} axisLine={{ stroke: '#2b3139' }} domain={[50, 75]} />
|
| 230 |
+
<Tooltip content={<Tip />} /><Area type="monotone" dataKey="value" stroke="#0ecb81" fill="url(#rg)" strokeWidth={2} />
|
| 231 |
</AreaChart>
|
| 232 |
</ResponsiveContainer>
|
| 233 |
</div>
|
|
|
|
| 237 |
<div className="flex items-center gap-1"><ArrowDownRight className="w-4 h-4 text-trading-up" /><span className="font-mono text-num-sm text-trading-up">-4.3%</span></div>
|
| 238 |
</div>
|
| 239 |
<ResponsiveContainer width="100%" height={220}>
|
| 240 |
+
<AreaChart data={churnData}>
|
| 241 |
+
<defs><linearGradient id="cg" x1="0" y1="0" x2="0" y2="1"><stop offset="5%" stopColor="#f6465d" stopOpacity={0.3} /><stop offset="95%" stopColor="#f6465d" stopOpacity={0} /></linearGradient></defs>
|
| 242 |
+
<CartesianGrid strokeDasharray="3 3" stroke="#2b3139" /><XAxis dataKey="date" tick={{ fill: '#707a8a', fontSize: 11 }} axisLine={{ stroke: '#2b3139' }} /><YAxis tick={{ fill: '#707a8a', fontSize: 11 }} axisLine={{ stroke: '#2b3139' }} domain={[0, 10]} />
|
| 243 |
+
<Tooltip content={<Tip />} /><Area type="monotone" dataKey="value" stroke="#f6465d" fill="url(#cg)" strokeWidth={2} />
|
| 244 |
</AreaChart>
|
| 245 |
</ResponsiveContainer>
|
| 246 |
</div>
|
|
|
|
| 250 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-5">
|
| 251 |
<h3 className="text-title-sm mb-4">Risk Distribution</h3>
|
| 252 |
<div className="flex items-center gap-6">
|
| 253 |
+
<ResponsiveContainer width={140} height={140}><PieChart><Pie data={riskDist} innerRadius={40} outerRadius={65} paddingAngle={3} dataKey="value">{riskDist.map((e, i) => <Cell key={i} fill={e.color} />)}</Pie></PieChart></ResponsiveContainer>
|
| 254 |
+
<div className="space-y-2">{riskDist.map(r => (<div key={r.name} className="flex items-center gap-2"><div className="w-2.5 h-2.5 rounded-full" style={{ backgroundColor: r.color }} /><span className="text-caption text-muted">{r.name}</span><span className="font-mono text-caption text-[#eaecef] ml-auto">{r.value}%</span></div>))}</div>
|
| 255 |
</div>
|
| 256 |
</div>
|
| 257 |
|
| 258 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-5">
|
| 259 |
<h3 className="text-title-sm mb-4">Recent Events</h3>
|
| 260 |
+
<div className="space-y-3">{events.slice(0, 5).map(e => (
|
| 261 |
<div key={e.id} className="flex items-start gap-3 pb-3 border-b border-hairline-dark/50 last:border-0">
|
| 262 |
<div className={cn('w-2 h-2 rounded-full mt-1.5 flex-shrink-0', e.resolved ? 'bg-trading-up' : 'bg-trading-down animate-pulse')} />
|
| 263 |
+
<div className="min-w-0"><p className="text-body-sm text-[#eaecef] truncate">{e.eventType.replace(/_/g, ' ').toUpperCase()}</p><p className="text-caption text-muted font-mono">{e.wallet}</p></div>
|
| 264 |
<span className="text-caption text-muted ml-auto whitespace-nowrap">{e.timestamp}</span>
|
| 265 |
</div>
|
| 266 |
))}</div>
|
|
|
|
| 268 |
|
| 269 |
<div className="rounded-xl bg-surface-card border border-hairline-dark p-5">
|
| 270 |
<h3 className="text-title-sm mb-4">Active Campaigns</h3>
|
| 271 |
+
<div className="space-y-3">{active.slice(0, 4).map(c => (
|
| 272 |
<div key={c.id} className="flex items-center gap-3 pb-3 border-b border-hairline-dark/50 last:border-0">
|
| 273 |
<CampaignBadge type={c.type as CampaignType} />
|
| 274 |
<div className="min-w-0 flex-1"><p className="text-body-sm text-[#eaecef] truncate">{c.name}</p><p className="text-caption text-muted font-mono">{fmtNum(c.participantCount)} users</p></div>
|
|
|
|
| 289 |
<th className="text-right text-caption text-muted uppercase tracking-wider px-4 py-3">Avg Streak</th>
|
| 290 |
</tr></thead><tbody>{protocols.map(p => (
|
| 291 |
<tr key={p.protocol} className="border-b border-hairline-dark/50 hover:bg-surface-hover transition-colors">
|
| 292 |
+
<td className="px-4 py-3"><div className="flex items-center gap-2"><div className="w-3 h-3 rounded-full" style={{ backgroundColor: p.color }} /><span className="text-body-md font-medium">{p.protocol}</span></div></td>
|
| 293 |
<td className="text-right px-4 py-3 font-mono text-num-md tabular-nums">{fmtUsd(p.volume)}</td>
|
| 294 |
<td className="text-right px-4 py-3 font-mono text-num-md tabular-nums">{fmtNum(p.users)}</td>
|
| 295 |
<td className="text-right px-4 py-3"><span className={cn('font-mono text-num-sm', p.churnRate <= 4 ? 'text-trading-up' : 'text-trading-down')}>{p.churnRate}%</span></td>
|
| 296 |
<td className="text-right px-4 py-3"><span className={cn('font-mono text-num-sm', p.retentionRate >= 70 ? 'text-trading-up' : 'text-brand-yellow')}>{p.retentionRate}%</span></td>
|
| 297 |
+
<td className="text-right px-4 py-3"><div className="flex items-center justify-end gap-1"><Flame className="w-3.5 h-3.5 text-brand-yellow" /><span className="font-mono text-num-sm">{p.avgStreak}d</span></div></td>
|
| 298 |
</tr>
|
| 299 |
))}</tbody></table>
|
| 300 |
</div>
|
|
@@ -2,39 +2,164 @@
|
|
| 2 |
import { cn, fmtUsd, shortAddr } from '@/lib/utils'
|
| 3 |
import { wallets } from '@/lib/mock-data'
|
| 4 |
import { RiskBadge } from '@/components/ui/RiskBadge'
|
| 5 |
-
import {
|
| 6 |
-
import {
|
| 7 |
-
import
|
|
|
|
| 8 |
|
| 9 |
type Filter = ChurnRisk | 'all'
|
| 10 |
-
const filters: Filter[] = ['all','critical','high','medium','low','safe']
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 11 |
|
| 12 |
export default function WalletsPage() {
|
|
|
|
| 13 |
const [f, setF] = useState<Filter>('all')
|
| 14 |
const [q, setQ] = useState('')
|
| 15 |
-
const [sort, setSort] = useState<'risk'|'volume'|'streak'>('risk')
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 16 |
const list = wallets
|
| 17 |
.filter(w => f === 'all' || w.churnRisk === f)
|
| 18 |
.filter(w => !q || w.address.toLowerCase().includes(q.toLowerCase()))
|
| 19 |
-
.sort((a,b) => sort === 'risk' ? b.riskScore - a.riskScore : sort === 'volume' ? b.totalVolume - a.totalVolume : b.streak - a.streak)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 20 |
|
| 21 |
return (
|
| 22 |
<div className="p-6 space-y-6">
|
| 23 |
-
<div
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 24 |
|
| 25 |
<div className="grid grid-cols-2 lg:grid-cols-6 gap-3">
|
| 26 |
{filters.map(r => (
|
| 27 |
-
<button key={r} onClick={() => setF(r)} className={cn('rounded-xl border p-3 text-center transition-all', f===r ? 'border-brand-yellow/40 bg-brand-yellow/5' : 'border-hairline-dark bg-surface-card hover:border-brand-yellow/20')}>
|
| 28 |
<span className="text-caption text-muted capitalize">{r}</span>
|
| 29 |
-
<p className="font-mono text-title-md text-[#eaecef] tabular-nums mt-0.5">{r==='all'?wallets.length:wallets.filter(w=>w.churnRisk===r).length}</p>
|
| 30 |
</button>
|
| 31 |
))}
|
| 32 |
</div>
|
| 33 |
|
| 34 |
<div className="flex items-center gap-3">
|
| 35 |
-
<div className="relative flex-1 max-w-md">
|
|
|
|
|
|
|
|
|
|
| 36 |
<div className="flex items-center gap-1 p-1 rounded-lg bg-surface-card border border-hairline-dark">
|
| 37 |
-
{(['risk','volume','streak'] as const).map(s => <button key={s} onClick={() => setSort(s)} className={cn('px-3 py-1.5 rounded-md text-caption transition capitalize', sort===s?'bg-brand-yellow text-ink font-semibold':'text-muted hover:text-[#eaecef]')}>{s}</button>)}
|
| 38 |
</div>
|
| 39 |
</div>
|
| 40 |
|
|
@@ -47,22 +172,69 @@ export default function WalletsPage() {
|
|
| 47 |
<th className="text-right text-caption text-muted uppercase tracking-wider px-5 py-3">Streak</th>
|
| 48 |
<th className="text-center text-caption text-muted uppercase tracking-wider px-5 py-3">Protocols</th>
|
| 49 |
<th className="text-right text-caption text-muted uppercase tracking-wider px-5 py-3">Last Active</th>
|
| 50 |
-
<th className="
|
| 51 |
-
</tr></thead><tbody>{list.map(w =>
|
| 52 |
-
|
| 53 |
-
|
| 54 |
-
|
| 55 |
-
|
| 56 |
-
|
| 57 |
-
<
|
| 58 |
-
|
| 59 |
-
|
| 60 |
-
|
| 61 |
-
|
| 62 |
-
|
| 63 |
-
|
| 64 |
-
|
| 65 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 66 |
</div>
|
| 67 |
</div>
|
| 68 |
)
|
|
|
|
| 2 |
import { cn, fmtUsd, shortAddr } from '@/lib/utils'
|
| 3 |
import { wallets } from '@/lib/mock-data'
|
| 4 |
import { RiskBadge } from '@/components/ui/RiskBadge'
|
| 5 |
+
import { useToast } from '@/components/ui/Toast'
|
| 6 |
+
import { Search, Flame, ExternalLink, ChevronRight, Zap, CheckCircle2, Loader2, Siren } from 'lucide-react'
|
| 7 |
+
import { useState, useCallback } from 'react'
|
| 8 |
+
import type { ChurnRisk, Wallet } from '@/lib/types'
|
| 9 |
|
| 10 |
type Filter = ChurnRisk | 'all'
|
| 11 |
+
const filters: Filter[] = ['all', 'critical', 'high', 'medium', 'low', 'safe']
|
| 12 |
+
|
| 13 |
+
const EVENT_MAP: Record<ChurnRisk, string> = {
|
| 14 |
+
critical: 'churn_risk_high', high: 'churn_risk_high',
|
| 15 |
+
medium: 'churn_risk_medium', low: 'inactivity_detected', safe: 'streak_maintained',
|
| 16 |
+
}
|
| 17 |
+
const ACTION_LABEL: Record<ChurnRisk, string> = {
|
| 18 |
+
critical: 'Send Gift', high: 'Enter Raffle', medium: 'Activate Rebate',
|
| 19 |
+
low: 'Flag Inactive', safe: 'Reward Streak',
|
| 20 |
+
}
|
| 21 |
+
const RISK_CAN_INTERVENE: ChurnRisk[] = ['critical', 'high', 'medium']
|
| 22 |
|
| 23 |
export default function WalletsPage() {
|
| 24 |
+
const { fire: toast } = useToast()
|
| 25 |
const [f, setF] = useState<Filter>('all')
|
| 26 |
const [q, setQ] = useState('')
|
| 27 |
+
const [sort, setSort] = useState<'risk' | 'volume' | 'streak'>('risk')
|
| 28 |
+
const [firing, setFiring] = useState<string | null>(null)
|
| 29 |
+
const [fired, setFired] = useState<Map<string, string>>(new Map())
|
| 30 |
+
const [bulkFiring, setBulkFiring] = useState(false)
|
| 31 |
+
const [bulkProgress, setBulkProgress] = useState<{ done: number; total: number } | null>(null)
|
| 32 |
+
|
| 33 |
const list = wallets
|
| 34 |
.filter(w => f === 'all' || w.churnRisk === f)
|
| 35 |
.filter(w => !q || w.address.toLowerCase().includes(q.toLowerCase()))
|
| 36 |
+
.sort((a, b) => sort === 'risk' ? b.riskScore - a.riskScore : sort === 'volume' ? b.totalVolume - a.totalVolume : b.streak - a.streak)
|
| 37 |
+
|
| 38 |
+
const criticalUnfired = wallets.filter(w =>
|
| 39 |
+
(w.churnRisk === 'critical' || w.churnRisk === 'high') && !fired.has(w.address)
|
| 40 |
+
)
|
| 41 |
+
|
| 42 |
+
const intervene = useCallback(async (w: Wallet) => {
|
| 43 |
+
if (firing || fired.has(w.address)) return
|
| 44 |
+
setFiring(w.address)
|
| 45 |
+
try {
|
| 46 |
+
const res = await fetch('/api/torque/events', {
|
| 47 |
+
method: 'POST',
|
| 48 |
+
headers: { 'Content-Type': 'application/json' },
|
| 49 |
+
body: JSON.stringify({
|
| 50 |
+
wallet: w.address,
|
| 51 |
+
eventName: EVENT_MAP[w.churnRisk],
|
| 52 |
+
data: { risk: w.churnRisk, score: w.riskScore, detectedBy: 'flowstate-dashboard' },
|
| 53 |
+
risk: w.churnRisk,
|
| 54 |
+
score: w.riskScore,
|
| 55 |
+
}),
|
| 56 |
+
})
|
| 57 |
+
const data = await res.json()
|
| 58 |
+
const eventId = data.eventId || 'sent'
|
| 59 |
+
setFired(prev => new Map(prev).set(w.address, eventId))
|
| 60 |
+
toast({
|
| 61 |
+
type: 'event',
|
| 62 |
+
title: `${ACTION_LABEL[w.churnRisk]} → ${shortAddr(w.address)}`,
|
| 63 |
+
body: `${EVENT_MAP[w.churnRisk]} · ${eventId.slice(0, 10)}`,
|
| 64 |
+
})
|
| 65 |
+
} catch {
|
| 66 |
+
setFired(prev => new Map(prev).set(w.address, 'error'))
|
| 67 |
+
toast({ type: 'error', title: 'Event failed', body: w.address.slice(0, 12) })
|
| 68 |
+
} finally {
|
| 69 |
+
setFiring(null)
|
| 70 |
+
}
|
| 71 |
+
}, [firing, fired, toast])
|
| 72 |
+
|
| 73 |
+
const bulkRescue = useCallback(async () => {
|
| 74 |
+
if (bulkFiring || criticalUnfired.length === 0) return
|
| 75 |
+
setBulkFiring(true)
|
| 76 |
+
setBulkProgress({ done: 0, total: criticalUnfired.length })
|
| 77 |
+
|
| 78 |
+
const targets = criticalUnfired.map(w => ({
|
| 79 |
+
wallet: w.address,
|
| 80 |
+
eventName: EVENT_MAP[w.churnRisk],
|
| 81 |
+
risk: w.churnRisk,
|
| 82 |
+
score: w.riskScore,
|
| 83 |
+
}))
|
| 84 |
+
|
| 85 |
+
try {
|
| 86 |
+
const res = await fetch('/api/torque/bulk-fire', {
|
| 87 |
+
method: 'POST',
|
| 88 |
+
headers: { 'Content-Type': 'application/json' },
|
| 89 |
+
body: JSON.stringify({ targets }),
|
| 90 |
+
})
|
| 91 |
+
const data = await res.json()
|
| 92 |
+
|
| 93 |
+
if (data.details) {
|
| 94 |
+
const newFired = new Map(fired)
|
| 95 |
+
data.details.forEach((d: any, i: number) => {
|
| 96 |
+
if (d.success) newFired.set(criticalUnfired[i].address, d.eventId || 'sent')
|
| 97 |
+
})
|
| 98 |
+
setFired(newFired)
|
| 99 |
+
setBulkProgress({ done: data.fired, total: data.total })
|
| 100 |
+
toast({
|
| 101 |
+
type: data.fired > 0 ? 'success' : 'error',
|
| 102 |
+
title: `Bulk rescue: ${data.fired}/${data.total} fired`,
|
| 103 |
+
body: data.fired > 0 ? 'All critical wallets targeted via Torque' : data.details[0]?.error,
|
| 104 |
+
})
|
| 105 |
+
}
|
| 106 |
+
} catch (e: any) {
|
| 107 |
+
toast({ type: 'error', title: 'Bulk fire failed', body: e.message })
|
| 108 |
+
} finally {
|
| 109 |
+
setBulkFiring(false)
|
| 110 |
+
setTimeout(() => setBulkProgress(null), 3000)
|
| 111 |
+
}
|
| 112 |
+
}, [bulkFiring, criticalUnfired, fired, toast])
|
| 113 |
|
| 114 |
return (
|
| 115 |
<div className="p-6 space-y-6">
|
| 116 |
+
<div className="flex items-center justify-between">
|
| 117 |
+
<div>
|
| 118 |
+
<h1 className="text-display-sm text-[#eaecef]">Wallets</h1>
|
| 119 |
+
<p className="text-body-md text-muted mt-1">Monitor wallet health, churn risk & activity patterns</p>
|
| 120 |
+
</div>
|
| 121 |
+
<div className="flex items-center gap-3">
|
| 122 |
+
{fired.size > 0 && (
|
| 123 |
+
<div className="flex items-center gap-2 px-3 py-1.5 rounded-lg bg-trading-up/10 border border-trading-up/30">
|
| 124 |
+
<CheckCircle2 className="w-4 h-4 text-trading-up" />
|
| 125 |
+
<span className="text-caption text-trading-up font-semibold">{fired.size} events sent</span>
|
| 126 |
+
</div>
|
| 127 |
+
)}
|
| 128 |
+
{criticalUnfired.length > 0 && (
|
| 129 |
+
<button
|
| 130 |
+
onClick={bulkRescue}
|
| 131 |
+
disabled={bulkFiring}
|
| 132 |
+
className="flex items-center gap-2 px-4 py-2 rounded-md bg-trading-down/10 border border-trading-down/40 text-trading-down text-button font-semibold hover:bg-trading-down/20 transition disabled:opacity-50"
|
| 133 |
+
>
|
| 134 |
+
{bulkFiring
|
| 135 |
+
? <Loader2 className="w-4 h-4 animate-spin" />
|
| 136 |
+
: <Siren className="w-4 h-4" />}
|
| 137 |
+
{bulkProgress
|
| 138 |
+
? `${bulkProgress.done}/${bulkProgress.total} fired`
|
| 139 |
+
: bulkFiring
|
| 140 |
+
? 'Rescuing...'
|
| 141 |
+
: `Rescue All Critical (${criticalUnfired.length})`}
|
| 142 |
+
</button>
|
| 143 |
+
)}
|
| 144 |
+
</div>
|
| 145 |
+
</div>
|
| 146 |
|
| 147 |
<div className="grid grid-cols-2 lg:grid-cols-6 gap-3">
|
| 148 |
{filters.map(r => (
|
| 149 |
+
<button key={r} onClick={() => setF(r)} className={cn('rounded-xl border p-3 text-center transition-all', f === r ? 'border-brand-yellow/40 bg-brand-yellow/5' : 'border-hairline-dark bg-surface-card hover:border-brand-yellow/20')}>
|
| 150 |
<span className="text-caption text-muted capitalize">{r}</span>
|
| 151 |
+
<p className="font-mono text-title-md text-[#eaecef] tabular-nums mt-0.5">{r === 'all' ? wallets.length : wallets.filter(w => w.churnRisk === r).length}</p>
|
| 152 |
</button>
|
| 153 |
))}
|
| 154 |
</div>
|
| 155 |
|
| 156 |
<div className="flex items-center gap-3">
|
| 157 |
+
<div className="relative flex-1 max-w-md">
|
| 158 |
+
<Search className="absolute left-3 top-1/2 -translate-y-1/2 w-4 h-4 text-muted" />
|
| 159 |
+
<input type="text" placeholder="Search wallet..." value={q} onChange={e => setQ(e.target.value)} className="w-full h-9 pl-9 pr-4 rounded-lg bg-surface-card border border-hairline-dark text-body-sm placeholder:text-muted focus:outline-none focus:ring-1 focus:ring-brand-yellow/50" />
|
| 160 |
+
</div>
|
| 161 |
<div className="flex items-center gap-1 p-1 rounded-lg bg-surface-card border border-hairline-dark">
|
| 162 |
+
{(['risk', 'volume', 'streak'] as const).map(s => <button key={s} onClick={() => setSort(s)} className={cn('px-3 py-1.5 rounded-md text-caption transition capitalize', sort === s ? 'bg-brand-yellow text-ink font-semibold' : 'text-muted hover:text-[#eaecef]')}>{s}</button>)}
|
| 163 |
</div>
|
| 164 |
</div>
|
| 165 |
|
|
|
|
| 172 |
<th className="text-right text-caption text-muted uppercase tracking-wider px-5 py-3">Streak</th>
|
| 173 |
<th className="text-center text-caption text-muted uppercase tracking-wider px-5 py-3">Protocols</th>
|
| 174 |
<th className="text-right text-caption text-muted uppercase tracking-wider px-5 py-3">Last Active</th>
|
| 175 |
+
<th className="text-right text-caption text-muted uppercase tracking-wider px-5 py-3">Action</th>
|
| 176 |
+
</tr></thead><tbody>{list.map(w => {
|
| 177 |
+
const isFiring = firing === w.address
|
| 178 |
+
const eventId = fired.get(w.address)
|
| 179 |
+
const canIntervene = RISK_CAN_INTERVENE.includes(w.churnRisk)
|
| 180 |
+
return (
|
| 181 |
+
<tr key={w.address} className="border-b border-hairline-dark/50 hover:bg-surface-hover transition-colors group">
|
| 182 |
+
<td className="px-5 py-4">
|
| 183 |
+
<div className="flex items-center gap-2">
|
| 184 |
+
<span className="font-mono text-body-md text-[#eaecef]">{shortAddr(w.address)}</span>
|
| 185 |
+
<a href={`https://solscan.io/account/${w.address}`} target="_blank" rel="noreferrer">
|
| 186 |
+
<ExternalLink className="w-3 h-3 text-muted opacity-0 group-hover:opacity-100 transition" />
|
| 187 |
+
</a>
|
| 188 |
+
</div>
|
| 189 |
+
{eventId && eventId !== 'error' && (
|
| 190 |
+
<p className="font-mono text-[10px] text-trading-up mt-0.5">✓ {eventId.slice(0, 10)}...</p>
|
| 191 |
+
)}
|
| 192 |
+
</td>
|
| 193 |
+
<td className="px-5 py-4 text-center"><RiskBadge risk={w.churnRisk} /></td>
|
| 194 |
+
<td className="px-5 py-4 text-right">
|
| 195 |
+
<div className="flex items-center justify-end gap-2">
|
| 196 |
+
<div className="w-16 h-1.5 rounded-full bg-surface-elevated overflow-hidden">
|
| 197 |
+
<div className={cn('h-full rounded-full transition-all duration-700', w.riskScore >= 80 ? 'bg-trading-down' : w.riskScore >= 60 ? 'bg-[#ff9500]' : w.riskScore >= 40 ? 'bg-brand-yellow' : w.riskScore >= 20 ? 'bg-trading-up' : 'bg-brand-turquoise')} style={{ width: w.riskScore + '%' }} />
|
| 198 |
+
</div>
|
| 199 |
+
<span className="font-mono text-num-sm text-muted tabular-nums w-8 text-right">{w.riskScore}</span>
|
| 200 |
+
</div>
|
| 201 |
+
</td>
|
| 202 |
+
<td className="px-5 py-4 text-right font-mono text-num-sm text-[#eaecef] tabular-nums">{fmtUsd(w.totalVolume)}</td>
|
| 203 |
+
<td className="px-5 py-4 text-right">
|
| 204 |
+
<div className="flex items-center justify-end gap-1">
|
| 205 |
+
<Flame className={cn('w-3.5 h-3.5', w.streak >= 30 ? 'text-brand-yellow' : w.streak >= 7 ? 'text-trading-up' : w.streak === 0 ? 'text-trading-down' : 'text-muted')} />
|
| 206 |
+
<span className="font-mono text-num-sm tabular-nums">{w.streak}d</span>
|
| 207 |
+
</div>
|
| 208 |
+
</td>
|
| 209 |
+
<td className="px-5 py-4">
|
| 210 |
+
<div className="flex flex-wrap justify-center gap-1">
|
| 211 |
+
{w.protocols.slice(0, 3).map(p => <span key={p} className="text-[10px] px-1.5 py-0.5 rounded bg-surface-elevated text-muted">{p}</span>)}
|
| 212 |
+
{w.protocols.length > 3 && <span className="text-[10px] px-1.5 py-0.5 rounded bg-surface-elevated text-muted">+{w.protocols.length - 3}</span>}
|
| 213 |
+
</div>
|
| 214 |
+
</td>
|
| 215 |
+
<td className="px-5 py-4 text-right text-body-sm text-muted">{w.lastActive}</td>
|
| 216 |
+
<td className="px-5 py-4 text-right">
|
| 217 |
+
{canIntervene && (
|
| 218 |
+
eventId ? (
|
| 219 |
+
<span className="inline-flex items-center gap-1 text-[10px] font-semibold text-trading-up">
|
| 220 |
+
<CheckCircle2 className="w-3 h-3" />Sent
|
| 221 |
+
</span>
|
| 222 |
+
) : (
|
| 223 |
+
<button
|
| 224 |
+
onClick={() => intervene(w)}
|
| 225 |
+
disabled={!!firing}
|
| 226 |
+
className="inline-flex items-center gap-1.5 px-2.5 py-1 rounded-md bg-brand-yellow/10 border border-brand-yellow/30 text-brand-yellow text-[11px] font-semibold hover:bg-brand-yellow/20 transition disabled:opacity-40"
|
| 227 |
+
>
|
| 228 |
+
{isFiring ? <Loader2 className="w-3 h-3 animate-spin" /> : <Zap className="w-3 h-3" />}
|
| 229 |
+
{isFiring ? 'Firing...' : ACTION_LABEL[w.churnRisk]}
|
| 230 |
+
</button>
|
| 231 |
+
)
|
| 232 |
+
)}
|
| 233 |
+
{!canIntervene && <ChevronRight className="w-4 h-4 text-muted opacity-0 group-hover:opacity-100 transition ml-auto" />}
|
| 234 |
+
</td>
|
| 235 |
+
</tr>
|
| 236 |
+
)
|
| 237 |
+
})}</tbody></table>
|
| 238 |
</div>
|
| 239 |
</div>
|
| 240 |
)
|
|
@@ -1,3 +1,77 @@
|
|
| 1 |
import { NextResponse } from 'next/server'
|
| 2 |
-
|
| 3 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
import { NextResponse } from 'next/server'
|
| 2 |
+
import { wallets } from '@/lib/mock-data'
|
| 3 |
+
import { calculateChurnScore } from '@/lib/agent-engine'
|
| 4 |
+
import { sendCustomEvent, isTorqueConfigured } from '@/lib/torque-mcp'
|
| 5 |
+
import { pushEvent } from '@/lib/event-store'
|
| 6 |
+
|
| 7 |
+
function walletToSignals(w: typeof wallets[0]) {
|
| 8 |
+
const daysMatch = w.lastActive.match(/(\d+)d/)
|
| 9 |
+
const daysInactive = daysMatch ? parseInt(daysMatch[1]) : 0
|
| 10 |
+
return {
|
| 11 |
+
daysInactive,
|
| 12 |
+
volumeDropPct: w.streak === 0 ? 80 : Math.max(0, (10 - Math.min(w.streak, 10)) * 8),
|
| 13 |
+
uniqueProtocols: w.protocols.length,
|
| 14 |
+
currentStreak: w.streak,
|
| 15 |
+
hasLiquidation: false,
|
| 16 |
+
}
|
| 17 |
+
}
|
| 18 |
+
|
| 19 |
+
export async function POST() {
|
| 20 |
+
const configured = isTorqueConfigured()
|
| 21 |
+
const detections: Array<{
|
| 22 |
+
wallet: string; risk: string; score: number; eventName: string;
|
| 23 |
+
eventSent: boolean; eventId?: string; error?: string
|
| 24 |
+
}> = []
|
| 25 |
+
|
| 26 |
+
for (const wallet of wallets) {
|
| 27 |
+
const signals = walletToSignals(wallet)
|
| 28 |
+
const { score, risk } = calculateChurnScore(signals)
|
| 29 |
+
|
| 30 |
+
if (risk === 'critical' || risk === 'high' || risk === 'medium') {
|
| 31 |
+
const eventName = risk === 'critical' || risk === 'high' ? 'churn_risk_high' : 'churn_risk_medium'
|
| 32 |
+
|
| 33 |
+
const result = configured
|
| 34 |
+
? await sendCustomEvent(wallet.address, eventName, {
|
| 35 |
+
risk, score,
|
| 36 |
+
daysInactive: signals.daysInactive,
|
| 37 |
+
protocols: wallet.protocols,
|
| 38 |
+
detectedBy: 'flowstate-ai-agent',
|
| 39 |
+
})
|
| 40 |
+
: { success: false, error: 'TORQUE_INGEST_KEY not configured' }
|
| 41 |
+
|
| 42 |
+
if (result.success && result.eventId) {
|
| 43 |
+
pushEvent({
|
| 44 |
+
ingestionId: result.eventId,
|
| 45 |
+
wallet: wallet.address,
|
| 46 |
+
eventName,
|
| 47 |
+
risk,
|
| 48 |
+
score,
|
| 49 |
+
firedAt: new Date().toISOString(),
|
| 50 |
+
source: 'scan',
|
| 51 |
+
})
|
| 52 |
+
}
|
| 53 |
+
|
| 54 |
+
detections.push({
|
| 55 |
+
wallet: wallet.address, risk, score, eventName,
|
| 56 |
+
eventSent: result.success,
|
| 57 |
+
eventId: result.eventId,
|
| 58 |
+
error: result.error,
|
| 59 |
+
})
|
| 60 |
+
}
|
| 61 |
+
}
|
| 62 |
+
|
| 63 |
+
return NextResponse.json({
|
| 64 |
+
detections,
|
| 65 |
+
count: detections.length,
|
| 66 |
+
configured,
|
| 67 |
+
timestamp: new Date().toISOString(),
|
| 68 |
+
})
|
| 69 |
+
}
|
| 70 |
+
|
| 71 |
+
export async function GET() {
|
| 72 |
+
return NextResponse.json({
|
| 73 |
+
status: 'active',
|
| 74 |
+
configured: isTorqueConfigured(),
|
| 75 |
+
capabilities: ['churn_detection', 'auto_campaign_creation', 'comeback_detection', 'streak_tracking'],
|
| 76 |
+
})
|
| 77 |
+
}
|
|
@@ -0,0 +1,45 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
import { NextResponse } from 'next/server'
|
| 2 |
+
import { sendCustomEvent, isTorqueConfigured } from '@/lib/torque-mcp'
|
| 3 |
+
import { pushEvent } from '@/lib/event-store'
|
| 4 |
+
|
| 5 |
+
interface BulkTarget { wallet: string; eventName: string; risk?: string; score?: number }
|
| 6 |
+
|
| 7 |
+
export async function POST(req: Request) {
|
| 8 |
+
const body = await req.json()
|
| 9 |
+
const { targets }: { targets: BulkTarget[] } = body
|
| 10 |
+
|
| 11 |
+
if (!Array.isArray(targets) || targets.length === 0) {
|
| 12 |
+
return NextResponse.json({ error: 'targets array required' }, { status: 400 })
|
| 13 |
+
}
|
| 14 |
+
if (!isTorqueConfigured()) {
|
| 15 |
+
return NextResponse.json({ error: 'TORQUE_INGEST_KEY not configured' }, { status: 503 })
|
| 16 |
+
}
|
| 17 |
+
|
| 18 |
+
const results = await Promise.allSettled(
|
| 19 |
+
targets.map(async (t) => {
|
| 20 |
+
const result = await sendCustomEvent(t.wallet, t.eventName, {
|
| 21 |
+
risk: t.risk,
|
| 22 |
+
score: t.score,
|
| 23 |
+
detectedBy: 'flowstate-bulk-rescue',
|
| 24 |
+
timestamp: new Date().toISOString(),
|
| 25 |
+
})
|
| 26 |
+
if (result.success && result.eventId) {
|
| 27 |
+
pushEvent({
|
| 28 |
+
ingestionId: result.eventId,
|
| 29 |
+
wallet: t.wallet,
|
| 30 |
+
eventName: t.eventName,
|
| 31 |
+
risk: t.risk,
|
| 32 |
+
score: t.score,
|
| 33 |
+
firedAt: new Date().toISOString(),
|
| 34 |
+
source: 'manual',
|
| 35 |
+
})
|
| 36 |
+
}
|
| 37 |
+
return { wallet: t.wallet, success: result.success, eventId: result.eventId, error: result.error }
|
| 38 |
+
})
|
| 39 |
+
)
|
| 40 |
+
|
| 41 |
+
const fired = results.filter(r => r.status === 'fulfilled' && (r.value as any).success).length
|
| 42 |
+
const details = results.map(r => r.status === 'fulfilled' ? r.value : { success: false, error: String((r as any).reason) })
|
| 43 |
+
|
| 44 |
+
return NextResponse.json({ fired, total: targets.length, details })
|
| 45 |
+
}
|
|
@@ -1,3 +1,26 @@
|
|
| 1 |
import { NextResponse } from 'next/server'
|
| 2 |
-
|
| 3 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
import { NextResponse } from 'next/server'
|
| 2 |
+
import { createCampaign, isTorqueConfigured } from '@/lib/torque-mcp'
|
| 3 |
+
|
| 4 |
+
export async function POST(req: Request) {
|
| 5 |
+
const body = await req.json()
|
| 6 |
+
|
| 7 |
+
if (!body.name || !body.type || !body.budget) {
|
| 8 |
+
return NextResponse.json({ error: 'name, type, and budget required' }, { status: 400 })
|
| 9 |
+
}
|
| 10 |
+
|
| 11 |
+
const result = await createCampaign(body)
|
| 12 |
+
|
| 13 |
+
if (!result.success) {
|
| 14 |
+
const status = isTorqueConfigured() ? 502 : 503
|
| 15 |
+
return NextResponse.json({ success: false, error: result.error }, { status })
|
| 16 |
+
}
|
| 17 |
+
|
| 18 |
+
return NextResponse.json({ success: true, campaignId: result.campaignId })
|
| 19 |
+
}
|
| 20 |
+
|
| 21 |
+
export async function GET() {
|
| 22 |
+
return NextResponse.json({
|
| 23 |
+
status: isTorqueConfigured() ? 'ok' : 'unconfigured',
|
| 24 |
+
campaigns: [],
|
| 25 |
+
})
|
| 26 |
+
}
|
|
@@ -0,0 +1,8 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
import { NextResponse } from 'next/server'
|
| 2 |
+
import { getEvents, getCount } from '@/lib/event-store'
|
| 3 |
+
|
| 4 |
+
export async function GET(req: Request) {
|
| 5 |
+
const { searchParams } = new URL(req.url)
|
| 6 |
+
const limit = Math.min(parseInt(searchParams.get('limit') || '20'), 50)
|
| 7 |
+
return NextResponse.json({ events: getEvents(limit), total: getCount() })
|
| 8 |
+
}
|
|
@@ -1,3 +1,40 @@
|
|
| 1 |
import { NextResponse } from 'next/server'
|
| 2 |
-
|
| 3 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
import { NextResponse } from 'next/server'
|
| 2 |
+
import { sendCustomEvent, isTorqueConfigured } from '@/lib/torque-mcp'
|
| 3 |
+
import { pushEvent } from '@/lib/event-store'
|
| 4 |
+
|
| 5 |
+
export async function POST(req: Request) {
|
| 6 |
+
const body = await req.json()
|
| 7 |
+
const { wallet, eventType, eventName, metadata, data, risk, score } = body
|
| 8 |
+
const name = eventName || eventType
|
| 9 |
+
const payload = data || metadata || {}
|
| 10 |
+
|
| 11 |
+
if (!wallet || !name) {
|
| 12 |
+
return NextResponse.json({ error: 'wallet and eventName required' }, { status: 400 })
|
| 13 |
+
}
|
| 14 |
+
|
| 15 |
+
const result = await sendCustomEvent(wallet, name, payload)
|
| 16 |
+
|
| 17 |
+
if (!result.success) {
|
| 18 |
+
const status = isTorqueConfigured() ? 502 : 503
|
| 19 |
+
return NextResponse.json({ success: false, error: result.error }, { status })
|
| 20 |
+
}
|
| 21 |
+
|
| 22 |
+
pushEvent({
|
| 23 |
+
ingestionId: result.eventId || 'local-' + Date.now(),
|
| 24 |
+
wallet,
|
| 25 |
+
eventName: name,
|
| 26 |
+
risk,
|
| 27 |
+
score,
|
| 28 |
+
firedAt: new Date().toISOString(),
|
| 29 |
+
source: 'manual',
|
| 30 |
+
})
|
| 31 |
+
|
| 32 |
+
return NextResponse.json({ success: true, eventId: result.eventId })
|
| 33 |
+
}
|
| 34 |
+
|
| 35 |
+
export async function GET() {
|
| 36 |
+
return NextResponse.json({
|
| 37 |
+
status: isTorqueConfigured() ? 'ok' : 'unconfigured',
|
| 38 |
+
events: ['churn_risk_high', 'churn_risk_medium', 'comeback_detected', 'streak_maintained', 'volume_milestone', 'referral_from_saved', 'inactivity_detected'],
|
| 39 |
+
})
|
| 40 |
+
}
|
|
@@ -0,0 +1,14 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
import { NextResponse } from 'next/server'
|
| 2 |
+
import { isTorqueConfigured } from '@/lib/torque-mcp'
|
| 3 |
+
import { getCount } from '@/lib/event-store'
|
| 4 |
+
|
| 5 |
+
export async function GET() {
|
| 6 |
+
const configured = isTorqueConfigured()
|
| 7 |
+
return NextResponse.json({
|
| 8 |
+
configured,
|
| 9 |
+
status: configured ? 'connected' : 'unconfigured',
|
| 10 |
+
sessionEvents: getCount(),
|
| 11 |
+
projectId: 'cmon6tgg20009jr1iz2b1eue8',
|
| 12 |
+
projectName: 'FlowState',
|
| 13 |
+
})
|
| 14 |
+
}
|
|
@@ -3,7 +3,7 @@ import Link from 'next/link'
|
|
| 3 |
import { usePathname } from 'next/navigation'
|
| 4 |
import { cn } from '@/lib/utils'
|
| 5 |
import { LayoutDashboard, Trophy, Megaphone, BarChart3, Wallet, Bot, Zap, Shield, ChevronLeft, ChevronRight } from 'lucide-react'
|
| 6 |
-
import { useState } from 'react'
|
| 7 |
|
| 8 |
const nav = [
|
| 9 |
{ label: 'Dashboard', href: '/', icon: LayoutDashboard },
|
|
@@ -17,6 +17,23 @@ const nav = [
|
|
| 17 |
export function Sidebar() {
|
| 18 |
const path = usePathname()
|
| 19 |
const [col, setCol] = useState(false)
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 20 |
return (
|
| 21 |
<aside className={cn('hidden md:flex flex-col border-r border-hairline-dark bg-surface-card transition-all duration-300', col ? 'w-[68px]' : 'w-[240px]')}>
|
| 22 |
<div className="flex items-center h-16 px-4 border-b border-hairline-dark">
|
|
@@ -40,7 +57,13 @@ export function Sidebar() {
|
|
| 40 |
{!col && (
|
| 41 |
<div className="mx-3 mb-3 p-3 rounded-xl bg-brand-yellow/5 border border-brand-yellow/20">
|
| 42 |
<div className="flex items-center gap-2 mb-1"><Shield className="w-4 h-4 text-trading-up" /><span className="text-caption text-trading-up font-semibold">AI Agent Active</span></div>
|
| 43 |
-
<p className="text-caption text-muted">
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 44 |
</div>
|
| 45 |
)}
|
| 46 |
<button onClick={() => setCol(!col)} className="flex items-center gap-3 px-3 py-2.5 rounded-lg text-nav text-muted hover:text-[#eaecef] hover:bg-surface-elevated w-full transition-colors">
|
|
|
|
| 3 |
import { usePathname } from 'next/navigation'
|
| 4 |
import { cn } from '@/lib/utils'
|
| 5 |
import { LayoutDashboard, Trophy, Megaphone, BarChart3, Wallet, Bot, Zap, Shield, ChevronLeft, ChevronRight } from 'lucide-react'
|
| 6 |
+
import { useState, useEffect } from 'react'
|
| 7 |
|
| 8 |
const nav = [
|
| 9 |
{ label: 'Dashboard', href: '/', icon: LayoutDashboard },
|
|
|
|
| 17 |
export function Sidebar() {
|
| 18 |
const path = usePathname()
|
| 19 |
const [col, setCol] = useState(false)
|
| 20 |
+
const [eventCount, setEventCount] = useState<number | null>(null)
|
| 21 |
+
|
| 22 |
+
useEffect(() => {
|
| 23 |
+
const fetchCount = async () => {
|
| 24 |
+
try {
|
| 25 |
+
const res = await fetch('/api/torque/events/recent?limit=1')
|
| 26 |
+
if (res.ok) {
|
| 27 |
+
const data = await res.json()
|
| 28 |
+
setEventCount(data.total ?? 0)
|
| 29 |
+
}
|
| 30 |
+
} catch {}
|
| 31 |
+
}
|
| 32 |
+
fetchCount()
|
| 33 |
+
const i = setInterval(fetchCount, 5000)
|
| 34 |
+
return () => clearInterval(i)
|
| 35 |
+
}, [])
|
| 36 |
+
|
| 37 |
return (
|
| 38 |
<aside className={cn('hidden md:flex flex-col border-r border-hairline-dark bg-surface-card transition-all duration-300', col ? 'w-[68px]' : 'w-[240px]')}>
|
| 39 |
<div className="flex items-center h-16 px-4 border-b border-hairline-dark">
|
|
|
|
| 57 |
{!col && (
|
| 58 |
<div className="mx-3 mb-3 p-3 rounded-xl bg-brand-yellow/5 border border-brand-yellow/20">
|
| 59 |
<div className="flex items-center gap-2 mb-1"><Shield className="w-4 h-4 text-trading-up" /><span className="text-caption text-trading-up font-semibold">AI Agent Active</span></div>
|
| 60 |
+
<p className="text-caption text-muted">
|
| 61 |
+
{eventCount === null
|
| 62 |
+
? 'Connecting...'
|
| 63 |
+
: eventCount === 0
|
| 64 |
+
? 'No events this session'
|
| 65 |
+
: `${eventCount} event${eventCount === 1 ? '' : 's'} fired today`}
|
| 66 |
+
</p>
|
| 67 |
</div>
|
| 68 |
)}
|
| 69 |
<button onClick={() => setCol(!col)} className="flex items-center gap-3 px-3 py-2.5 rounded-lg text-nav text-muted hover:text-[#eaecef] hover:bg-surface-elevated w-full transition-colors">
|
|
@@ -1,14 +1,32 @@
|
|
| 1 |
'use client'
|
| 2 |
-
import { Bell, Search, Globe, Wifi } from 'lucide-react'
|
| 3 |
import { useState, useEffect } from 'react'
|
| 4 |
|
|
|
|
|
|
|
| 5 |
export function Topbar() {
|
| 6 |
const [time, setTime] = useState('')
|
|
|
|
|
|
|
| 7 |
useEffect(() => {
|
| 8 |
const u = () => setTime(new Date().toLocaleTimeString('en-US', { hour12: false }))
|
| 9 |
u(); const i = setInterval(u, 1000); return () => clearInterval(i)
|
| 10 |
}, [])
|
| 11 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 12 |
return (
|
| 13 |
<header className="h-14 border-b border-hairline-dark bg-surface-card/80 backdrop-blur-sm flex items-center justify-between px-6">
|
| 14 |
<div className="relative">
|
|
@@ -16,13 +34,29 @@ export function Topbar() {
|
|
| 16 |
<input type="text" placeholder="Search wallets, campaigns..." className="w-64 h-9 pl-9 pr-4 rounded-lg bg-surface-elevated border border-hairline-dark text-body-sm text-[#eaecef] placeholder:text-muted focus:outline-none focus:ring-1 focus:ring-brand-yellow/50 transition" />
|
| 17 |
</div>
|
| 18 |
<div className="flex items-center gap-4">
|
| 19 |
-
|
| 20 |
-
<div className=
|
| 21 |
-
|
| 22 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 23 |
<span className="font-mono text-num-sm text-muted tabular-nums">{time}</span>
|
| 24 |
<button className="flex items-center gap-1.5 text-muted hover:text-[#eaecef] transition">
|
| 25 |
-
<Wifi className=
|
| 26 |
</button>
|
| 27 |
<button className="relative p-2 rounded-lg hover:bg-surface-elevated transition">
|
| 28 |
<Bell className="w-4 h-4 text-muted" />
|
|
|
|
| 1 |
'use client'
|
| 2 |
+
import { Bell, Search, Globe, Wifi, Zap } from 'lucide-react'
|
| 3 |
import { useState, useEffect } from 'react'
|
| 4 |
|
| 5 |
+
interface TorqueStatus { configured: boolean; status: string; sessionEvents: number }
|
| 6 |
+
|
| 7 |
export function Topbar() {
|
| 8 |
const [time, setTime] = useState('')
|
| 9 |
+
const [torque, setTorque] = useState<TorqueStatus | null>(null)
|
| 10 |
+
|
| 11 |
useEffect(() => {
|
| 12 |
const u = () => setTime(new Date().toLocaleTimeString('en-US', { hour12: false }))
|
| 13 |
u(); const i = setInterval(u, 1000); return () => clearInterval(i)
|
| 14 |
}, [])
|
| 15 |
|
| 16 |
+
useEffect(() => {
|
| 17 |
+
const fetchStatus = async () => {
|
| 18 |
+
try {
|
| 19 |
+
const res = await fetch('/api/torque/status')
|
| 20 |
+
if (res.ok) setTorque(await res.json())
|
| 21 |
+
} catch {}
|
| 22 |
+
}
|
| 23 |
+
fetchStatus()
|
| 24 |
+
const i = setInterval(fetchStatus, 8000)
|
| 25 |
+
return () => clearInterval(i)
|
| 26 |
+
}, [])
|
| 27 |
+
|
| 28 |
+
const isLive = torque?.configured && torque?.status === 'connected'
|
| 29 |
+
|
| 30 |
return (
|
| 31 |
<header className="h-14 border-b border-hairline-dark bg-surface-card/80 backdrop-blur-sm flex items-center justify-between px-6">
|
| 32 |
<div className="relative">
|
|
|
|
| 34 |
<input type="text" placeholder="Search wallets, campaigns..." className="w-64 h-9 pl-9 pr-4 rounded-lg bg-surface-elevated border border-hairline-dark text-body-sm text-[#eaecef] placeholder:text-muted focus:outline-none focus:ring-1 focus:ring-brand-yellow/50 transition" />
|
| 35 |
</div>
|
| 36 |
<div className="flex items-center gap-4">
|
| 37 |
+
{torque && (
|
| 38 |
+
<div className={`flex items-center gap-2 px-3 py-1.5 rounded-pill border ${isLive ? 'bg-trading-up/10 border-trading-up/20' : 'bg-trading-down/10 border-trading-down/20'}`}>
|
| 39 |
+
<div className={`w-2 h-2 rounded-full ${isLive ? 'bg-trading-up animate-pulse' : 'bg-trading-down'}`} />
|
| 40 |
+
<span className={`text-caption font-semibold ${isLive ? 'text-trading-up' : 'text-trading-down'}`}>
|
| 41 |
+
TORQUE {isLive ? 'LIVE' : 'OFFLINE'}
|
| 42 |
+
</span>
|
| 43 |
+
</div>
|
| 44 |
+
)}
|
| 45 |
+
{!torque && (
|
| 46 |
+
<div className="flex items-center gap-2 px-3 py-1.5 rounded-pill bg-muted/10 border border-muted/20">
|
| 47 |
+
<div className="w-2 h-2 rounded-full bg-muted animate-pulse" />
|
| 48 |
+
<span className="text-caption text-muted font-semibold">CONNECTING...</span>
|
| 49 |
+
</div>
|
| 50 |
+
)}
|
| 51 |
+
{torque && torque.sessionEvents > 0 && (
|
| 52 |
+
<div className="flex items-center gap-1.5 px-2.5 py-1 rounded-pill bg-brand-yellow/10 border border-brand-yellow/20">
|
| 53 |
+
<Zap className="w-3 h-3 text-brand-yellow" />
|
| 54 |
+
<span className="font-mono text-caption text-brand-yellow font-semibold">{torque.sessionEvents}</span>
|
| 55 |
+
</div>
|
| 56 |
+
)}
|
| 57 |
<span className="font-mono text-num-sm text-muted tabular-nums">{time}</span>
|
| 58 |
<button className="flex items-center gap-1.5 text-muted hover:text-[#eaecef] transition">
|
| 59 |
+
<Wifi className={`w-4 h-4 ${isLive ? 'text-trading-up' : 'text-muted'}`} /><span className="text-caption">Solana</span>
|
| 60 |
</button>
|
| 61 |
<button className="relative p-2 rounded-lg hover:bg-surface-elevated transition">
|
| 62 |
<Bell className="w-4 h-4 text-muted" />
|
|
@@ -1,26 +1,75 @@
|
|
| 1 |
'use client'
|
| 2 |
-
import { useState, useEffect } from 'react'
|
| 3 |
import { cn } from '@/lib/utils'
|
| 4 |
import { agentMsgs } from '@/lib/mock-data'
|
| 5 |
-
import { Bot, ChevronDown, ChevronUp } from 'lucide-react'
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 6 |
|
| 7 |
export function AgentFeed() {
|
| 8 |
-
const [msgs, setMsgs] = useState<{ text: string; time: string }[]>([])
|
| 9 |
const [open, setOpen] = useState(true)
|
|
|
|
|
|
|
| 10 |
|
|
|
|
| 11 |
useEffect(() => {
|
| 12 |
const init = agentMsgs.slice(0, 5).map((t, i) => ({
|
| 13 |
-
text: t,
|
|
|
|
| 14 |
}))
|
| 15 |
setMsgs(init)
|
| 16 |
let c = 5
|
| 17 |
const iv = setInterval(() => {
|
| 18 |
const t = agentMsgs[c++ % agentMsgs.length]
|
| 19 |
-
setMsgs(p => [{ text: t, time: new Date().toLocaleTimeString('en-US', { hour12: false }) }, ...p].slice(0,
|
| 20 |
}, 4000)
|
| 21 |
return () => clearInterval(iv)
|
| 22 |
}, [])
|
| 23 |
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 24 |
return (
|
| 25 |
<div className="rounded-xl bg-surface-card border border-hairline-dark overflow-hidden">
|
| 26 |
<button onClick={() => setOpen(!open)} className="w-full flex items-center justify-between px-5 py-3 border-b border-hairline-dark hover:bg-surface-elevated transition">
|
|
@@ -28,15 +77,26 @@ export function AgentFeed() {
|
|
| 28 |
<Bot className="w-4 h-4 text-brand-yellow" />
|
| 29 |
<span className="text-title-sm">AI Agent Feed</span>
|
| 30 |
<div className="w-2 h-2 rounded-full bg-trading-up animate-pulse" />
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 31 |
</div>
|
| 32 |
{open ? <ChevronUp className="w-4 h-4 text-muted" /> : <ChevronDown className="w-4 h-4 text-muted" />}
|
| 33 |
</button>
|
| 34 |
{open && (
|
| 35 |
<div className="max-h-[300px] overflow-y-auto">
|
| 36 |
{msgs.map((m, i) => (
|
| 37 |
-
<div key={i} className={cn(
|
|
|
|
|
|
|
|
|
|
|
|
|
| 38 |
<span className="font-mono text-caption text-muted tabular-nums whitespace-nowrap mt-0.5">{m.time}</span>
|
| 39 |
-
<span className=
|
|
|
|
| 40 |
</div>
|
| 41 |
))}
|
| 42 |
</div>
|
|
|
|
| 1 |
'use client'
|
| 2 |
+
import { useState, useEffect, useRef } from 'react'
|
| 3 |
import { cn } from '@/lib/utils'
|
| 4 |
import { agentMsgs } from '@/lib/mock-data'
|
| 5 |
+
import { Bot, ChevronDown, ChevronUp, Zap } from 'lucide-react'
|
| 6 |
+
|
| 7 |
+
interface LiveEvent {
|
| 8 |
+
ingestionId: string; wallet: string; eventName: string
|
| 9 |
+
risk?: string; score?: number; firedAt: string; source: string
|
| 10 |
+
}
|
| 11 |
+
|
| 12 |
+
const EVENT_EMOJI: Record<string, string> = {
|
| 13 |
+
churn_risk_high: '🚨',
|
| 14 |
+
churn_risk_medium: '⚠️',
|
| 15 |
+
comeback_detected: '🔥',
|
| 16 |
+
streak_maintained: '⚡',
|
| 17 |
+
volume_milestone: '🏆',
|
| 18 |
+
inactivity_detected: '💤',
|
| 19 |
+
referral_from_saved: '🤝',
|
| 20 |
+
}
|
| 21 |
+
|
| 22 |
+
function liveEventToMsg(e: LiveEvent): string {
|
| 23 |
+
const emoji = EVENT_EMOJI[e.eventName] || '📡'
|
| 24 |
+
const short = `${e.wallet.slice(0, 6)}...${e.wallet.slice(-4)}`
|
| 25 |
+
const label = e.eventName.replace(/_/g, ' ')
|
| 26 |
+
const score = e.score !== undefined ? ` (score=${e.score})` : ''
|
| 27 |
+
return `${emoji} LIVE: ${label} → ${short}${score} [${e.ingestionId.slice(0, 8)}]`
|
| 28 |
+
}
|
| 29 |
|
| 30 |
export function AgentFeed() {
|
| 31 |
+
const [msgs, setMsgs] = useState<{ text: string; time: string; live?: boolean }[]>([])
|
| 32 |
const [open, setOpen] = useState(true)
|
| 33 |
+
const [liveCount, setLiveCount] = useState(0)
|
| 34 |
+
const seenIds = useRef(new Set<string>())
|
| 35 |
|
| 36 |
+
// Seed mock messages
|
| 37 |
useEffect(() => {
|
| 38 |
const init = agentMsgs.slice(0, 5).map((t, i) => ({
|
| 39 |
+
text: t,
|
| 40 |
+
time: new Date(Date.now() - i * 120000).toLocaleTimeString('en-US', { hour12: false }),
|
| 41 |
}))
|
| 42 |
setMsgs(init)
|
| 43 |
let c = 5
|
| 44 |
const iv = setInterval(() => {
|
| 45 |
const t = agentMsgs[c++ % agentMsgs.length]
|
| 46 |
+
setMsgs(p => [{ text: t, time: new Date().toLocaleTimeString('en-US', { hour12: false }) }, ...p].slice(0, 30))
|
| 47 |
}, 4000)
|
| 48 |
return () => clearInterval(iv)
|
| 49 |
}, [])
|
| 50 |
|
| 51 |
+
// Poll real events and inject at top
|
| 52 |
+
useEffect(() => {
|
| 53 |
+
const poll = async () => {
|
| 54 |
+
try {
|
| 55 |
+
const res = await fetch('/api/torque/events/recent?limit=10')
|
| 56 |
+
if (!res.ok) return
|
| 57 |
+
const data = await res.json()
|
| 58 |
+
const events: LiveEvent[] = data.events || []
|
| 59 |
+
const newEvents = events.filter(e => !seenIds.current.has(e.ingestionId))
|
| 60 |
+
if (newEvents.length === 0) return
|
| 61 |
+
newEvents.forEach(e => seenIds.current.add(e.ingestionId))
|
| 62 |
+
const now = new Date().toLocaleTimeString('en-US', { hour12: false })
|
| 63 |
+
const liveLines = newEvents.map(e => ({ text: liveEventToMsg(e), time: now, live: true }))
|
| 64 |
+
setMsgs(p => [...liveLines, ...p].slice(0, 30))
|
| 65 |
+
setLiveCount(data.total || 0)
|
| 66 |
+
} catch {}
|
| 67 |
+
}
|
| 68 |
+
poll()
|
| 69 |
+
const iv = setInterval(poll, 4000)
|
| 70 |
+
return () => clearInterval(iv)
|
| 71 |
+
}, [])
|
| 72 |
+
|
| 73 |
return (
|
| 74 |
<div className="rounded-xl bg-surface-card border border-hairline-dark overflow-hidden">
|
| 75 |
<button onClick={() => setOpen(!open)} className="w-full flex items-center justify-between px-5 py-3 border-b border-hairline-dark hover:bg-surface-elevated transition">
|
|
|
|
| 77 |
<Bot className="w-4 h-4 text-brand-yellow" />
|
| 78 |
<span className="text-title-sm">AI Agent Feed</span>
|
| 79 |
<div className="w-2 h-2 rounded-full bg-trading-up animate-pulse" />
|
| 80 |
+
{liveCount > 0 && (
|
| 81 |
+
<div className="flex items-center gap-1 px-2 py-0.5 rounded-pill bg-trading-up/10 border border-trading-up/20">
|
| 82 |
+
<Zap className="w-2.5 h-2.5 text-trading-up" />
|
| 83 |
+
<span className="text-[10px] text-trading-up font-semibold">{liveCount} live</span>
|
| 84 |
+
</div>
|
| 85 |
+
)}
|
| 86 |
</div>
|
| 87 |
{open ? <ChevronUp className="w-4 h-4 text-muted" /> : <ChevronDown className="w-4 h-4 text-muted" />}
|
| 88 |
</button>
|
| 89 |
{open && (
|
| 90 |
<div className="max-h-[300px] overflow-y-auto">
|
| 91 |
{msgs.map((m, i) => (
|
| 92 |
+
<div key={i} className={cn(
|
| 93 |
+
'flex items-start gap-3 px-5 py-2.5 border-b border-hairline-dark/50 transition-colors',
|
| 94 |
+
i === 0 && 'animate-slide-up',
|
| 95 |
+
m.live ? 'bg-trading-up/5 border-l-2 border-trading-up/50' : i === 0 && 'bg-brand-yellow/5'
|
| 96 |
+
)}>
|
| 97 |
<span className="font-mono text-caption text-muted tabular-nums whitespace-nowrap mt-0.5">{m.time}</span>
|
| 98 |
+
<span className={cn('text-body-sm flex-1', m.live ? 'text-[#eaecef] font-medium' : 'text-[#eaecef]')}>{m.text}</span>
|
| 99 |
+
{m.live && <span className="text-[10px] text-trading-up font-semibold uppercase tracking-wider whitespace-nowrap">LIVE</span>}
|
| 100 |
</div>
|
| 101 |
))}
|
| 102 |
</div>
|
|
@@ -0,0 +1,82 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
'use client'
|
| 2 |
+
import { createContext, useContext, useState, useCallback, useEffect, useRef } from 'react'
|
| 3 |
+
import { CheckCircle2, AlertCircle, Zap, X } from 'lucide-react'
|
| 4 |
+
import { cn } from '@/lib/utils'
|
| 5 |
+
|
| 6 |
+
type ToastType = 'success' | 'error' | 'event'
|
| 7 |
+
interface Toast { id: string; type: ToastType; title: string; body?: string }
|
| 8 |
+
interface ToastCtx { fire: (t: Omit<Toast, 'id'>) => void }
|
| 9 |
+
|
| 10 |
+
const Ctx = createContext<ToastCtx>({ fire: () => {} })
|
| 11 |
+
export const useToast = () => useContext(Ctx)
|
| 12 |
+
|
| 13 |
+
const ICONS = {
|
| 14 |
+
success: CheckCircle2,
|
| 15 |
+
error: AlertCircle,
|
| 16 |
+
event: Zap,
|
| 17 |
+
}
|
| 18 |
+
const COLORS = {
|
| 19 |
+
success: 'border-trading-up/40 bg-trading-up/10',
|
| 20 |
+
error: 'border-trading-down/40 bg-trading-down/10',
|
| 21 |
+
event: 'border-brand-yellow/40 bg-brand-yellow/10',
|
| 22 |
+
}
|
| 23 |
+
const ICON_COLORS = {
|
| 24 |
+
success: 'text-trading-up',
|
| 25 |
+
error: 'text-trading-down',
|
| 26 |
+
event: 'text-brand-yellow',
|
| 27 |
+
}
|
| 28 |
+
|
| 29 |
+
function ToastItem({ toast, onDismiss }: { toast: Toast; onDismiss: () => void }) {
|
| 30 |
+
const [visible, setVisible] = useState(false)
|
| 31 |
+
const Icon = ICONS[toast.type]
|
| 32 |
+
|
| 33 |
+
useEffect(() => {
|
| 34 |
+
requestAnimationFrame(() => setVisible(true))
|
| 35 |
+
const t = setTimeout(() => { setVisible(false); setTimeout(onDismiss, 300) }, 4000)
|
| 36 |
+
return () => clearTimeout(t)
|
| 37 |
+
}, [onDismiss])
|
| 38 |
+
|
| 39 |
+
return (
|
| 40 |
+
<div className={cn(
|
| 41 |
+
'flex items-start gap-3 w-80 rounded-xl border p-3.5 shadow-2xl backdrop-blur-sm transition-all duration-300',
|
| 42 |
+
COLORS[toast.type],
|
| 43 |
+
visible ? 'opacity-100 translate-x-0' : 'opacity-0 translate-x-full'
|
| 44 |
+
)}>
|
| 45 |
+
<Icon className={cn('w-4 h-4 mt-0.5 flex-shrink-0', ICON_COLORS[toast.type])} />
|
| 46 |
+
<div className="flex-1 min-w-0">
|
| 47 |
+
<p className="text-body-sm font-semibold text-[#eaecef] leading-tight">{toast.title}</p>
|
| 48 |
+
{toast.body && <p className="font-mono text-[10px] text-muted mt-0.5 truncate">{toast.body}</p>}
|
| 49 |
+
</div>
|
| 50 |
+
<button onClick={() => { setVisible(false); setTimeout(onDismiss, 300) }} className="text-muted hover:text-[#eaecef] transition flex-shrink-0">
|
| 51 |
+
<X className="w-3.5 h-3.5" />
|
| 52 |
+
</button>
|
| 53 |
+
</div>
|
| 54 |
+
)
|
| 55 |
+
}
|
| 56 |
+
|
| 57 |
+
export function ToastProvider({ children }: { children: React.ReactNode }) {
|
| 58 |
+
const [toasts, setToasts] = useState<Toast[]>([])
|
| 59 |
+
const counter = useRef(0)
|
| 60 |
+
|
| 61 |
+
const fire = useCallback((t: Omit<Toast, 'id'>) => {
|
| 62 |
+
const id = `t-${counter.current++}`
|
| 63 |
+
setToasts(p => [...p.slice(-4), { ...t, id }])
|
| 64 |
+
}, [])
|
| 65 |
+
|
| 66 |
+
const dismiss = useCallback((id: string) => {
|
| 67 |
+
setToasts(p => p.filter(t => t.id !== id))
|
| 68 |
+
}, [])
|
| 69 |
+
|
| 70 |
+
return (
|
| 71 |
+
<Ctx.Provider value={{ fire }}>
|
| 72 |
+
{children}
|
| 73 |
+
<div className="fixed bottom-6 right-6 z-[100] flex flex-col gap-2 items-end pointer-events-none">
|
| 74 |
+
{toasts.map(t => (
|
| 75 |
+
<div key={t.id} className="pointer-events-auto">
|
| 76 |
+
<ToastItem toast={t} onDismiss={() => dismiss(t.id)} />
|
| 77 |
+
</div>
|
| 78 |
+
))}
|
| 79 |
+
</div>
|
| 80 |
+
</Ctx.Provider>
|
| 81 |
+
)
|
| 82 |
+
}
|
|
@@ -66,5 +66,12 @@ export function selectResponse(risk: ChurnRisk, wallet: string): { action: strin
|
|
| 66 |
}
|
| 67 |
|
| 68 |
export async function executeResponse(wallet: string, action: string) {
|
| 69 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 70 |
}
|
|
|
|
| 66 |
}
|
| 67 |
|
| 68 |
export async function executeResponse(wallet: string, action: string) {
|
| 69 |
+
// gift/raffle → fire churn_risk_high (triggers Anti-Churn and Comeback campaigns)
|
| 70 |
+
// rebate → fire churn_risk_medium (triggers Streak Rebate campaign)
|
| 71 |
+
// detect → fire inactivity_detected
|
| 72 |
+
const eventName =
|
| 73 |
+
action === 'gift' || action === 'raffle' ? 'churn_risk_high'
|
| 74 |
+
: action === 'rebate' ? 'churn_risk_medium'
|
| 75 |
+
: 'inactivity_detected'
|
| 76 |
+
return sendCustomEvent(wallet, eventName, { triggeredAction: action, detectedBy: 'flowstate-ai-agent', timestamp: new Date().toISOString() })
|
| 77 |
}
|
|
@@ -0,0 +1,27 @@
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 1 |
+
// In-memory store for live Torque events fired this session.
|
| 2 |
+
// Module-level singleton — persists across API route calls in the same Node.js process.
|
| 3 |
+
|
| 4 |
+
export interface LiveEvent {
|
| 5 |
+
ingestionId: string
|
| 6 |
+
wallet: string
|
| 7 |
+
eventName: string
|
| 8 |
+
risk?: string
|
| 9 |
+
score?: number
|
| 10 |
+
firedAt: string
|
| 11 |
+
source: 'scan' | 'manual' | 'agent'
|
| 12 |
+
}
|
| 13 |
+
|
| 14 |
+
const store: LiveEvent[] = []
|
| 15 |
+
|
| 16 |
+
export function pushEvent(e: LiveEvent) {
|
| 17 |
+
store.unshift(e)
|
| 18 |
+
if (store.length > 100) store.splice(100)
|
| 19 |
+
}
|
| 20 |
+
|
| 21 |
+
export function getEvents(limit = 20): LiveEvent[] {
|
| 22 |
+
return store.slice(0, limit)
|
| 23 |
+
}
|
| 24 |
+
|
| 25 |
+
export function getCount(): number {
|
| 26 |
+
return store.length
|
| 27 |
+
}
|
|
@@ -1,42 +1,99 @@
|
|
| 1 |
/**
|
| 2 |
-
* Torque
|
| 3 |
-
*
|
|
|
|
| 4 |
*/
|
| 5 |
-
const
|
| 6 |
-
const
|
|
|
|
|
|
|
|
|
|
|
|
|
| 7 |
|
| 8 |
-
function
|
| 9 |
-
return
|
| 10 |
}
|
| 11 |
|
| 12 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 13 |
try {
|
| 14 |
-
const r = await fetch(
|
| 15 |
-
|
| 16 |
-
|
| 17 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 18 |
}
|
| 19 |
|
| 20 |
-
export async function createCampaign(params: {
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 21 |
try {
|
| 22 |
const now = new Date()
|
| 23 |
-
const r = await fetch(API + '/campaigns', {
|
| 24 |
-
|
| 25 |
-
|
| 26 |
-
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
|
| 27 |
}
|
| 28 |
|
| 29 |
-
export async function getLeaderboard(campaignId: string, limit = 50) {
|
|
|
|
| 30 |
try {
|
| 31 |
-
const r = await fetch(API + '/campaigns/' + campaignId + '/leaderboard?limit=' + limit, { headers:
|
| 32 |
if (!r.ok) return []
|
| 33 |
return (await r.json()).entries || []
|
| 34 |
-
} catch {
|
|
|
|
|
|
|
| 35 |
}
|
| 36 |
|
| 37 |
export async function fireChurnRiskEvent(wallet: string, risk: string, score: number, daysInactive: number, volumeDrop: number) {
|
| 38 |
-
const
|
| 39 |
-
return sendCustomEvent(wallet,
|
| 40 |
}
|
| 41 |
|
| 42 |
export async function fireComebackEvent(wallet: string, inactiveDays: number, returnProtocol: string) {
|
|
@@ -48,7 +105,7 @@ export async function fireStreakEvent(wallet: string, streakDays: number, protoc
|
|
| 48 |
}
|
| 49 |
|
| 50 |
export const MCP_TOOLS = {
|
| 51 |
-
send_custom_event: { name: 'send_custom_event', description: 'Send a custom event to Torque for a wallet', inputSchema: { type: 'object', properties: { wallet: { type: 'string' },
|
| 52 |
-
create_campaign: { name: 'create_campaign', description: 'Create a new Torque campaign', inputSchema: { type: 'object', properties: { name: { type: 'string' }, type: { type: 'string', enum: ['leaderboard','rebate','raffle','gift'] }, budget: { type: 'number' } }, required: ['name', 'type', 'budget'] } },
|
| 53 |
get_leaderboard: { name: 'get_leaderboard', description: 'Get leaderboard rankings', inputSchema: { type: 'object', properties: { campaignId: { type: 'string' } }, required: ['campaignId'] } },
|
| 54 |
} as const
|
|
|
|
| 1 |
/**
|
| 2 |
+
* Torque Ingestion Client — FlowState integration with Torque Protocol
|
| 3 |
+
* Docs: https://ingest.torque.so/events
|
| 4 |
+
* Auth: x-api-key header (user-scoped key from Torque dashboard)
|
| 5 |
*/
|
| 6 |
+
const INGEST = process.env.TORQUE_INGESTER_URL || 'https://ingest.torque.so/events'
|
| 7 |
+
const API = process.env.TORQUE_API_URL || 'https://server.torque.so'
|
| 8 |
+
// JWT from platform.torque.so/connect-mcp — for MCP auth and REST API calls
|
| 9 |
+
const JWT = process.env.TORQUE_API_KEY || process.env.TORQUE_API_TOKEN || ''
|
| 10 |
+
// tq_... key created via create_api_key MCP tool — for the ingest endpoint
|
| 11 |
+
const INGEST_KEY = process.env.TORQUE_INGEST_KEY || JWT
|
| 12 |
|
| 13 |
+
export function isTorqueConfigured(): boolean {
|
| 14 |
+
return INGEST_KEY !== '' && !INGEST_KEY.startsWith('your')
|
| 15 |
}
|
| 16 |
|
| 17 |
+
function ingestHeaders(): Record<string, string> {
|
| 18 |
+
return { 'x-api-key': INGEST_KEY, 'Content-Type': 'application/json' }
|
| 19 |
+
}
|
| 20 |
+
|
| 21 |
+
function apiHeaders(): Record<string, string> {
|
| 22 |
+
return { 'Authorization': 'Bearer ' + JWT, 'Content-Type': 'application/json' }
|
| 23 |
+
}
|
| 24 |
+
|
| 25 |
+
export async function sendCustomEvent(
|
| 26 |
+
wallet: string,
|
| 27 |
+
eventName: string,
|
| 28 |
+
data: Record<string, unknown> = {}
|
| 29 |
+
): Promise<{ success: boolean; eventId?: string; error?: string }> {
|
| 30 |
+
if (!isTorqueConfigured()) {
|
| 31 |
+
return { success: false, error: 'TORQUE_API_KEY not configured' }
|
| 32 |
+
}
|
| 33 |
try {
|
| 34 |
+
const r = await fetch(INGEST, {
|
| 35 |
+
method: 'POST',
|
| 36 |
+
headers: ingestHeaders(),
|
| 37 |
+
body: JSON.stringify({
|
| 38 |
+
userPubkey: wallet,
|
| 39 |
+
timestamp: Date.now(),
|
| 40 |
+
eventName,
|
| 41 |
+
data,
|
| 42 |
+
}),
|
| 43 |
+
})
|
| 44 |
+
if (!r.ok) {
|
| 45 |
+
const text = await r.text()
|
| 46 |
+
return { success: false, error: `Torque ingest ${r.status}: ${text}` }
|
| 47 |
+
}
|
| 48 |
+
const res = await r.json()
|
| 49 |
+
return { success: true, eventId: res.id || res.eventId }
|
| 50 |
+
} catch (e) {
|
| 51 |
+
return { success: false, error: String(e) }
|
| 52 |
+
}
|
| 53 |
}
|
| 54 |
|
| 55 |
+
export async function createCampaign(params: {
|
| 56 |
+
name: string; type: string; description: string; budget: number; tokenMint: string; formula?: string
|
| 57 |
+
}): Promise<{ success: boolean; campaignId?: string; error?: string }> {
|
| 58 |
+
if (!isTorqueConfigured()) {
|
| 59 |
+
return { success: false, error: 'TORQUE_API_KEY not configured' }
|
| 60 |
+
}
|
| 61 |
try {
|
| 62 |
const now = new Date()
|
| 63 |
+
const r = await fetch(API + '/campaigns', {
|
| 64 |
+
method: 'POST',
|
| 65 |
+
headers: apiHeaders(),
|
| 66 |
+
body: JSON.stringify({
|
| 67 |
+
...params,
|
| 68 |
+
startTime: now.toISOString(),
|
| 69 |
+
endTime: new Date(now.getTime() + 7 * 86400000).toISOString(),
|
| 70 |
+
}),
|
| 71 |
+
})
|
| 72 |
+
if (!r.ok) {
|
| 73 |
+
const text = await r.text()
|
| 74 |
+
return { success: false, error: `Torque API ${r.status}: ${text}` }
|
| 75 |
+
}
|
| 76 |
+
const data = await r.json()
|
| 77 |
+
return { success: true, campaignId: data.id }
|
| 78 |
+
} catch (e) {
|
| 79 |
+
return { success: false, error: String(e) }
|
| 80 |
+
}
|
| 81 |
}
|
| 82 |
|
| 83 |
+
export async function getLeaderboard(campaignId: string, limit = 50): Promise<unknown[]> {
|
| 84 |
+
if (!isTorqueConfigured()) return []
|
| 85 |
try {
|
| 86 |
+
const r = await fetch(API + '/campaigns/' + campaignId + '/leaderboard?limit=' + limit, { headers: apiHeaders() })
|
| 87 |
if (!r.ok) return []
|
| 88 |
return (await r.json()).entries || []
|
| 89 |
+
} catch {
|
| 90 |
+
return []
|
| 91 |
+
}
|
| 92 |
}
|
| 93 |
|
| 94 |
export async function fireChurnRiskEvent(wallet: string, risk: string, score: number, daysInactive: number, volumeDrop: number) {
|
| 95 |
+
const eventName = risk === 'critical' || risk === 'high' ? 'churn_risk_high' : 'churn_risk_medium'
|
| 96 |
+
return sendCustomEvent(wallet, eventName, { risk, score, daysInactive, volumeDrop, detectedBy: 'flowstate-ai-agent' })
|
| 97 |
}
|
| 98 |
|
| 99 |
export async function fireComebackEvent(wallet: string, inactiveDays: number, returnProtocol: string) {
|
|
|
|
| 105 |
}
|
| 106 |
|
| 107 |
export const MCP_TOOLS = {
|
| 108 |
+
send_custom_event: { name: 'send_custom_event', description: 'Send a custom event to Torque for a wallet', inputSchema: { type: 'object', properties: { wallet: { type: 'string' }, eventName: { type: 'string' }, data: { type: 'object' } }, required: ['wallet', 'eventName'] } },
|
| 109 |
+
create_campaign: { name: 'create_campaign', description: 'Create a new Torque campaign', inputSchema: { type: 'object', properties: { name: { type: 'string' }, type: { type: 'string', enum: ['leaderboard', 'rebate', 'raffle', 'gift'] }, budget: { type: 'number' } }, required: ['name', 'type', 'budget'] } },
|
| 110 |
get_leaderboard: { name: 'get_leaderboard', description: 'Get leaderboard rankings', inputSchema: { type: 'object', properties: { campaignId: { type: 'string' } }, required: ['campaignId'] } },
|
| 111 |
} as const
|