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
package/tools/mint.js
DELETED
|
@@ -1,377 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* vibe_mint — Mint a /vibe moment as an NFT
|
|
3
|
-
*
|
|
4
|
-
* Supports multiple chains:
|
|
5
|
-
* - vibe-l2 (default) — VIBE L2 with Shipback (80% gas fees to creator)
|
|
6
|
-
* - base — Base mainnet
|
|
7
|
-
* - base-sepolia — Base testnet
|
|
8
|
-
* - ethereum — Ethereum mainnet
|
|
9
|
-
*
|
|
10
|
-
* Flow:
|
|
11
|
-
* 1. Upload metadata to IPFS (Pinata)
|
|
12
|
-
* 2. Call VibeArtifacts.mintArtifact() with session provenance
|
|
13
|
-
* 3. Return tx hash + token ID + marketplace links
|
|
14
|
-
*/
|
|
15
|
-
|
|
16
|
-
// Lazy-load ethers to avoid startup crash if not installed
|
|
17
|
-
let ethers = null;
|
|
18
|
-
function getEthers() {
|
|
19
|
-
if (!ethers) {
|
|
20
|
-
try {
|
|
21
|
-
ethers = require('ethers');
|
|
22
|
-
} catch (e) {
|
|
23
|
-
throw new Error('ethers package not installed. Run: npm install ethers');
|
|
24
|
-
}
|
|
25
|
-
}
|
|
26
|
-
return ethers;
|
|
27
|
-
}
|
|
28
|
-
|
|
29
|
-
// Polyfill fetch for Node <18
|
|
30
|
-
const fetch = globalThis.fetch || require('node-fetch');
|
|
31
|
-
|
|
32
|
-
const config = require('../config');
|
|
33
|
-
|
|
34
|
-
// Chain configurations
|
|
35
|
-
// Env var format: VIBE_ARTIFACTS_MAINNET / VIBE_ARTIFACTS_TESTNET (matches deploy script)
|
|
36
|
-
const CHAINS = {
|
|
37
|
-
'vibe-l2': {
|
|
38
|
-
name: 'VIBE L2',
|
|
39
|
-
chainId: 84532000, // Mainnet
|
|
40
|
-
rpcUrl: process.env.VIBE_L2_RPC || 'https://rpc.vibe.network',
|
|
41
|
-
contractAddress: process.env.VIBE_ARTIFACTS_MAINNET,
|
|
42
|
-
explorerUrl: 'https://explorer.vibe.network',
|
|
43
|
-
marketplaceUrl: 'https://slashvibe.dev/a',
|
|
44
|
-
shipbackEnabled: true,
|
|
45
|
-
recommended: true
|
|
46
|
-
},
|
|
47
|
-
'vibe-l2-testnet': {
|
|
48
|
-
name: 'VIBE L2 Testnet',
|
|
49
|
-
chainId: 84532001, // Testnet
|
|
50
|
-
rpcUrl: process.env.VIBE_L2_TESTNET_RPC || 'https://rpc-testnet.vibe.network',
|
|
51
|
-
contractAddress: process.env.VIBE_ARTIFACTS_TESTNET,
|
|
52
|
-
explorerUrl: 'https://explorer-testnet.vibe.network',
|
|
53
|
-
marketplaceUrl: 'https://slashvibe.dev/a',
|
|
54
|
-
shipbackEnabled: true
|
|
55
|
-
},
|
|
56
|
-
'base-sepolia': {
|
|
57
|
-
name: 'Base Sepolia',
|
|
58
|
-
chainId: 84532,
|
|
59
|
-
rpcUrl: process.env.BASE_SEPOLIA_RPC_URL || 'https://sepolia.base.org',
|
|
60
|
-
contractAddress: process.env.VIBE_ARTIFACTS_BASE_SEPOLIA,
|
|
61
|
-
explorerUrl: 'https://sepolia.basescan.org',
|
|
62
|
-
marketplaceUrl: 'https://testnets.opensea.io/assets/base-sepolia',
|
|
63
|
-
shipbackEnabled: false
|
|
64
|
-
},
|
|
65
|
-
'base': {
|
|
66
|
-
name: 'Base',
|
|
67
|
-
chainId: 8453,
|
|
68
|
-
rpcUrl: process.env.BASE_RPC_URL || 'https://mainnet.base.org',
|
|
69
|
-
contractAddress: process.env.VIBE_ARTIFACTS_BASE,
|
|
70
|
-
explorerUrl: 'https://basescan.org',
|
|
71
|
-
marketplaceUrl: 'https://opensea.io/assets/base',
|
|
72
|
-
shipbackEnabled: false
|
|
73
|
-
},
|
|
74
|
-
'ethereum': {
|
|
75
|
-
name: 'Ethereum',
|
|
76
|
-
chainId: 1,
|
|
77
|
-
rpcUrl: process.env.ETH_RPC_URL || 'https://eth.llamarpc.com',
|
|
78
|
-
contractAddress: process.env.VIBE_ARTIFACTS_ADDRESS,
|
|
79
|
-
explorerUrl: 'https://etherscan.io',
|
|
80
|
-
marketplaceUrl: 'https://opensea.io/assets/ethereum',
|
|
81
|
-
shipbackEnabled: false
|
|
82
|
-
}
|
|
83
|
-
};
|
|
84
|
-
|
|
85
|
-
// Get default chain (prefer VIBE L2 if configured)
|
|
86
|
-
function getDefaultChain() {
|
|
87
|
-
if (CHAINS['vibe-l2'].contractAddress) return 'vibe-l2';
|
|
88
|
-
if (CHAINS['vibe-l2-testnet'].contractAddress) return 'vibe-l2-testnet';
|
|
89
|
-
if (CHAINS['base-sepolia'].contractAddress) return 'base-sepolia';
|
|
90
|
-
if (CHAINS['base'].contractAddress) return 'base';
|
|
91
|
-
return 'ethereum';
|
|
92
|
-
}
|
|
93
|
-
|
|
94
|
-
// Contract ABI (extended for session provenance)
|
|
95
|
-
const VIBE_ARTIFACTS_ABI = [
|
|
96
|
-
// Legacy simple mint
|
|
97
|
-
'function mint(address to, string memory uri) public returns (uint256)',
|
|
98
|
-
// New mint with provenance
|
|
99
|
-
'function mintArtifact(address to, string uri, string sessionId, string creatorHandle, string artifactType, bytes32 sessionHash, address royaltyRecipient, uint96 royaltyBps) returns (uint256)',
|
|
100
|
-
'function totalSupply() public view returns (uint256)',
|
|
101
|
-
'event ArtifactMinted(uint256 indexed tokenId, address indexed creator, string sessionId, string artifactType, bytes32 sessionHash)'
|
|
102
|
-
];
|
|
103
|
-
|
|
104
|
-
const PINATA_JWT = process.env.PINATA_JWT || null;
|
|
105
|
-
const MINTER_PRIVATE_KEY = process.env.VIBE_MINTER_PRIVATE_KEY || null;
|
|
106
|
-
|
|
107
|
-
const definition = {
|
|
108
|
-
name: 'vibe_mint',
|
|
109
|
-
description: 'Mint a /vibe artifact as an NFT. Default: VIBE L2 (80% Shipback). Also supports Base, Ethereum.',
|
|
110
|
-
inputSchema: {
|
|
111
|
-
type: 'object',
|
|
112
|
-
properties: {
|
|
113
|
-
title: {
|
|
114
|
-
type: 'string',
|
|
115
|
-
description: 'Title of the artifact'
|
|
116
|
-
},
|
|
117
|
-
description: {
|
|
118
|
-
type: 'string',
|
|
119
|
-
description: 'Description of what happened'
|
|
120
|
-
},
|
|
121
|
-
chain: {
|
|
122
|
-
type: 'string',
|
|
123
|
-
description: 'Chain to mint on: vibe-l2, base, base-sepolia, ethereum (default: vibe-l2)'
|
|
124
|
-
},
|
|
125
|
-
session_id: {
|
|
126
|
-
type: 'string',
|
|
127
|
-
description: 'Session ID for provenance (ses_xxx)'
|
|
128
|
-
},
|
|
129
|
-
attributes: {
|
|
130
|
-
type: 'object',
|
|
131
|
-
description: 'Optional attributes',
|
|
132
|
-
additionalProperties: true
|
|
133
|
-
},
|
|
134
|
-
image_url: {
|
|
135
|
-
type: 'string',
|
|
136
|
-
description: 'Optional image URL'
|
|
137
|
-
},
|
|
138
|
-
dry_run: {
|
|
139
|
-
type: 'boolean',
|
|
140
|
-
description: 'Preview without minting'
|
|
141
|
-
}
|
|
142
|
-
},
|
|
143
|
-
required: ['title', 'description']
|
|
144
|
-
}
|
|
145
|
-
};
|
|
146
|
-
|
|
147
|
-
async function uploadToIPFS(metadata) {
|
|
148
|
-
if (!PINATA_JWT) {
|
|
149
|
-
throw new Error('PINATA_JWT not configured');
|
|
150
|
-
}
|
|
151
|
-
|
|
152
|
-
const response = await fetch('https://api.pinata.cloud/pinning/pinJSONToIPFS', {
|
|
153
|
-
method: 'POST',
|
|
154
|
-
headers: {
|
|
155
|
-
'Content-Type': 'application/json',
|
|
156
|
-
'Authorization': `Bearer ${PINATA_JWT}`
|
|
157
|
-
},
|
|
158
|
-
body: JSON.stringify({
|
|
159
|
-
pinataContent: metadata,
|
|
160
|
-
pinataMetadata: {
|
|
161
|
-
name: `vibe-artifact-${Date.now()}`
|
|
162
|
-
}
|
|
163
|
-
})
|
|
164
|
-
});
|
|
165
|
-
|
|
166
|
-
if (!response.ok) {
|
|
167
|
-
throw new Error(`Pinata upload failed: ${response.statusText}`);
|
|
168
|
-
}
|
|
169
|
-
|
|
170
|
-
const result = await response.json();
|
|
171
|
-
return `ipfs://${result.IpfsHash}`;
|
|
172
|
-
}
|
|
173
|
-
|
|
174
|
-
async function mintOnchain(uri, chain, creatorHandle, sessionId) {
|
|
175
|
-
if (!chain.contractAddress) {
|
|
176
|
-
throw new Error(`Contract not configured for ${chain.name}`);
|
|
177
|
-
}
|
|
178
|
-
if (!MINTER_PRIVATE_KEY) {
|
|
179
|
-
throw new Error('VIBE_MINTER_PRIVATE_KEY not configured');
|
|
180
|
-
}
|
|
181
|
-
|
|
182
|
-
const ethers = getEthers();
|
|
183
|
-
const crypto = require('crypto');
|
|
184
|
-
|
|
185
|
-
const provider = new ethers.JsonRpcProvider(chain.rpcUrl);
|
|
186
|
-
const wallet = new ethers.Wallet(MINTER_PRIVATE_KEY, provider);
|
|
187
|
-
const contract = new ethers.Contract(chain.contractAddress, VIBE_ARTIFACTS_ABI, wallet);
|
|
188
|
-
|
|
189
|
-
// Compute session hash for provenance
|
|
190
|
-
const sessionData = JSON.stringify({ sessionId, creatorHandle, timestamp: Date.now() });
|
|
191
|
-
const sessionHash = '0x' + crypto.createHash('sha256').update(sessionData).digest('hex');
|
|
192
|
-
|
|
193
|
-
let tx;
|
|
194
|
-
try {
|
|
195
|
-
// Try new mintArtifact with provenance
|
|
196
|
-
tx = await contract.mintArtifact(
|
|
197
|
-
wallet.address, // to
|
|
198
|
-
uri, // metadata URI
|
|
199
|
-
sessionId, // session ID
|
|
200
|
-
creatorHandle, // creator handle
|
|
201
|
-
'session_artifact', // artifact type
|
|
202
|
-
sessionHash, // session hash
|
|
203
|
-
wallet.address, // royalty recipient
|
|
204
|
-
1000 // 10% royalty (1000 bps)
|
|
205
|
-
);
|
|
206
|
-
} catch (e) {
|
|
207
|
-
// Fallback to legacy mint if new function not available
|
|
208
|
-
console.log('[mint] Falling back to legacy mint:', e.message);
|
|
209
|
-
tx = await contract.mint(wallet.address, uri);
|
|
210
|
-
}
|
|
211
|
-
|
|
212
|
-
const receipt = await tx.wait();
|
|
213
|
-
|
|
214
|
-
// Parse the event to get tokenId
|
|
215
|
-
const event = receipt.logs.find(log => {
|
|
216
|
-
try {
|
|
217
|
-
const parsed = contract.interface.parseLog(log);
|
|
218
|
-
return parsed?.name === 'ArtifactMinted';
|
|
219
|
-
} catch {
|
|
220
|
-
return false;
|
|
221
|
-
}
|
|
222
|
-
});
|
|
223
|
-
|
|
224
|
-
let tokenId = null;
|
|
225
|
-
if (event) {
|
|
226
|
-
const parsed = contract.interface.parseLog(event);
|
|
227
|
-
tokenId = parsed.args.tokenId.toString();
|
|
228
|
-
}
|
|
229
|
-
|
|
230
|
-
return {
|
|
231
|
-
txHash: receipt.hash,
|
|
232
|
-
tokenId,
|
|
233
|
-
contractAddress: chain.contractAddress
|
|
234
|
-
};
|
|
235
|
-
}
|
|
236
|
-
|
|
237
|
-
async function handler(args) {
|
|
238
|
-
const {
|
|
239
|
-
title,
|
|
240
|
-
description,
|
|
241
|
-
chain: chainArg,
|
|
242
|
-
session_id,
|
|
243
|
-
attributes = {},
|
|
244
|
-
image_url,
|
|
245
|
-
dry_run = false
|
|
246
|
-
} = args;
|
|
247
|
-
|
|
248
|
-
// Get handle from config
|
|
249
|
-
const handle = config.isInitialized() ? config.getHandle() : 'anonymous';
|
|
250
|
-
|
|
251
|
-
// Determine chain
|
|
252
|
-
const chainName = chainArg || getDefaultChain();
|
|
253
|
-
const chain = CHAINS[chainName];
|
|
254
|
-
|
|
255
|
-
if (!chain) {
|
|
256
|
-
return {
|
|
257
|
-
display: `Unknown chain: ${chainArg}\n\nAvailable chains:\n` +
|
|
258
|
-
Object.entries(CHAINS).map(([k, v]) =>
|
|
259
|
-
`• ${k} - ${v.name}${v.shipbackEnabled ? ' (80% Shipback)' : ''}${v.recommended ? ' ✨' : ''}`
|
|
260
|
-
).join('\n')
|
|
261
|
-
};
|
|
262
|
-
}
|
|
263
|
-
|
|
264
|
-
// Build metadata with session provenance
|
|
265
|
-
const sessionId = session_id || `ses_${Date.now()}`;
|
|
266
|
-
const metadata = {
|
|
267
|
-
name: title,
|
|
268
|
-
description,
|
|
269
|
-
attributes: [
|
|
270
|
-
{ trait_type: 'Source', value: '/vibe MCP' },
|
|
271
|
-
{ trait_type: 'Minted Via', value: 'Claude Code' },
|
|
272
|
-
{ trait_type: 'Chain', value: chain.name },
|
|
273
|
-
{ trait_type: 'Creator', value: `@${handle}` },
|
|
274
|
-
{ trait_type: 'Session', value: sessionId },
|
|
275
|
-
{ trait_type: 'Date', value: new Date().toISOString().split('T')[0] },
|
|
276
|
-
...Object.entries(attributes).map(([key, value]) => ({
|
|
277
|
-
trait_type: key,
|
|
278
|
-
value: String(value)
|
|
279
|
-
}))
|
|
280
|
-
],
|
|
281
|
-
external_url: `https://slashvibe.dev/a/${sessionId}`,
|
|
282
|
-
// Session provenance
|
|
283
|
-
provenance: {
|
|
284
|
-
sessionId,
|
|
285
|
-
creatorHandle: handle,
|
|
286
|
-
mintedAt: new Date().toISOString(),
|
|
287
|
-
chain: chainName,
|
|
288
|
-
shipbackEnabled: chain.shipbackEnabled
|
|
289
|
-
}
|
|
290
|
-
};
|
|
291
|
-
|
|
292
|
-
if (image_url) {
|
|
293
|
-
metadata.image = image_url;
|
|
294
|
-
}
|
|
295
|
-
|
|
296
|
-
// Dry run - show metadata and chain info
|
|
297
|
-
if (dry_run) {
|
|
298
|
-
let display = `## Dry Run - Metadata Preview\n\n`;
|
|
299
|
-
display += `**Chain:** ${chain.name}${chain.shipbackEnabled ? ' (80% Shipback enabled!)' : ''}\n\n`;
|
|
300
|
-
display += '```json\n' + JSON.stringify(metadata, null, 2) + '\n```\n\n';
|
|
301
|
-
display += `Ready to mint. Run without \`dry_run\` to mint onchain.`;
|
|
302
|
-
return { display };
|
|
303
|
-
}
|
|
304
|
-
|
|
305
|
-
// Check configuration
|
|
306
|
-
if (!chain.contractAddress) {
|
|
307
|
-
let display = `## Configuration Required\n\n`;
|
|
308
|
-
display += `Chain **${chain.name}** is not configured.\n\n`;
|
|
309
|
-
|
|
310
|
-
if (chain.shipbackEnabled) {
|
|
311
|
-
display += `To use VIBE L2 (recommended - 80% Shipback):\n`;
|
|
312
|
-
display += `1. Sign up at https://conduit.xyz\n`;
|
|
313
|
-
display += `2. Deploy contracts: \`node scripts/deploy-vibe-l2.js\`\n`;
|
|
314
|
-
display += `3. Set env var: \`VIBE_ARTIFACTS_TESTNET=0x...\`\n\n`;
|
|
315
|
-
}
|
|
316
|
-
|
|
317
|
-
display += `Available configured chains:\n`;
|
|
318
|
-
Object.entries(CHAINS).forEach(([k, v]) => {
|
|
319
|
-
if (v.contractAddress) {
|
|
320
|
-
display += `• ${k} - ${v.name}\n`;
|
|
321
|
-
}
|
|
322
|
-
});
|
|
323
|
-
|
|
324
|
-
if (!Object.values(CHAINS).some(c => c.contractAddress)) {
|
|
325
|
-
display += `\nNo chains configured. Set environment variables:\n`;
|
|
326
|
-
display += `• VIBE_ARTIFACTS_TESTNET (recommended)\n`;
|
|
327
|
-
display += `• VIBE_ARTIFACTS_BASE_SEPOLIA\n`;
|
|
328
|
-
display += `• VIBE_MINTER_PRIVATE_KEY\n`;
|
|
329
|
-
display += `• PINATA_JWT\n`;
|
|
330
|
-
}
|
|
331
|
-
|
|
332
|
-
return { display };
|
|
333
|
-
}
|
|
334
|
-
|
|
335
|
-
try {
|
|
336
|
-
// 1. Upload to IPFS
|
|
337
|
-
const ipfsUri = await uploadToIPFS(metadata);
|
|
338
|
-
|
|
339
|
-
// 2. Mint onchain
|
|
340
|
-
const { txHash, tokenId } = await mintOnchain(ipfsUri, chain, handle, sessionId);
|
|
341
|
-
|
|
342
|
-
// 3. Build result
|
|
343
|
-
const explorerUrl = `${chain.explorerUrl}/tx/${txHash}`;
|
|
344
|
-
const marketplaceUrl = tokenId
|
|
345
|
-
? `${chain.marketplaceUrl}/${chain.contractAddress}/${tokenId}`
|
|
346
|
-
: null;
|
|
347
|
-
|
|
348
|
-
let display = `## 🎨 VIBE #${tokenId || '?'} Minted on ${chain.name}\n\n`;
|
|
349
|
-
display += `**${title}**\n\n`;
|
|
350
|
-
display += `• IPFS: \`${ipfsUri}\`\n`;
|
|
351
|
-
display += `• Tx: [${txHash.slice(0, 10)}...](${explorerUrl})\n`;
|
|
352
|
-
display += `• Creator: @${handle}\n`;
|
|
353
|
-
display += `• Session: ${sessionId}\n`;
|
|
354
|
-
|
|
355
|
-
if (marketplaceUrl) {
|
|
356
|
-
display += `• View: [Marketplace](${marketplaceUrl})\n`;
|
|
357
|
-
}
|
|
358
|
-
|
|
359
|
-
if (chain.shipbackEnabled) {
|
|
360
|
-
display += `\n💰 **Shipback enabled!** You'll earn 80% of all gas fees from interactions.\n`;
|
|
361
|
-
display += `Check earnings: \`vibe l2 shipback\``;
|
|
362
|
-
}
|
|
363
|
-
|
|
364
|
-
display += `\n\n_Onchain. Forever._`;
|
|
365
|
-
|
|
366
|
-
return { display };
|
|
367
|
-
|
|
368
|
-
} catch (err) {
|
|
369
|
-
return {
|
|
370
|
-
display: `## Mint Failed\n\n${err.message}\n\n` +
|
|
371
|
-
`Chain: ${chain.name}\n` +
|
|
372
|
-
`Check configuration and try again.`
|
|
373
|
-
};
|
|
374
|
-
}
|
|
375
|
-
}
|
|
376
|
-
|
|
377
|
-
module.exports = { definition, handler };
|
package/tools/plan.js
DELETED
|
@@ -1,225 +0,0 @@
|
|
|
1
|
-
/**
|
|
2
|
-
* vibe_plan — Create a plan and request approval
|
|
3
|
-
*
|
|
4
|
-
* Plans are structured proposals that can be sent for async approval.
|
|
5
|
-
* They're artifacts with an approval workflow built in.
|
|
6
|
-
*/
|
|
7
|
-
|
|
8
|
-
const config = require('../config');
|
|
9
|
-
const store = require('../store');
|
|
10
|
-
const { requireInit, normalizeHandle } = require('./_shared');
|
|
11
|
-
|
|
12
|
-
const definition = {
|
|
13
|
-
name: 'vibe_plan',
|
|
14
|
-
description: 'Create a plan and request approval from collaborators. Plans are structured proposals with summary, problem, changes, and impact.',
|
|
15
|
-
inputSchema: {
|
|
16
|
-
type: 'object',
|
|
17
|
-
properties: {
|
|
18
|
-
title: {
|
|
19
|
-
type: 'string',
|
|
20
|
-
description: 'Plan title (e.g., "Refactor auth system", "CLAUDE.md cleanup")'
|
|
21
|
-
},
|
|
22
|
-
summary: {
|
|
23
|
-
type: 'string',
|
|
24
|
-
description: 'One-line summary of what this plan does'
|
|
25
|
-
},
|
|
26
|
-
problem: {
|
|
27
|
-
type: 'string',
|
|
28
|
-
description: 'What problem does this solve? (optional)'
|
|
29
|
-
},
|
|
30
|
-
changes: {
|
|
31
|
-
type: 'array',
|
|
32
|
-
items: { type: 'string' },
|
|
33
|
-
description: 'List of proposed changes'
|
|
34
|
-
},
|
|
35
|
-
impact: {
|
|
36
|
-
type: 'string',
|
|
37
|
-
description: 'What will be affected by this change? (optional)'
|
|
38
|
-
},
|
|
39
|
-
request_approval_from: {
|
|
40
|
-
type: 'array',
|
|
41
|
-
items: { type: 'string' },
|
|
42
|
-
description: 'Handles to request approval from (e.g., ["@brightseth", "@alice"])'
|
|
43
|
-
},
|
|
44
|
-
requires: {
|
|
45
|
-
type: 'string',
|
|
46
|
-
enum: ['any', 'all'],
|
|
47
|
-
description: 'Require approval from any one person (default) or all of them'
|
|
48
|
-
}
|
|
49
|
-
},
|
|
50
|
-
required: ['title', 'summary', 'changes', 'request_approval_from']
|
|
51
|
-
}
|
|
52
|
-
};
|
|
53
|
-
|
|
54
|
-
// Generate artifact ID and slug
|
|
55
|
-
function generateArtifactId() {
|
|
56
|
-
return `plan_${Date.now()}_${Math.random().toString(36).substring(7)}`;
|
|
57
|
-
}
|
|
58
|
-
|
|
59
|
-
function generateSlug(title) {
|
|
60
|
-
return title
|
|
61
|
-
.toLowerCase()
|
|
62
|
-
.replace(/[^a-z0-9]+/g, '-')
|
|
63
|
-
.replace(/^-|-$/g, '')
|
|
64
|
-
.substring(0, 60);
|
|
65
|
-
}
|
|
66
|
-
|
|
67
|
-
async function handler(args) {
|
|
68
|
-
const initCheck = requireInit();
|
|
69
|
-
if (initCheck) return initCheck;
|
|
70
|
-
|
|
71
|
-
const {
|
|
72
|
-
title,
|
|
73
|
-
summary,
|
|
74
|
-
problem,
|
|
75
|
-
changes,
|
|
76
|
-
impact,
|
|
77
|
-
request_approval_from,
|
|
78
|
-
requires = 'any'
|
|
79
|
-
} = args;
|
|
80
|
-
|
|
81
|
-
const creator = config.getHandle();
|
|
82
|
-
|
|
83
|
-
// Validate approvers
|
|
84
|
-
if (!request_approval_from || request_approval_from.length === 0) {
|
|
85
|
-
return {
|
|
86
|
-
display: '❌ Must specify at least one approver in request_approval_from'
|
|
87
|
-
};
|
|
88
|
-
}
|
|
89
|
-
|
|
90
|
-
// Normalize approver handles
|
|
91
|
-
const approvers = request_approval_from.map(h => normalizeHandle(h));
|
|
92
|
-
|
|
93
|
-
// Build content blocks for the plan
|
|
94
|
-
const blocks = [
|
|
95
|
-
{ type: 'summary', data: { text: summary } }
|
|
96
|
-
];
|
|
97
|
-
|
|
98
|
-
if (problem) {
|
|
99
|
-
blocks.push({ type: 'problem', data: { text: problem } });
|
|
100
|
-
}
|
|
101
|
-
|
|
102
|
-
blocks.push({
|
|
103
|
-
type: 'proposed_changes',
|
|
104
|
-
data: { items: changes }
|
|
105
|
-
});
|
|
106
|
-
|
|
107
|
-
if (impact) {
|
|
108
|
-
blocks.push({ type: 'impact', data: { text: impact } });
|
|
109
|
-
}
|
|
110
|
-
|
|
111
|
-
// Build the plan artifact
|
|
112
|
-
const planId = generateArtifactId();
|
|
113
|
-
const slug = generateSlug(title);
|
|
114
|
-
|
|
115
|
-
// Build audience list (creator + all approvers)
|
|
116
|
-
const audience = new Set([creator, ...approvers]);
|
|
117
|
-
|
|
118
|
-
const plan = {
|
|
119
|
-
id: planId,
|
|
120
|
-
slug,
|
|
121
|
-
title,
|
|
122
|
-
template: 'plan',
|
|
123
|
-
content: { blocks },
|
|
124
|
-
|
|
125
|
-
// Social metadata
|
|
126
|
-
created_by: creator,
|
|
127
|
-
created_for: approvers[0], // Primary approver
|
|
128
|
-
thread_id: null,
|
|
129
|
-
|
|
130
|
-
// Privacy - plans are unlisted by default
|
|
131
|
-
visibility: 'unlisted',
|
|
132
|
-
audience: Array.from(audience),
|
|
133
|
-
|
|
134
|
-
// Provenance
|
|
135
|
-
provenance: {
|
|
136
|
-
source_type: 'manual',
|
|
137
|
-
personalized_for: null,
|
|
138
|
-
notes: null
|
|
139
|
-
},
|
|
140
|
-
|
|
141
|
-
// Lifecycle
|
|
142
|
-
created_at: new Date().toISOString(),
|
|
143
|
-
updated_at: new Date().toISOString(),
|
|
144
|
-
expires_at: null,
|
|
145
|
-
|
|
146
|
-
// Evolution
|
|
147
|
-
revision: 1,
|
|
148
|
-
forked_from: null,
|
|
149
|
-
|
|
150
|
-
// Approval workflow
|
|
151
|
-
approval: {
|
|
152
|
-
status: 'pending',
|
|
153
|
-
requested_from: approvers,
|
|
154
|
-
requires,
|
|
155
|
-
responses: [],
|
|
156
|
-
resolved_at: null
|
|
157
|
-
}
|
|
158
|
-
};
|
|
159
|
-
|
|
160
|
-
// Store the plan
|
|
161
|
-
const storeResult = await store.createArtifact(plan);
|
|
162
|
-
|
|
163
|
-
if (!storeResult.success) {
|
|
164
|
-
return {
|
|
165
|
-
display: `❌ Failed to create plan: ${storeResult.error}`
|
|
166
|
-
};
|
|
167
|
-
}
|
|
168
|
-
|
|
169
|
-
const planUrl = `https://slashvibe.dev/a/${slug}`;
|
|
170
|
-
|
|
171
|
-
// Send DM to each approver
|
|
172
|
-
const dmResults = [];
|
|
173
|
-
for (const approver of approvers) {
|
|
174
|
-
try {
|
|
175
|
-
const dmResult = await store.sendMessage(creator, approver,
|
|
176
|
-
`📋 **Plan: ${title}**\n\n` +
|
|
177
|
-
`**Summary:** ${summary}\n\n` +
|
|
178
|
-
(problem ? `**Problem:** ${problem}\n\n` : '') +
|
|
179
|
-
`**Changes:**\n${changes.map(c => `• ${c}`).join('\n')}\n\n` +
|
|
180
|
-
(impact ? `**Impact:** ${impact}\n\n` : '') +
|
|
181
|
-
`---\n` +
|
|
182
|
-
`👍 Approve · 🔄 Request changes · ❌ Reject\n` +
|
|
183
|
-
`Use: \`vibe approve ${slug}\` or \`vibe approve ${slug} --reject\``,
|
|
184
|
-
{
|
|
185
|
-
type: 'plan_card',
|
|
186
|
-
plan_id: planId,
|
|
187
|
-
plan_slug: slug,
|
|
188
|
-
plan_url: planUrl
|
|
189
|
-
}
|
|
190
|
-
);
|
|
191
|
-
dmResults.push({ approver, success: dmResult.success });
|
|
192
|
-
} catch (error) {
|
|
193
|
-
console.error(`[PLAN] Failed to DM ${approver}:`, error);
|
|
194
|
-
dmResults.push({ approver, success: false, error: error.message });
|
|
195
|
-
}
|
|
196
|
-
}
|
|
197
|
-
|
|
198
|
-
// Build response
|
|
199
|
-
const successfulDMs = dmResults.filter(r => r.success).length;
|
|
200
|
-
const approverList = approvers.map(a => `@${a}`).join(', ');
|
|
201
|
-
|
|
202
|
-
let display = `📋 **Plan created: ${title}**\n\n`;
|
|
203
|
-
display += `**Status:** ⏳ Pending approval\n`;
|
|
204
|
-
display += `**Requested from:** ${approverList}\n`;
|
|
205
|
-
display += `**Requires:** ${requires === 'all' ? 'All approvers' : 'Any approver'}\n\n`;
|
|
206
|
-
|
|
207
|
-
if (successfulDMs === approvers.length) {
|
|
208
|
-
display += `✓ Sent to ${successfulDMs} approver${successfulDMs > 1 ? 's' : ''}\n`;
|
|
209
|
-
} else {
|
|
210
|
-
display += `⚠️ Sent to ${successfulDMs}/${approvers.length} approvers\n`;
|
|
211
|
-
}
|
|
212
|
-
|
|
213
|
-
display += `\n**Plan ID:** \`${slug}\``;
|
|
214
|
-
|
|
215
|
-
return {
|
|
216
|
-
display,
|
|
217
|
-
plan_id: planId,
|
|
218
|
-
plan_slug: slug,
|
|
219
|
-
plan_url: planUrl,
|
|
220
|
-
approval_status: 'pending',
|
|
221
|
-
approvers
|
|
222
|
-
};
|
|
223
|
-
}
|
|
224
|
-
|
|
225
|
-
module.exports = { definition, handler };
|