slashvibe-mcp 0.3.21 → 0.3.22
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/config.js +36 -31
- package/crypto.js +1 -6
- package/discord.js +19 -19
- 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/notify.js +39 -14
- package/package.json +27 -20
- 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/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/analytics.js +0 -107
- package/auth-store.js +0 -148
- package/auto-update.js +0 -130
- 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/notification-emitter.js +0 -77
- 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,400 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* vibe bridge-health — Monitor and diagnose bridge connections
|
|
3
|
-
*
|
|
4
|
-
* Real-time health monitoring for all social bridges.
|
|
5
|
-
* Checks connectivity, rate limits, and provides recovery suggestions.
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
const twitter = require('../twitter');
|
|
9
|
-
const telegram = require('../bridges/telegram');
|
|
10
|
-
const discord = require('../discord');
|
|
11
|
-
const discordBot = require('../bridges/discord-bot');
|
|
12
|
-
const farcaster = require('../bridges/farcaster');
|
|
13
|
-
const { requireInit, header, divider, success, warning, error } = require('./_shared');
|
|
14
|
-
|
|
15
|
-
const definition = {
|
|
16
|
-
name: 'vibe_bridge_health',
|
|
17
|
-
description: 'Monitor health and connectivity of all social bridges',
|
|
18
|
-
inputSchema: {
|
|
19
|
-
type: 'object',
|
|
20
|
-
properties: {
|
|
21
|
-
platform: {
|
|
22
|
-
type: 'string',
|
|
23
|
-
enum: ['all', 'x', 'twitter', 'telegram', 'discord', 'farcaster'],
|
|
24
|
-
description: 'Check specific platform or all (default: all)'
|
|
25
|
-
},
|
|
26
|
-
fix: {
|
|
27
|
-
type: 'boolean',
|
|
28
|
-
description: 'Attempt to fix detected issues (default: false)'
|
|
29
|
-
},
|
|
30
|
-
details: {
|
|
31
|
-
type: 'boolean',
|
|
32
|
-
description: 'Show detailed diagnostics (default: false)'
|
|
33
|
-
}
|
|
34
|
-
}
|
|
35
|
-
}
|
|
36
|
-
};
|
|
37
|
-
|
|
38
|
-
async function handler(args) {
|
|
39
|
-
const initCheck = requireInit();
|
|
40
|
-
if (initCheck) return initCheck;
|
|
41
|
-
|
|
42
|
-
const { platform = 'all', fix = false, details = false } = args;
|
|
43
|
-
|
|
44
|
-
try {
|
|
45
|
-
const results = await runHealthChecks(platform);
|
|
46
|
-
return formatHealthReport(results, fix, details);
|
|
47
|
-
} catch (e) {
|
|
48
|
-
return {
|
|
49
|
-
display: `${header('Bridge Health Check')}\n\n${error('Health check failed: ' + e.message)}`
|
|
50
|
-
};
|
|
51
|
-
}
|
|
52
|
-
}
|
|
53
|
-
|
|
54
|
-
async function runHealthChecks(platform) {
|
|
55
|
-
const checks = {};
|
|
56
|
-
|
|
57
|
-
if (platform === 'all' || platform === 'x' || platform === 'twitter') {
|
|
58
|
-
checks.x = await checkXHealth();
|
|
59
|
-
}
|
|
60
|
-
|
|
61
|
-
if (platform === 'all' || platform === 'telegram') {
|
|
62
|
-
checks.telegram = await checkTelegramHealth();
|
|
63
|
-
}
|
|
64
|
-
|
|
65
|
-
if (platform === 'all' || platform === 'discord') {
|
|
66
|
-
checks.discord = await checkDiscordHealth();
|
|
67
|
-
}
|
|
68
|
-
|
|
69
|
-
if (platform === 'all' || platform === 'farcaster') {
|
|
70
|
-
checks.farcaster = await checkFarcasterHealth();
|
|
71
|
-
}
|
|
72
|
-
|
|
73
|
-
return checks;
|
|
74
|
-
}
|
|
75
|
-
|
|
76
|
-
async function checkXHealth() {
|
|
77
|
-
const health = {
|
|
78
|
-
platform: 'x',
|
|
79
|
-
configured: twitter.isConfigured(),
|
|
80
|
-
connected: false,
|
|
81
|
-
canRead: false,
|
|
82
|
-
canWrite: false,
|
|
83
|
-
rateLimits: {},
|
|
84
|
-
issues: [],
|
|
85
|
-
suggestions: [],
|
|
86
|
-
lastCheck: new Date().toISOString()
|
|
87
|
-
};
|
|
88
|
-
|
|
89
|
-
if (!health.configured) {
|
|
90
|
-
health.issues.push('API credentials not configured');
|
|
91
|
-
health.suggestions.push('Add X API credentials to config.json');
|
|
92
|
-
health.suggestions.push('Requires paid X API access for posting');
|
|
93
|
-
return health;
|
|
94
|
-
}
|
|
95
|
-
|
|
96
|
-
try {
|
|
97
|
-
// Test authentication
|
|
98
|
-
const me = await twitter.getMe();
|
|
99
|
-
health.connected = true;
|
|
100
|
-
health.username = me.data.username;
|
|
101
|
-
health.userId = me.data.id;
|
|
102
|
-
health.canRead = true;
|
|
103
|
-
|
|
104
|
-
// Test read capability
|
|
105
|
-
try {
|
|
106
|
-
await twitter.getMentions();
|
|
107
|
-
health.canRead = true;
|
|
108
|
-
} catch (e) {
|
|
109
|
-
health.issues.push(`Read test failed: ${e.message}`);
|
|
110
|
-
if (e.message.includes('rate limit')) {
|
|
111
|
-
health.rateLimits.mentions = 'exceeded';
|
|
112
|
-
}
|
|
113
|
-
}
|
|
114
|
-
|
|
115
|
-
// Note: We don't test write capability to avoid spam
|
|
116
|
-
health.canWrite = true; // Assume true if connected
|
|
117
|
-
|
|
118
|
-
} catch (e) {
|
|
119
|
-
health.issues.push(`Connection failed: ${e.message}`);
|
|
120
|
-
|
|
121
|
-
if (e.message.includes('rate limit')) {
|
|
122
|
-
health.suggestions.push('Wait for rate limit to reset (15 minutes)');
|
|
123
|
-
} else if (e.message.includes('unauthorized')) {
|
|
124
|
-
health.suggestions.push('Check API credentials and permissions');
|
|
125
|
-
} else {
|
|
126
|
-
health.suggestions.push('Verify API credentials and X API access level');
|
|
127
|
-
}
|
|
128
|
-
}
|
|
129
|
-
|
|
130
|
-
return health;
|
|
131
|
-
}
|
|
132
|
-
|
|
133
|
-
async function checkTelegramHealth() {
|
|
134
|
-
const health = {
|
|
135
|
-
platform: 'telegram',
|
|
136
|
-
configured: telegram.isConfigured(),
|
|
137
|
-
connected: false,
|
|
138
|
-
canRead: true, // Telegram bot always can read if configured
|
|
139
|
-
canWrite: false,
|
|
140
|
-
webhookConfigured: false,
|
|
141
|
-
issues: [],
|
|
142
|
-
suggestions: [],
|
|
143
|
-
lastCheck: new Date().toISOString()
|
|
144
|
-
};
|
|
145
|
-
|
|
146
|
-
if (!health.configured) {
|
|
147
|
-
health.issues.push('Bot token not configured');
|
|
148
|
-
health.suggestions.push('Run: vibe telegram-bot --action setup');
|
|
149
|
-
return health;
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
try {
|
|
153
|
-
const botInfo = await telegram.getBotInfo();
|
|
154
|
-
health.connected = true;
|
|
155
|
-
health.canWrite = true;
|
|
156
|
-
health.username = botInfo.username;
|
|
157
|
-
health.botName = botInfo.first_name;
|
|
158
|
-
health.botId = botInfo.id;
|
|
159
|
-
health.canJoinGroups = botInfo.can_join_groups;
|
|
160
|
-
|
|
161
|
-
// Check if webhook is set up (optional for polling mode)
|
|
162
|
-
try {
|
|
163
|
-
// This would require a separate API call to getWebhookInfo
|
|
164
|
-
// For now, we'll assume webhook setup is external
|
|
165
|
-
health.webhookConfigured = false;
|
|
166
|
-
health.suggestions.push('Consider setting up webhook for real-time updates');
|
|
167
|
-
} catch (e) {
|
|
168
|
-
// Non-critical
|
|
169
|
-
}
|
|
170
|
-
|
|
171
|
-
} catch (e) {
|
|
172
|
-
health.issues.push(`Bot connection failed: ${e.message}`);
|
|
173
|
-
|
|
174
|
-
if (e.message.includes('token')) {
|
|
175
|
-
health.suggestions.push('Check bot token in config.json');
|
|
176
|
-
} else {
|
|
177
|
-
health.suggestions.push('Verify bot is active and token is valid');
|
|
178
|
-
}
|
|
179
|
-
}
|
|
180
|
-
|
|
181
|
-
return health;
|
|
182
|
-
}
|
|
183
|
-
|
|
184
|
-
async function checkDiscordHealth() {
|
|
185
|
-
const health = {
|
|
186
|
-
platform: 'discord',
|
|
187
|
-
configured: discord.isConfigured() || discordBot.isConfigured(),
|
|
188
|
-
connected: false,
|
|
189
|
-
canRead: false,
|
|
190
|
-
canWrite: false,
|
|
191
|
-
webhookOnly: false,
|
|
192
|
-
botConfigured: discordBot.isConfigured(),
|
|
193
|
-
issues: [],
|
|
194
|
-
suggestions: [],
|
|
195
|
-
lastCheck: new Date().toISOString()
|
|
196
|
-
};
|
|
197
|
-
|
|
198
|
-
// Check webhook first (simpler)
|
|
199
|
-
if (discord.isConfigured()) {
|
|
200
|
-
try {
|
|
201
|
-
const testSent = await discord.post('🔍 Health check - ignore this test');
|
|
202
|
-
if (testSent) {
|
|
203
|
-
health.canWrite = true;
|
|
204
|
-
health.webhookOnly = true;
|
|
205
|
-
health.connected = true;
|
|
206
|
-
} else {
|
|
207
|
-
health.issues.push('Webhook test failed');
|
|
208
|
-
}
|
|
209
|
-
} catch (e) {
|
|
210
|
-
health.issues.push(`Webhook error: ${e.message}`);
|
|
211
|
-
health.suggestions.push('Check DISCORD_WEBHOOK_URL in config.json');
|
|
212
|
-
}
|
|
213
|
-
}
|
|
214
|
-
|
|
215
|
-
// Check bot (if configured)
|
|
216
|
-
if (discordBot.isConfigured()) {
|
|
217
|
-
try {
|
|
218
|
-
const botInfo = await discordBot.getBotInfo();
|
|
219
|
-
health.connected = true;
|
|
220
|
-
health.canRead = true;
|
|
221
|
-
health.canWrite = true;
|
|
222
|
-
health.botUsername = botInfo.username;
|
|
223
|
-
health.botId = botInfo.id;
|
|
224
|
-
|
|
225
|
-
if (health.webhookOnly) {
|
|
226
|
-
health.suggestions.push('Both webhook and bot configured - consider using bot only');
|
|
227
|
-
}
|
|
228
|
-
} catch (e) {
|
|
229
|
-
health.issues.push(`Bot connection failed: ${e.message}`);
|
|
230
|
-
health.suggestions.push('Check DISCORD_BOT_TOKEN and bot permissions');
|
|
231
|
-
}
|
|
232
|
-
}
|
|
233
|
-
|
|
234
|
-
if (!health.configured) {
|
|
235
|
-
health.issues.push('Neither webhook nor bot configured');
|
|
236
|
-
health.suggestions.push('Add DISCORD_WEBHOOK_URL for one-way posting');
|
|
237
|
-
health.suggestions.push('Or add DISCORD_BOT_TOKEN for full bot features');
|
|
238
|
-
}
|
|
239
|
-
|
|
240
|
-
return health;
|
|
241
|
-
}
|
|
242
|
-
|
|
243
|
-
async function checkFarcasterHealth() {
|
|
244
|
-
const health = {
|
|
245
|
-
platform: 'farcaster',
|
|
246
|
-
configured: farcaster.isConfigured(),
|
|
247
|
-
connected: false,
|
|
248
|
-
canRead: false,
|
|
249
|
-
canWrite: false,
|
|
250
|
-
issues: [],
|
|
251
|
-
suggestions: [],
|
|
252
|
-
lastCheck: new Date().toISOString()
|
|
253
|
-
};
|
|
254
|
-
|
|
255
|
-
if (!health.configured) {
|
|
256
|
-
health.issues.push('Farcaster credentials not configured');
|
|
257
|
-
health.suggestions.push('Need: NEYNAR_API_KEY, FARCASTER_SIGNER_UUID, FARCASTER_FID');
|
|
258
|
-
health.suggestions.push('Get API key from https://neynar.com');
|
|
259
|
-
health.suggestions.push('Create signer via Neynar developer tools');
|
|
260
|
-
return health;
|
|
261
|
-
}
|
|
262
|
-
|
|
263
|
-
try {
|
|
264
|
-
const userInfo = await farcaster.getUser();
|
|
265
|
-
const user = userInfo.users[0];
|
|
266
|
-
|
|
267
|
-
health.connected = true;
|
|
268
|
-
health.canRead = true;
|
|
269
|
-
health.canWrite = true; // Assume true if we can read user info
|
|
270
|
-
health.username = user.username;
|
|
271
|
-
health.displayName = user.display_name;
|
|
272
|
-
health.fid = user.fid;
|
|
273
|
-
health.followers = user.follower_count;
|
|
274
|
-
|
|
275
|
-
// Test read functionality
|
|
276
|
-
try {
|
|
277
|
-
await farcaster.getFeed(null, 5);
|
|
278
|
-
health.canRead = true;
|
|
279
|
-
} catch (e) {
|
|
280
|
-
health.issues.push(`Feed read failed: ${e.message}`);
|
|
281
|
-
health.canRead = false;
|
|
282
|
-
}
|
|
283
|
-
|
|
284
|
-
} catch (e) {
|
|
285
|
-
health.issues.push(`Connection failed: ${e.message}`);
|
|
286
|
-
|
|
287
|
-
if (e.message.includes('API key')) {
|
|
288
|
-
health.suggestions.push('Check NEYNAR_API_KEY in config.json');
|
|
289
|
-
} else if (e.message.includes('signer')) {
|
|
290
|
-
health.suggestions.push('Verify FARCASTER_SIGNER_UUID is valid and active');
|
|
291
|
-
} else if (e.message.includes('FID')) {
|
|
292
|
-
health.suggestions.push('Check FARCASTER_FID matches your account');
|
|
293
|
-
} else {
|
|
294
|
-
health.suggestions.push('Verify all Farcaster credentials in config.json');
|
|
295
|
-
}
|
|
296
|
-
}
|
|
297
|
-
|
|
298
|
-
return health;
|
|
299
|
-
}
|
|
300
|
-
|
|
301
|
-
function formatHealthReport(results, fix, details) {
|
|
302
|
-
let display = header('Bridge Health Report');
|
|
303
|
-
display += `\n\n_Checked at ${new Date().toLocaleString()}_\n`;
|
|
304
|
-
display += divider();
|
|
305
|
-
display += '\n';
|
|
306
|
-
|
|
307
|
-
const allPlatforms = Object.keys(results);
|
|
308
|
-
const healthyCount = allPlatforms.filter(p => results[p].connected).length;
|
|
309
|
-
const configuredCount = allPlatforms.filter(p => results[p].configured).length;
|
|
310
|
-
|
|
311
|
-
display += `**Summary:** ${healthyCount}/${allPlatforms.length} connected, ${configuredCount} configured\n\n`;
|
|
312
|
-
|
|
313
|
-
// Individual platform reports
|
|
314
|
-
for (const [platformName, health] of Object.entries(results)) {
|
|
315
|
-
display += formatPlatformHealth(health, details);
|
|
316
|
-
display += '\n';
|
|
317
|
-
}
|
|
318
|
-
|
|
319
|
-
// Overall recommendations
|
|
320
|
-
display += divider();
|
|
321
|
-
display += '**Quick Actions:**\n';
|
|
322
|
-
|
|
323
|
-
const unconfigured = allPlatforms.filter(p => !results[p].configured);
|
|
324
|
-
if (unconfigured.length > 0) {
|
|
325
|
-
display += `• Configure: ${unconfigured.join(', ')}\n`;
|
|
326
|
-
}
|
|
327
|
-
|
|
328
|
-
const hasIssues = allPlatforms.filter(p => results[p].issues.length > 0);
|
|
329
|
-
if (hasIssues.length > 0) {
|
|
330
|
-
display += `• Fix issues: ${hasIssues.join(', ')}\n`;
|
|
331
|
-
}
|
|
332
|
-
|
|
333
|
-
display += `• Test posting: \`vibe social-post --dry_run --content "test" --channels ["${allPlatforms[0]}"]\`\n`;
|
|
334
|
-
display += `• View unified inbox: \`vibe social-inbox --refresh\``;
|
|
335
|
-
|
|
336
|
-
if (fix) {
|
|
337
|
-
display += '\n\n' + divider();
|
|
338
|
-
display += '**Auto-fix Results:**\n';
|
|
339
|
-
display += '_Auto-fix not yet implemented. Check suggestions above._';
|
|
340
|
-
}
|
|
341
|
-
|
|
342
|
-
return { display };
|
|
343
|
-
}
|
|
344
|
-
|
|
345
|
-
function formatPlatformHealth(health, showDetails) {
|
|
346
|
-
const statusIcon = health.connected ? '✅' : (health.configured ? '⚠️' : '❌');
|
|
347
|
-
const platformName = health.platform.toUpperCase();
|
|
348
|
-
|
|
349
|
-
let result = `${statusIcon} **${platformName}**`;
|
|
350
|
-
|
|
351
|
-
if (health.username) {
|
|
352
|
-
result += ` (@${health.username})`;
|
|
353
|
-
}
|
|
354
|
-
|
|
355
|
-
result += '\n';
|
|
356
|
-
|
|
357
|
-
// Capabilities
|
|
358
|
-
const caps = [];
|
|
359
|
-
if (health.canRead) caps.push('read');
|
|
360
|
-
if (health.canWrite) caps.push('write');
|
|
361
|
-
if (health.webhookOnly) caps.push('webhook-only');
|
|
362
|
-
if (health.botConfigured) caps.push('bot');
|
|
363
|
-
if (caps.length > 0) {
|
|
364
|
-
result += ` ${caps.join(' • ')}\n`;
|
|
365
|
-
}
|
|
366
|
-
|
|
367
|
-
// Issues
|
|
368
|
-
if (health.issues.length > 0) {
|
|
369
|
-
for (const issue of health.issues) {
|
|
370
|
-
result += ` ⚠️ ${issue}\n`;
|
|
371
|
-
}
|
|
372
|
-
}
|
|
373
|
-
|
|
374
|
-
// Show details if requested
|
|
375
|
-
if (showDetails) {
|
|
376
|
-
if (health.rateLimits && Object.keys(health.rateLimits).length > 0) {
|
|
377
|
-
result += ` Rate limits: ${JSON.stringify(health.rateLimits)}\n`;
|
|
378
|
-
}
|
|
379
|
-
|
|
380
|
-
if (health.followers !== undefined) {
|
|
381
|
-
result += ` Followers: ${health.followers}\n`;
|
|
382
|
-
}
|
|
383
|
-
|
|
384
|
-
if (health.lastCheck) {
|
|
385
|
-
result += ` Last checked: ${new Date(health.lastCheck).toLocaleTimeString()}\n`;
|
|
386
|
-
}
|
|
387
|
-
}
|
|
388
|
-
|
|
389
|
-
// Suggestions (first 2 to keep concise)
|
|
390
|
-
const suggestions = health.suggestions.slice(0, 2);
|
|
391
|
-
if (suggestions.length > 0) {
|
|
392
|
-
for (const suggestion of suggestions) {
|
|
393
|
-
result += ` 💡 ${suggestion}\n`;
|
|
394
|
-
}
|
|
395
|
-
}
|
|
396
|
-
|
|
397
|
-
return result;
|
|
398
|
-
}
|
|
399
|
-
|
|
400
|
-
module.exports = { definition, handler };
|