slashvibe-mcp 0.3.21 → 0.3.23
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/LICENSE +21 -0
- package/README.md +280 -47
- package/auto-update.js +10 -15
- package/config.js +36 -31
- package/crypto.js +1 -6
- package/debug.js +12 -0
- package/discord.js +19 -19
- package/eslint.config.js +54 -0
- package/index.js +217 -207
- package/intelligence/index.js +2 -9
- package/intelligence/infer.js +10 -16
- package/intelligence/patterns.js +23 -18
- package/intelligence/proactive.js +16 -15
- package/intelligence/serendipity.js +57 -20
- package/memory.js +13 -8
- package/migrate-v2.js +72 -0
- package/notification-emitter.js +2 -2
- package/notify.js +39 -14
- package/package.json +28 -29
- package/post-install.js +141 -0
- package/presence.js +2 -2
- package/prompts.js +5 -9
- package/protocol/index.js +123 -87
- package/protocol/telegram-commands.js +36 -37
- package/store/api.js +358 -529
- package/store/local.js +9 -10
- package/store/profiles.js +48 -192
- package/store/reservations.js +2 -9
- package/store/skills.js +69 -71
- package/store/sqlite.js +355 -0
- package/test-skills-bootstrap.js +20 -0
- package/test-v2-integration.js +385 -0
- package/tools/_actions.js +48 -387
- package/tools/_connection-queue.js +45 -56
- package/tools/_discovery-enhanced.js +52 -57
- package/tools/_discovery.js +87 -185
- package/tools/{l2-status.js → _experimental/l2-status.js} +68 -70
- package/tools/{shipback.js → _experimental/shipback.js} +4 -3
- package/tools/_proactive-discovery.js +60 -73
- package/tools/_shared/index.js +41 -64
- package/tools/admin-inbox.js +10 -15
- package/tools/agents.js +1 -1
- package/tools/artifact-create.js +13 -23
- package/tools/artifact-view.js +4 -4
- package/tools/{_deprecated/back.js → back.js} +1 -1
- package/tools/bye.js +3 -5
- package/tools/consent.js +2 -2
- package/tools/context.js +9 -10
- package/tools/crossword.js +3 -2
- package/tools/discover.js +94 -356
- package/tools/dm.js +27 -86
- package/tools/doctor.js +12 -41
- package/tools/drawing.js +34 -20
- package/tools/echo.js +11 -11
- package/tools/feed.js +30 -58
- package/tools/follow.js +64 -187
- package/tools/{_deprecated/forget.js → forget.js} +4 -7
- package/tools/game.js +144 -48
- package/tools/handoff.js +6 -8
- package/tools/help.js +3 -3
- package/tools/idea.js +15 -27
- package/tools/inbox.js +121 -293
- package/tools/init.js +54 -151
- package/tools/invite.js +8 -21
- package/tools/migrate.js +27 -24
- package/tools/multiplayer-game.js +50 -40
- package/tools/{_deprecated/mute.js → mute.js} +4 -3
- package/tools/notifications.js +58 -48
- package/tools/observe.js +12 -15
- package/tools/onboarding.js +8 -11
- package/tools/open.js +13 -144
- package/tools/party-game.js +23 -12
- package/tools/patterns.js +2 -1
- package/tools/ping.js +5 -7
- package/tools/react.js +28 -30
- package/tools/{_deprecated/recall.js → recall.js} +5 -10
- package/tools/release.js +4 -2
- package/tools/{_deprecated/remember.js → remember.js} +4 -6
- package/tools/report.js +2 -2
- package/tools/request.js +6 -26
- package/tools/reserve.js +1 -1
- package/tools/session-fork.js +97 -0
- package/tools/session-save.js +109 -0
- package/tools/settings.js +30 -99
- package/tools/ship.js +74 -56
- package/tools/{_deprecated/skills-exchange.js → skills-exchange.js} +38 -39
- package/tools/social-inbox.js +22 -28
- package/tools/social-post.js +24 -27
- package/tools/solo-game.js +54 -46
- package/tools/start.js +14 -148
- package/tools/status.js +21 -68
- package/tools/submit.js +4 -2
- package/tools/suggest-tags.js +36 -33
- package/tools/summarize.js +19 -16
- package/tools/tag-suggestions.js +72 -73
- package/tools/test.js +1 -1
- package/tools/{_deprecated/tictactoe.js → tictactoe.js} +26 -26
- package/tools/token.js +4 -4
- package/tools/update.js +1 -2
- package/tools/watch.js +132 -112
- package/tools/who.js +20 -40
- package/tools/{_deprecated/wordassociation.js → wordassociation.js} +23 -20
- package/tools/workshop-buddy.js +52 -53
- package/tools/x-mentions.js +0 -1
- package/tools/x-reply.js +0 -1
- package/twitter.js +14 -20
- package/version.json +8 -10
- package/webhook-runner.js +132 -0
- package/auth-store.js +0 -148
- package/bridges/bridge-monitor.js +0 -388
- package/bridges/discord-bot.js +0 -431
- package/bridges/farcaster.js +0 -299
- package/bridges/telegram.js +0 -261
- package/bridges/webhook-health.js +0 -420
- package/bridges/webhook-server.js +0 -437
- package/bridges/whatsapp.js +0 -441
- package/bridges/x-webhook.js +0 -423
- package/games/arcade.js +0 -406
- package/games/chess.js +0 -451
- package/games/colorguess.js +0 -343
- package/games/crossword-words.js +0 -171
- package/games/crossword.js +0 -461
- package/games/drawing.js +0 -347
- package/games/gameroulette.js +0 -300
- package/games/gamerouter.js +0 -336
- package/games/gamestatus.js +0 -337
- package/games/guessnumber.js +0 -209
- package/games/hangman.js +0 -279
- package/games/memory.js +0 -338
- package/games/multiplayer-tictactoe.js +0 -389
- package/games/pixelart.js +0 -399
- package/games/quickduel.js +0 -354
- package/games/riddle.js +0 -371
- package/games/rockpaperscissors.js +0 -291
- package/games/snake.js +0 -406
- package/games/storybuilder.js +0 -343
- package/games/tictactoe.js +0 -345
- package/games/twentyquestions.js +0 -286
- package/games/twotruths.js +0 -207
- package/games/werewolf.js +0 -508
- package/games/wordassociation.js +0 -247
- package/games/wordchain.js +0 -135
- package/intelligence/interests.js +0 -369
- package/setup.js +0 -480
- package/smart-inbox.js +0 -276
- package/tools/_deprecated/auto-suggest-connections.js +0 -304
- package/tools/_deprecated/bootstrap-skills.js +0 -231
- package/tools/_deprecated/bridge-dashboard.js +0 -342
- package/tools/_deprecated/bridge-health.js +0 -400
- package/tools/_deprecated/bridge-live.js +0 -384
- package/tools/_deprecated/bridges.js +0 -383
- package/tools/_deprecated/colorguess.js +0 -281
- package/tools/_deprecated/discover-insights.js +0 -379
- package/tools/_deprecated/discover-momentum.js +0 -256
- package/tools/_deprecated/discovery-analytics.js +0 -345
- package/tools/_deprecated/discovery-auto-suggest.js +0 -275
- package/tools/_deprecated/discovery-bootstrap.js +0 -267
- package/tools/_deprecated/discovery-daily.js +0 -375
- package/tools/_deprecated/discovery-dashboard.js +0 -385
- package/tools/_deprecated/discovery-digest.js +0 -314
- package/tools/_deprecated/discovery-hub.js +0 -357
- package/tools/_deprecated/discovery-insights.js +0 -384
- package/tools/_deprecated/discovery-momentum.js +0 -281
- package/tools/_deprecated/discovery-monitor.js +0 -319
- package/tools/_deprecated/discovery-proactive.js +0 -300
- package/tools/_deprecated/draw.js +0 -317
- package/tools/_deprecated/farcaster.js +0 -307
- package/tools/_deprecated/games-catalog.js +0 -376
- package/tools/_deprecated/games.js +0 -313
- package/tools/_deprecated/guessnumber.js +0 -194
- package/tools/_deprecated/hangman.js +0 -129
- package/tools/_deprecated/multiplayer-tictactoe.js +0 -303
- package/tools/_deprecated/riddle.js +0 -240
- package/tools/_deprecated/run-bootstrap.js +0 -69
- package/tools/_deprecated/skills-analytics.js +0 -349
- package/tools/_deprecated/skills-bootstrap.js +0 -301
- package/tools/_deprecated/skills-dashboard.js +0 -268
- package/tools/_deprecated/skills.js +0 -380
- package/tools/_deprecated/smart-intro.js +0 -353
- package/tools/_deprecated/storybuilder.js +0 -331
- package/tools/_deprecated/telegram-bot.js +0 -183
- package/tools/_deprecated/telegram-setup.js +0 -214
- package/tools/_deprecated/twentyquestions.js +0 -143
- package/tools/_shared.js +0 -234
- package/tools/_work-context.js +0 -338
- package/tools/_work-context.manual-test.js +0 -199
- package/tools/_work-context.test.js +0 -260
- package/tools/activity.js +0 -220
- package/tools/agent-treasury.js +0 -288
- package/tools/analytics.js +0 -191
- package/tools/approve.js +0 -197
- package/tools/arcade.js +0 -173
- package/tools/artifacts-price.js +0 -107
- package/tools/ask-expert.js +0 -160
- package/tools/available.js +0 -120
- package/tools/become-expert.js +0 -150
- package/tools/broadcast.js +0 -325
- package/tools/chat.js +0 -202
- package/tools/collaborative-drawing.js +0 -286
- package/tools/connection-status.js +0 -178
- package/tools/earnings.js +0 -126
- package/tools/friends.js +0 -207
- package/tools/genesis.js +0 -233
- package/tools/gig-browse.js +0 -206
- package/tools/gig-complete.js +0 -144
- package/tools/health.js +0 -87
- package/tools/leaderboard.js +0 -117
- package/tools/lib/git-apply.js +0 -206
- package/tools/lib/git-bundle.js +0 -407
- package/tools/mint.js +0 -377
- package/tools/plan.js +0 -225
- package/tools/profile.js +0 -219
- package/tools/proof-of-work.js +0 -144
- package/tools/pulse.js +0 -218
- package/tools/reply.js +0 -166
- package/tools/reputation.js +0 -175
- package/tools/schedule.js +0 -367
- package/tools/search-messages.js +0 -123
- package/tools/session.js +0 -467
- package/tools/session_price.js +0 -128
- package/tools/smart-check.js +0 -201
- package/tools/social-processor.js +0 -445
- package/tools/streak.js +0 -147
- package/tools/stuck.js +0 -297
- package/tools/subscribe.js +0 -148
- package/tools/subscriptions.js +0 -134
- package/tools/tip.js +0 -193
- package/tools/wallet.js +0 -269
- package/tools/webhook-test.js +0 -388
- package/tools/withdraw.js +0 -145
- package/tools/work-summary.js +0 -96
- package/tools/workshop.js +0 -327
- /package/tools/{l2-bridge.js → _experimental/l2-bridge.js} +0 -0
- /package/tools/{l2.js → _experimental/l2.js} +0 -0
- /package/tools/{_deprecated/away.js → away.js} +0 -0
|
@@ -1,231 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* vibe bootstrap-skills — Bootstrap Skills Exchange with sample data
|
|
3
|
-
*
|
|
4
|
-
* Creates sample skill offerings and requests to populate the marketplace.
|
|
5
|
-
* Only runs if no existing skill exchange data exists.
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
const store = require('../store');
|
|
9
|
-
const { requireInit } = require('./_shared');
|
|
10
|
-
|
|
11
|
-
const definition = {
|
|
12
|
-
name: 'vibe_bootstrap_skills',
|
|
13
|
-
description: 'Bootstrap the skills exchange marketplace with sample data.',
|
|
14
|
-
inputSchema: {
|
|
15
|
-
type: 'object',
|
|
16
|
-
properties: {
|
|
17
|
-
force: {
|
|
18
|
-
type: 'boolean',
|
|
19
|
-
description: 'Force bootstrap even if data exists',
|
|
20
|
-
default: false
|
|
21
|
-
}
|
|
22
|
-
}
|
|
23
|
-
}
|
|
24
|
-
};
|
|
25
|
-
|
|
26
|
-
// Sample skill posts to bootstrap the marketplace
|
|
27
|
-
const samplePosts = [
|
|
28
|
-
// Technical Skills
|
|
29
|
-
{
|
|
30
|
-
handle: 'alice',
|
|
31
|
-
type: 'offer',
|
|
32
|
-
skill: 'React development',
|
|
33
|
-
details: '5+ years experience, can help with components, state management, and performance optimization',
|
|
34
|
-
category: 'technical'
|
|
35
|
-
},
|
|
36
|
-
{
|
|
37
|
-
handle: 'alice',
|
|
38
|
-
type: 'offer',
|
|
39
|
-
skill: 'TypeScript mentoring',
|
|
40
|
-
details: 'Help with type safety, advanced patterns, and migration strategies',
|
|
41
|
-
category: 'technical'
|
|
42
|
-
},
|
|
43
|
-
{
|
|
44
|
-
handle: 'alice',
|
|
45
|
-
type: 'request',
|
|
46
|
-
skill: 'Machine learning model deployment',
|
|
47
|
-
details: 'Need help deploying ML models at scale, particularly with Docker and Kubernetes',
|
|
48
|
-
category: 'technical'
|
|
49
|
-
},
|
|
50
|
-
{
|
|
51
|
-
handle: 'bob',
|
|
52
|
-
type: 'offer',
|
|
53
|
-
skill: 'Python backend development',
|
|
54
|
-
details: 'FastAPI, Django, microservices architecture, and API design',
|
|
55
|
-
category: 'technical'
|
|
56
|
-
},
|
|
57
|
-
{
|
|
58
|
-
handle: 'bob',
|
|
59
|
-
type: 'offer',
|
|
60
|
-
skill: 'Kubernetes deployment',
|
|
61
|
-
details: 'Container orchestration, auto-scaling, and DevOps best practices',
|
|
62
|
-
category: 'technical'
|
|
63
|
-
},
|
|
64
|
-
{
|
|
65
|
-
handle: 'bob',
|
|
66
|
-
type: 'request',
|
|
67
|
-
skill: 'Frontend UI/UX feedback',
|
|
68
|
-
details: 'Backend engineer looking for design sense and user experience guidance',
|
|
69
|
-
category: 'design'
|
|
70
|
-
},
|
|
71
|
-
|
|
72
|
-
// Design Skills
|
|
73
|
-
{
|
|
74
|
-
handle: 'carol',
|
|
75
|
-
type: 'offer',
|
|
76
|
-
skill: 'Figma design systems',
|
|
77
|
-
details: 'Creating scalable design systems, component libraries, and design tokens',
|
|
78
|
-
category: 'design'
|
|
79
|
-
},
|
|
80
|
-
{
|
|
81
|
-
handle: 'carol',
|
|
82
|
-
type: 'offer',
|
|
83
|
-
skill: 'UX research methodology',
|
|
84
|
-
details: 'User interviews, usability testing, and research planning',
|
|
85
|
-
category: 'design'
|
|
86
|
-
},
|
|
87
|
-
{
|
|
88
|
-
handle: 'carol',
|
|
89
|
-
type: 'request',
|
|
90
|
-
skill: 'React component implementation',
|
|
91
|
-
details: 'Need help turning designs into high-quality React components',
|
|
92
|
-
category: 'technical'
|
|
93
|
-
},
|
|
94
|
-
|
|
95
|
-
// Mobile & Crypto
|
|
96
|
-
{
|
|
97
|
-
handle: 'dave',
|
|
98
|
-
type: 'offer',
|
|
99
|
-
skill: 'React Native development',
|
|
100
|
-
details: 'Cross-platform mobile apps, navigation, and native module integration',
|
|
101
|
-
category: 'technical'
|
|
102
|
-
},
|
|
103
|
-
{
|
|
104
|
-
handle: 'dave',
|
|
105
|
-
type: 'offer',
|
|
106
|
-
skill: 'Blockchain integration',
|
|
107
|
-
details: 'Smart contracts, Web3 APIs, and crypto trading algorithms',
|
|
108
|
-
category: 'technical'
|
|
109
|
-
},
|
|
110
|
-
{
|
|
111
|
-
handle: 'dave',
|
|
112
|
-
type: 'request',
|
|
113
|
-
skill: 'Product strategy',
|
|
114
|
-
details: 'Looking for help with go-to-market strategy and user acquisition',
|
|
115
|
-
category: 'business'
|
|
116
|
-
},
|
|
117
|
-
|
|
118
|
-
// Content & AI
|
|
119
|
-
{
|
|
120
|
-
handle: 'eve',
|
|
121
|
-
type: 'offer',
|
|
122
|
-
skill: 'Content strategy',
|
|
123
|
-
details: 'Blog writing, social media content, and content marketing strategies',
|
|
124
|
-
category: 'creative'
|
|
125
|
-
},
|
|
126
|
-
{
|
|
127
|
-
handle: 'eve',
|
|
128
|
-
type: 'offer',
|
|
129
|
-
skill: 'GPT prompt engineering',
|
|
130
|
-
details: 'Optimizing AI prompts for better outputs, fine-tuning strategies',
|
|
131
|
-
category: 'technical'
|
|
132
|
-
},
|
|
133
|
-
{
|
|
134
|
-
handle: 'eve',
|
|
135
|
-
type: 'request',
|
|
136
|
-
skill: 'Backend API development',
|
|
137
|
-
details: 'Need help building robust APIs for AI content generation tools',
|
|
138
|
-
category: 'technical'
|
|
139
|
-
}
|
|
140
|
-
];
|
|
141
|
-
|
|
142
|
-
async function createSkillPost(postData) {
|
|
143
|
-
const post = {
|
|
144
|
-
id: Date.now() + Math.random(),
|
|
145
|
-
...postData,
|
|
146
|
-
timestamp: Date.now() - Math.floor(Math.random() * 3 * 24 * 60 * 60 * 1000), // Random time in last 3 days
|
|
147
|
-
status: 'active'
|
|
148
|
-
};
|
|
149
|
-
|
|
150
|
-
await store.appendSkillExchange(post);
|
|
151
|
-
return post;
|
|
152
|
-
}
|
|
153
|
-
|
|
154
|
-
async function handler(args) {
|
|
155
|
-
const initCheck = requireInit();
|
|
156
|
-
if (initCheck) return initCheck;
|
|
157
|
-
|
|
158
|
-
let display = '';
|
|
159
|
-
|
|
160
|
-
try {
|
|
161
|
-
// Check if skills exchange already has posts
|
|
162
|
-
const existingPosts = await store.getSkillExchanges() || [];
|
|
163
|
-
|
|
164
|
-
if (existingPosts.length > 0 && !args.force) {
|
|
165
|
-
display = `## Skills Exchange Already Active! 🎯\n\n`;
|
|
166
|
-
display += `Found ${existingPosts.length} existing skill posts.\n\n`;
|
|
167
|
-
display += `**Current Status:**\n`;
|
|
168
|
-
|
|
169
|
-
const offers = existingPosts.filter(p => p.type === 'offer').length;
|
|
170
|
-
const requests = existingPosts.filter(p => p.type === 'request').length;
|
|
171
|
-
|
|
172
|
-
display += `• ${offers} skill offerings available\n`;
|
|
173
|
-
display += `• ${requests} skill requests posted\n`;
|
|
174
|
-
display += `• Ready for matching and connections\n\n`;
|
|
175
|
-
|
|
176
|
-
display += `**Use the marketplace:**\n`;
|
|
177
|
-
display += `• \`skills-exchange browse\` — See all postings\n`;
|
|
178
|
-
display += `• \`skills-exchange match\` — Find your matches\n`;
|
|
179
|
-
display += `• \`skills-exchange post\` — Add your skills\n\n`;
|
|
180
|
-
|
|
181
|
-
display += `**Force refresh:** \`bootstrap-skills --force true\``;
|
|
182
|
-
|
|
183
|
-
return { display };
|
|
184
|
-
}
|
|
185
|
-
|
|
186
|
-
display = `## Bootstrapping Skills Exchange Marketplace 🚀\n\n`;
|
|
187
|
-
display += `Creating sample skill posts to populate the marketplace...\n\n`;
|
|
188
|
-
|
|
189
|
-
// Create all sample posts
|
|
190
|
-
let created = 0;
|
|
191
|
-
for (const postData of samplePosts) {
|
|
192
|
-
await createSkillPost(postData);
|
|
193
|
-
created++;
|
|
194
|
-
}
|
|
195
|
-
|
|
196
|
-
display += `✅ **Successfully created ${created} skill posts!**\n\n`;
|
|
197
|
-
|
|
198
|
-
// Summary
|
|
199
|
-
const offers = samplePosts.filter(p => p.type === 'offer').length;
|
|
200
|
-
const requests = samplePosts.filter(p => p.type === 'request').length;
|
|
201
|
-
const categories = [...new Set(samplePosts.map(p => p.category))];
|
|
202
|
-
const users = [...new Set(samplePosts.map(p => p.handle))];
|
|
203
|
-
|
|
204
|
-
display += `**Marketplace Overview:**\n`;
|
|
205
|
-
display += `• ${offers} skill offerings from experts\n`;
|
|
206
|
-
display += `• ${requests} skill requests from learners\n`;
|
|
207
|
-
display += `• ${categories.length} categories: ${categories.join(', ')}\n`;
|
|
208
|
-
display += `• ${users.length} active users: ${users.map(u => '@' + u).join(', ')}\n\n`;
|
|
209
|
-
|
|
210
|
-
display += `**Perfect Match Examples:**\n`;
|
|
211
|
-
display += `• Alice (React expert) ↔ Carol (needs React components)\n`;
|
|
212
|
-
display += `• Bob (Python/K8s) ↔ Alice (needs ML deployment)\n`;
|
|
213
|
-
display += `• Carol (UX/Design) ↔ Bob (needs UI feedback)\n`;
|
|
214
|
-
display += `• Dave (Mobile dev) ↔ Eve (needs backend APIs)\n\n`;
|
|
215
|
-
|
|
216
|
-
display += `**Ready to Use:**\n`;
|
|
217
|
-
display += `• \`skills-exchange browse\` — Explore all offerings\n`;
|
|
218
|
-
display += `• \`skills-exchange match\` — Find personalized matches\n`;
|
|
219
|
-
display += `• \`skills-exchange post --type offer --skill "your expertise"\`\n`;
|
|
220
|
-
display += `• \`dm @username "I saw your skills post..."\` — Connect!\n\n`;
|
|
221
|
-
|
|
222
|
-
display += `🎯 **The Skills Exchange marketplace is now live and ready for connections!**`;
|
|
223
|
-
|
|
224
|
-
} catch (error) {
|
|
225
|
-
display = `## Bootstrap Skills Error\n\n${error.message}\n\nTry: \`bootstrap-skills\` to retry`;
|
|
226
|
-
}
|
|
227
|
-
|
|
228
|
-
return { display };
|
|
229
|
-
}
|
|
230
|
-
|
|
231
|
-
module.exports = { definition, handler };
|
|
@@ -1,342 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* vibe bridge-dashboard — Real-time bridge monitoring dashboard
|
|
3
|
-
*
|
|
4
|
-
* Live view of all bridge health, recent events, and webhook activity.
|
|
5
|
-
* Auto-refreshes to show real-time status of social connections.
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
const twitter = require('../twitter');
|
|
9
|
-
const telegram = require('../bridges/telegram');
|
|
10
|
-
const discord = require('../discord');
|
|
11
|
-
const farcaster = require('../bridges/farcaster');
|
|
12
|
-
const { requireInit, header, divider, success, warning, error } = require('./_shared');
|
|
13
|
-
|
|
14
|
-
const definition = {
|
|
15
|
-
name: 'vibe_bridge_dashboard',
|
|
16
|
-
description: 'Real-time monitoring dashboard for all social bridges and webhook activity',
|
|
17
|
-
inputSchema: {
|
|
18
|
-
type: 'object',
|
|
19
|
-
properties: {
|
|
20
|
-
refresh_rate: {
|
|
21
|
-
type: 'number',
|
|
22
|
-
description: 'Auto-refresh interval in seconds (0 = no auto-refresh, default: 30)'
|
|
23
|
-
},
|
|
24
|
-
show_events: {
|
|
25
|
-
type: 'boolean',
|
|
26
|
-
description: 'Show recent webhook events (default: true)'
|
|
27
|
-
},
|
|
28
|
-
compact: {
|
|
29
|
-
type: 'boolean',
|
|
30
|
-
description: 'Compact view with minimal details (default: false)'
|
|
31
|
-
}
|
|
32
|
-
}
|
|
33
|
-
}
|
|
34
|
-
};
|
|
35
|
-
|
|
36
|
-
async function handler(args) {
|
|
37
|
-
const initCheck = requireInit();
|
|
38
|
-
if (initCheck) return initCheck;
|
|
39
|
-
|
|
40
|
-
const {
|
|
41
|
-
refresh_rate = 30,
|
|
42
|
-
show_events = true,
|
|
43
|
-
compact = false
|
|
44
|
-
} = args;
|
|
45
|
-
|
|
46
|
-
try {
|
|
47
|
-
// Get comprehensive bridge status
|
|
48
|
-
const bridgeStatuses = await getAllBridgeStatuses();
|
|
49
|
-
const webhookStats = await getWebhookStats();
|
|
50
|
-
const recentEvents = show_events ? await getRecentWebhookEvents(10) : [];
|
|
51
|
-
|
|
52
|
-
return formatDashboard(bridgeStatuses, webhookStats, recentEvents, compact, refresh_rate);
|
|
53
|
-
|
|
54
|
-
} catch (e) {
|
|
55
|
-
return {
|
|
56
|
-
display: `${header('Bridge Dashboard')}\n\n${error('Dashboard error: ' + e.message)}`
|
|
57
|
-
};
|
|
58
|
-
}
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
async function getAllBridgeStatuses() {
|
|
62
|
-
const statuses = {};
|
|
63
|
-
|
|
64
|
-
// X Bridge
|
|
65
|
-
statuses.x = await getBridgeStatus('x', twitter);
|
|
66
|
-
|
|
67
|
-
// Telegram Bridge
|
|
68
|
-
statuses.telegram = await getBridgeStatus('telegram', telegram);
|
|
69
|
-
|
|
70
|
-
// Discord Bridge
|
|
71
|
-
statuses.discord = await getBridgeStatus('discord', discord);
|
|
72
|
-
|
|
73
|
-
// Farcaster Bridge
|
|
74
|
-
statuses.farcaster = await getBridgeStatus('farcaster', farcaster);
|
|
75
|
-
|
|
76
|
-
return statuses;
|
|
77
|
-
}
|
|
78
|
-
|
|
79
|
-
async function getBridgeStatus(platform, module) {
|
|
80
|
-
const status = {
|
|
81
|
-
platform,
|
|
82
|
-
configured: module.isConfigured(),
|
|
83
|
-
connected: false,
|
|
84
|
-
username: null,
|
|
85
|
-
lastActivity: null,
|
|
86
|
-
errors: [],
|
|
87
|
-
capabilities: getCapabilities(platform),
|
|
88
|
-
health: 'unknown'
|
|
89
|
-
};
|
|
90
|
-
|
|
91
|
-
if (!status.configured) {
|
|
92
|
-
status.health = 'not_configured';
|
|
93
|
-
return status;
|
|
94
|
-
}
|
|
95
|
-
|
|
96
|
-
try {
|
|
97
|
-
// Test connection based on platform
|
|
98
|
-
switch (platform) {
|
|
99
|
-
case 'x':
|
|
100
|
-
const me = await twitter.getMe();
|
|
101
|
-
status.connected = true;
|
|
102
|
-
status.username = me.data.username;
|
|
103
|
-
status.health = 'healthy';
|
|
104
|
-
break;
|
|
105
|
-
|
|
106
|
-
case 'telegram':
|
|
107
|
-
const botInfo = await telegram.getBotInfo();
|
|
108
|
-
status.connected = true;
|
|
109
|
-
status.username = botInfo.username;
|
|
110
|
-
status.health = 'healthy';
|
|
111
|
-
break;
|
|
112
|
-
|
|
113
|
-
case 'discord':
|
|
114
|
-
// Discord webhook doesn't have a test endpoint, assume healthy if configured
|
|
115
|
-
status.connected = true;
|
|
116
|
-
status.health = 'healthy';
|
|
117
|
-
status.username = 'webhook';
|
|
118
|
-
break;
|
|
119
|
-
|
|
120
|
-
case 'farcaster':
|
|
121
|
-
const userInfo = await farcaster.getUser();
|
|
122
|
-
status.connected = true;
|
|
123
|
-
status.username = userInfo.users[0].username;
|
|
124
|
-
status.health = 'healthy';
|
|
125
|
-
break;
|
|
126
|
-
}
|
|
127
|
-
|
|
128
|
-
// Check for recent activity (would require KV access for webhook events)
|
|
129
|
-
status.lastActivity = await getLastActivity(platform);
|
|
130
|
-
|
|
131
|
-
} catch (e) {
|
|
132
|
-
status.health = 'error';
|
|
133
|
-
status.errors.push(e.message);
|
|
134
|
-
}
|
|
135
|
-
|
|
136
|
-
return status;
|
|
137
|
-
}
|
|
138
|
-
|
|
139
|
-
async function getWebhookStats() {
|
|
140
|
-
try {
|
|
141
|
-
// Try to get KV store
|
|
142
|
-
const KV_CONFIGURED = !!(process.env.KV_REST_API_URL && process.env.KV_REST_API_TOKEN);
|
|
143
|
-
if (!KV_CONFIGURED) {
|
|
144
|
-
return {
|
|
145
|
-
x: { total_deliveries: 0, events_processed: 0, last_delivery: null }
|
|
146
|
-
};
|
|
147
|
-
}
|
|
148
|
-
|
|
149
|
-
const { kv } = await import('@vercel/kv');
|
|
150
|
-
|
|
151
|
-
// Get X webhook stats
|
|
152
|
-
const xStats = await kv.hgetall('vibe:x_webhook_stats') || {};
|
|
153
|
-
|
|
154
|
-
return {
|
|
155
|
-
x: {
|
|
156
|
-
total_deliveries: parseInt(xStats.total_deliveries) || 0,
|
|
157
|
-
events_processed: parseInt(xStats.events_processed) || 0,
|
|
158
|
-
last_delivery: xStats.last_delivery || null,
|
|
159
|
-
kv_available: true
|
|
160
|
-
}
|
|
161
|
-
};
|
|
162
|
-
|
|
163
|
-
} catch (e) {
|
|
164
|
-
return {
|
|
165
|
-
x: { total_deliveries: 0, events_processed: 0, last_delivery: null, error: e.message }
|
|
166
|
-
};
|
|
167
|
-
}
|
|
168
|
-
}
|
|
169
|
-
|
|
170
|
-
async function getRecentWebhookEvents(limit) {
|
|
171
|
-
try {
|
|
172
|
-
const KV_CONFIGURED = !!(process.env.KV_REST_API_URL && process.env.KV_REST_API_TOKEN);
|
|
173
|
-
if (!KV_CONFIGURED) return [];
|
|
174
|
-
|
|
175
|
-
const { kv } = await import('@vercel/kv');
|
|
176
|
-
const inboxKey = 'vibe:social_inbox';
|
|
177
|
-
|
|
178
|
-
const rawEvents = await kv.lrange(inboxKey, 0, limit - 1);
|
|
179
|
-
|
|
180
|
-
return rawEvents.map(eventStr => {
|
|
181
|
-
const event = JSON.parse(eventStr);
|
|
182
|
-
return {
|
|
183
|
-
id: event.id,
|
|
184
|
-
platform: event.platform,
|
|
185
|
-
type: event.type,
|
|
186
|
-
from: event.from?.handle || 'unknown',
|
|
187
|
-
timestamp: event.timestamp,
|
|
188
|
-
timeAgo: formatTimeAgo(new Date(event.timestamp))
|
|
189
|
-
};
|
|
190
|
-
});
|
|
191
|
-
|
|
192
|
-
} catch (e) {
|
|
193
|
-
return [];
|
|
194
|
-
}
|
|
195
|
-
}
|
|
196
|
-
|
|
197
|
-
async function getLastActivity(platform) {
|
|
198
|
-
// Placeholder - would check KV for last webhook event from this platform
|
|
199
|
-
return null;
|
|
200
|
-
}
|
|
201
|
-
|
|
202
|
-
function getCapabilities(platform) {
|
|
203
|
-
const capabilities = {
|
|
204
|
-
x: ['read', 'write', 'dm', 'webhook'],
|
|
205
|
-
telegram: ['read', 'write', 'groups', 'bot'],
|
|
206
|
-
discord: ['write', 'webhook'],
|
|
207
|
-
farcaster: ['read', 'write', 'channels', 'frames']
|
|
208
|
-
};
|
|
209
|
-
return capabilities[platform] || [];
|
|
210
|
-
}
|
|
211
|
-
|
|
212
|
-
function formatTimeAgo(date) {
|
|
213
|
-
const now = new Date();
|
|
214
|
-
const diffMs = now - date;
|
|
215
|
-
const diffMins = Math.floor(diffMs / (1000 * 60));
|
|
216
|
-
const diffHours = Math.floor(diffMs / (1000 * 60 * 60));
|
|
217
|
-
const diffDays = Math.floor(diffMs / (1000 * 60 * 60 * 24));
|
|
218
|
-
|
|
219
|
-
if (diffMins < 1) return 'just now';
|
|
220
|
-
if (diffMins < 60) return `${diffMins}m ago`;
|
|
221
|
-
if (diffHours < 24) return `${diffHours}h ago`;
|
|
222
|
-
return `${diffDays}d ago`;
|
|
223
|
-
}
|
|
224
|
-
|
|
225
|
-
function formatDashboard(bridges, webhookStats, events, compact, refreshRate) {
|
|
226
|
-
let display = header('🔗 Bridge Dashboard');
|
|
227
|
-
display += `\n\n_Live status • ${new Date().toLocaleTimeString()}_\n`;
|
|
228
|
-
|
|
229
|
-
if (refreshRate > 0) {
|
|
230
|
-
display += `_Auto-refresh: ${refreshRate}s_\n`;
|
|
231
|
-
}
|
|
232
|
-
|
|
233
|
-
display += divider();
|
|
234
|
-
display += '\n';
|
|
235
|
-
|
|
236
|
-
// Overall health summary
|
|
237
|
-
const totalBridges = Object.keys(bridges).length;
|
|
238
|
-
const connectedBridges = Object.values(bridges).filter(b => b.connected).length;
|
|
239
|
-
const healthyBridges = Object.values(bridges).filter(b => b.health === 'healthy').length;
|
|
240
|
-
|
|
241
|
-
let healthIcon = '✅';
|
|
242
|
-
if (healthyBridges === 0) healthIcon = '❌';
|
|
243
|
-
else if (healthyBridges < connectedBridges) healthIcon = '⚠️';
|
|
244
|
-
|
|
245
|
-
display += `${healthIcon} **System Health:** ${healthyBridges}/${totalBridges} bridges healthy\n`;
|
|
246
|
-
|
|
247
|
-
// Quick stats
|
|
248
|
-
if (webhookStats.x.total_deliveries > 0) {
|
|
249
|
-
display += `🔔 **Webhooks:** ${webhookStats.x.total_deliveries} deliveries, ${webhookStats.x.events_processed} events processed\n`;
|
|
250
|
-
}
|
|
251
|
-
|
|
252
|
-
if (events.length > 0) {
|
|
253
|
-
const recentCount = events.filter(e => new Date() - new Date(e.timestamp) < 60 * 60 * 1000).length;
|
|
254
|
-
display += `📬 **Recent Activity:** ${recentCount} events in last hour\n`;
|
|
255
|
-
}
|
|
256
|
-
|
|
257
|
-
display += '\n' + divider();
|
|
258
|
-
display += '**BRIDGE STATUS**\n\n';
|
|
259
|
-
|
|
260
|
-
// Individual bridge status
|
|
261
|
-
for (const [platformName, bridge] of Object.entries(bridges)) {
|
|
262
|
-
display += formatBridgeQuickStatus(bridge, compact);
|
|
263
|
-
}
|
|
264
|
-
|
|
265
|
-
// Recent webhook events
|
|
266
|
-
if (events.length > 0) {
|
|
267
|
-
display += '\n' + divider();
|
|
268
|
-
display += '**RECENT EVENTS**\n\n';
|
|
269
|
-
|
|
270
|
-
for (const event of events.slice(0, compact ? 3 : 5)) {
|
|
271
|
-
display += formatEventQuickStatus(event);
|
|
272
|
-
}
|
|
273
|
-
|
|
274
|
-
if (events.length > (compact ? 3 : 5)) {
|
|
275
|
-
display += `_...and ${events.length - (compact ? 3 : 5)} more events_\n`;
|
|
276
|
-
}
|
|
277
|
-
}
|
|
278
|
-
|
|
279
|
-
// Quick actions
|
|
280
|
-
display += '\n' + divider();
|
|
281
|
-
display += '**QUICK ACTIONS**\n';
|
|
282
|
-
display += '• `vibe social-inbox` - View all messages\n';
|
|
283
|
-
display += '• `vibe bridge-health --details` - Detailed diagnostics\n';
|
|
284
|
-
display += '• `vibe bridges --action test --platform x` - Test specific bridge\n';
|
|
285
|
-
display += '• `vibe bridge-dashboard --compact` - Compact view';
|
|
286
|
-
|
|
287
|
-
return { display };
|
|
288
|
-
}
|
|
289
|
-
|
|
290
|
-
function formatBridgeQuickStatus(bridge, compact) {
|
|
291
|
-
const statusIcons = {
|
|
292
|
-
healthy: '🟢',
|
|
293
|
-
error: '🔴',
|
|
294
|
-
not_configured: '⚪',
|
|
295
|
-
unknown: '🟡'
|
|
296
|
-
};
|
|
297
|
-
|
|
298
|
-
const icon = statusIcons[bridge.health] || '🟡';
|
|
299
|
-
const platformName = bridge.platform.toUpperCase().padEnd(10);
|
|
300
|
-
|
|
301
|
-
let line = `${icon} ${platformName}`;
|
|
302
|
-
|
|
303
|
-
if (bridge.connected && bridge.username) {
|
|
304
|
-
line += ` @${bridge.username}`;
|
|
305
|
-
} else if (bridge.health === 'not_configured') {
|
|
306
|
-
line += ` Not configured`;
|
|
307
|
-
} else if (bridge.errors.length > 0) {
|
|
308
|
-
line += ` ${bridge.errors[0]}`;
|
|
309
|
-
}
|
|
310
|
-
|
|
311
|
-
line += '\n';
|
|
312
|
-
|
|
313
|
-
if (!compact && bridge.capabilities.length > 0) {
|
|
314
|
-
line += ` ${bridge.capabilities.join(' • ')}\n`;
|
|
315
|
-
}
|
|
316
|
-
|
|
317
|
-
return line;
|
|
318
|
-
}
|
|
319
|
-
|
|
320
|
-
function formatEventQuickStatus(event) {
|
|
321
|
-
const platformIcons = {
|
|
322
|
-
x: '𝕏',
|
|
323
|
-
farcaster: '🟣',
|
|
324
|
-
telegram: '✈️',
|
|
325
|
-
discord: '💬'
|
|
326
|
-
};
|
|
327
|
-
|
|
328
|
-
const typeIcons = {
|
|
329
|
-
mention: '@',
|
|
330
|
-
dm: '✉️',
|
|
331
|
-
like: '❤️',
|
|
332
|
-
follow: '👤',
|
|
333
|
-
cast: '📡'
|
|
334
|
-
};
|
|
335
|
-
|
|
336
|
-
const platformIcon = platformIcons[event.platform] || '📱';
|
|
337
|
-
const typeIcon = typeIcons[event.type] || '';
|
|
338
|
-
|
|
339
|
-
return `${platformIcon} ${typeIcon} @${event.from} • ${event.timeAgo}\n`;
|
|
340
|
-
}
|
|
341
|
-
|
|
342
|
-
module.exports = { definition, handler };
|