obol-mcp 2.0.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/README.md +195 -0
- package/dist/app.d.ts +3 -0
- package/dist/app.d.ts.map +1 -0
- package/dist/app.js +130 -0
- package/dist/app.js.map +1 -0
- package/dist/config/env.d.ts +47 -0
- package/dist/config/env.d.ts.map +1 -0
- package/dist/config/env.js +63 -0
- package/dist/config/env.js.map +1 -0
- package/dist/config/pricing.d.ts +12 -0
- package/dist/config/pricing.d.ts.map +1 -0
- package/dist/config/pricing.js +99 -0
- package/dist/config/pricing.js.map +1 -0
- package/dist/mcp.d.ts +35 -0
- package/dist/mcp.d.ts.map +1 -0
- package/dist/mcp.js +150 -0
- package/dist/mcp.js.map +1 -0
- package/dist/middleware/validation.d.ts +6 -0
- package/dist/middleware/validation.d.ts.map +1 -0
- package/dist/middleware/validation.js +40 -0
- package/dist/middleware/validation.js.map +1 -0
- package/dist/middleware/x402.d.ts +33 -0
- package/dist/middleware/x402.d.ts.map +1 -0
- package/dist/middleware/x402.js +294 -0
- package/dist/middleware/x402.js.map +1 -0
- package/dist/plugins/defi/routes.d.ts +12 -0
- package/dist/plugins/defi/routes.d.ts.map +1 -0
- package/dist/plugins/defi/routes.js +430 -0
- package/dist/plugins/defi/routes.js.map +1 -0
- package/dist/plugins/token/routes.d.ts +10 -0
- package/dist/plugins/token/routes.d.ts.map +1 -0
- package/dist/plugins/token/routes.js +111 -0
- package/dist/plugins/token/routes.js.map +1 -0
- package/dist/plugins/wallet/routes.d.ts +13 -0
- package/dist/plugins/wallet/routes.d.ts.map +1 -0
- package/dist/plugins/wallet/routes.js +235 -0
- package/dist/plugins/wallet/routes.js.map +1 -0
- package/dist/server.d.ts +2 -0
- package/dist/server.d.ts.map +1 -0
- package/dist/server.js +38 -0
- package/dist/server.js.map +1 -0
- package/dist/services/cache.d.ts +25 -0
- package/dist/services/cache.d.ts.map +1 -0
- package/dist/services/cache.js +82 -0
- package/dist/services/cache.js.map +1 -0
- package/dist/services/helius.d.ts +102 -0
- package/dist/services/helius.d.ts.map +1 -0
- package/dist/services/helius.js +456 -0
- package/dist/services/helius.js.map +1 -0
- package/dist/utils/logger.d.ts +3 -0
- package/dist/utils/logger.d.ts.map +1 -0
- package/dist/utils/logger.js +15 -0
- package/dist/utils/logger.js.map +1 -0
- package/dist/utils/retry.d.ts +9 -0
- package/dist/utils/retry.d.ts.map +1 -0
- package/dist/utils/retry.js +45 -0
- package/dist/utils/retry.js.map +1 -0
- package/package.json +81 -0
package/README.md
ADDED
|
@@ -0,0 +1,195 @@
|
|
|
1
|
+
# obol
|
|
2
|
+
|
|
3
|
+
Pay the ferryman. Solana agent gateway via x402.
|
|
4
|
+
|
|
5
|
+
Obol is a pay-per-use Solana API for AI agents. No API keys, no subscriptions — agents pay per request in USDC using the [x402 payment protocol](https://x402.org). Solana's sub-cent transaction costs make micropayments viable for the first time.
|
|
6
|
+
|
|
7
|
+
Named after the coin placed on the tongue of the dead to pay Charon for passage across the River Styx. The original micropayment.
|
|
8
|
+
|
|
9
|
+
## How it works
|
|
10
|
+
|
|
11
|
+
1. Agent requests data from a paid endpoint
|
|
12
|
+
2. Obol returns HTTP 402 with a payment requirement (amount, recipient, network)
|
|
13
|
+
3. Agent sends USDC on Solana matching the requirement
|
|
14
|
+
4. Agent retries with transaction proof in the `X-PAYMENT` header
|
|
15
|
+
5. Obol verifies on-chain and returns the data
|
|
16
|
+
|
|
17
|
+
No accounts. No tokens. No onboarding. Just pay and go.
|
|
18
|
+
|
|
19
|
+
## Endpoints
|
|
20
|
+
|
|
21
|
+
### Wallet Analytics
|
|
22
|
+
| Endpoint | Price | Description |
|
|
23
|
+
|----------|-------|-------------|
|
|
24
|
+
| `GET /api/v1/wallet/:addr/overview` | $0.01 | SOL balance, token count, total value |
|
|
25
|
+
| `GET /api/v1/wallet/:addr/portfolio` | $0.05 | Full holdings with prices, NFTs, breakdown |
|
|
26
|
+
| `GET /api/v1/wallet/:addr/activity` | $0.05 | Transaction history with categorization |
|
|
27
|
+
| `GET /api/v1/wallet/:addr/risk` | $0.10 | Multi-factor risk assessment |
|
|
28
|
+
| `GET /api/v1/wallet/:addr/pnl` | $0.15 | Token flow analysis, current values, P&L |
|
|
29
|
+
|
|
30
|
+
### Token Data
|
|
31
|
+
| Endpoint | Price | Description |
|
|
32
|
+
|----------|-------|-------------|
|
|
33
|
+
| `GET /api/v1/token/:mint/price` | $0.005 | Real-time price via Jupiter |
|
|
34
|
+
| `GET /api/v1/token/:mint/metadata` | $0.01 | Name, symbol, supply, decimals |
|
|
35
|
+
|
|
36
|
+
### DeFi
|
|
37
|
+
| Endpoint | Price | Description |
|
|
38
|
+
|----------|-------|-------------|
|
|
39
|
+
| `GET /api/v1/defi/swap/quote` | $0.005 | Jupiter swap quote with route planning |
|
|
40
|
+
| `POST /api/v1/defi/swap/execute` | $0.25 | Jupiter swap transaction builder |
|
|
41
|
+
| `GET /api/v1/defi/positions/:addr` | $0.10 | DeFi positions — LSTs, LPs, lending |
|
|
42
|
+
| `GET /api/v1/defi/lst/yields` | $0.02 | LST yield comparison across Solana |
|
|
43
|
+
|
|
44
|
+
### Free
|
|
45
|
+
| Endpoint | Description |
|
|
46
|
+
|----------|-------------|
|
|
47
|
+
| `GET /` | API info and pricing |
|
|
48
|
+
| `GET /health` | Service status |
|
|
49
|
+
| `POST /api/v1/rpc` | Proxied Helius RPC (allowlisted methods) |
|
|
50
|
+
|
|
51
|
+
## Agent Example
|
|
52
|
+
|
|
53
|
+
See `examples/agent-client.ts` for a full reference implementation. The key flow:
|
|
54
|
+
|
|
55
|
+
```typescript
|
|
56
|
+
import { ObolAgent } from './examples/agent-client';
|
|
57
|
+
|
|
58
|
+
const agent = new ObolAgent();
|
|
59
|
+
await agent.discover(); // fetch pricing
|
|
60
|
+
agent.loadWallet(process.env.AGENT_PRIVATE_KEY);
|
|
61
|
+
|
|
62
|
+
// Auto-discovers price, pays, and returns data
|
|
63
|
+
const price = await agent.fetch('/api/v1/token/USDC_MINT/price');
|
|
64
|
+
```
|
|
65
|
+
|
|
66
|
+
Run in discovery-only mode (no wallet needed):
|
|
67
|
+
```bash
|
|
68
|
+
npx tsx examples/agent-client.ts
|
|
69
|
+
```
|
|
70
|
+
|
|
71
|
+
## MCP Server
|
|
72
|
+
|
|
73
|
+
Obol ships as an MCP server — any AI agent that supports the [Model Context Protocol](https://modelcontextprotocol.io) can discover and call Obol's tools natively.
|
|
74
|
+
|
|
75
|
+
### Install from npm
|
|
76
|
+
|
|
77
|
+
```bash
|
|
78
|
+
npm install -g obol-mcp
|
|
79
|
+
```
|
|
80
|
+
|
|
81
|
+
### Claude Desktop / Claude Code
|
|
82
|
+
|
|
83
|
+
Add to your MCP config:
|
|
84
|
+
|
|
85
|
+
```json
|
|
86
|
+
{
|
|
87
|
+
"mcpServers": {
|
|
88
|
+
"obol": {
|
|
89
|
+
"command": "npx",
|
|
90
|
+
"args": ["-y", "obol-mcp"],
|
|
91
|
+
"env": {
|
|
92
|
+
"OBOL_URL": "https://obol-production.up.railway.app"
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
}
|
|
96
|
+
}
|
|
97
|
+
```
|
|
98
|
+
|
|
99
|
+
Or if developing locally:
|
|
100
|
+
|
|
101
|
+
```json
|
|
102
|
+
{
|
|
103
|
+
"mcpServers": {
|
|
104
|
+
"obol": {
|
|
105
|
+
"command": "npx",
|
|
106
|
+
"args": ["tsx", "/path/to/obol/src/mcp.ts"],
|
|
107
|
+
"env": {
|
|
108
|
+
"OBOL_URL": "http://localhost:3000"
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
}
|
|
112
|
+
}
|
|
113
|
+
```
|
|
114
|
+
|
|
115
|
+
### Available MCP Tools
|
|
116
|
+
|
|
117
|
+
| Tool | Description |
|
|
118
|
+
|------|-------------|
|
|
119
|
+
| `obol_wallet_overview` | SOL balance, token count, total value |
|
|
120
|
+
| `obol_wallet_portfolio` | Full holdings with prices and breakdown |
|
|
121
|
+
| `obol_wallet_activity` | Transaction history with categorization |
|
|
122
|
+
| `obol_wallet_risk` | Multi-factor risk assessment |
|
|
123
|
+
| `obol_wallet_pnl` | Token flow analysis and P&L |
|
|
124
|
+
| `obol_token_price` | Real-time price via Jupiter |
|
|
125
|
+
| `obol_token_metadata` | Name, symbol, supply, decimals |
|
|
126
|
+
| `obol_swap_quote` | Jupiter swap quote with routing |
|
|
127
|
+
| `obol_swap_execute` | Build swap transaction for signing |
|
|
128
|
+
| `obol_defi_positions` | LSTs, LPs, lending positions |
|
|
129
|
+
| `obol_lst_yields` | LST yield comparison |
|
|
130
|
+
| `obol_health` | API health check (free) |
|
|
131
|
+
| `obol_info` | Endpoint pricing and info (free) |
|
|
132
|
+
|
|
133
|
+
### Run locally
|
|
134
|
+
|
|
135
|
+
```bash
|
|
136
|
+
npm run mcp
|
|
137
|
+
```
|
|
138
|
+
|
|
139
|
+
## Stack
|
|
140
|
+
|
|
141
|
+
- **Fastify 5** — high-performance HTTP
|
|
142
|
+
- **@x402/svm** — official x402 SDK for Solana
|
|
143
|
+
- **Helius** — RPC + DAS API
|
|
144
|
+
- **Jupiter** — token prices + swap execution
|
|
145
|
+
- **Upstash Redis** — cache + payment receipts
|
|
146
|
+
- **TypeScript** — full type safety
|
|
147
|
+
- **Zod** — runtime validation
|
|
148
|
+
|
|
149
|
+
## Setup
|
|
150
|
+
|
|
151
|
+
```bash
|
|
152
|
+
git clone https://github.com/halfkey/obol.git
|
|
153
|
+
cd obol
|
|
154
|
+
npm install
|
|
155
|
+
cp .env.example .env
|
|
156
|
+
# Edit .env with your Helius key and merchant wallet address
|
|
157
|
+
npm run dev
|
|
158
|
+
```
|
|
159
|
+
|
|
160
|
+
## Testing
|
|
161
|
+
|
|
162
|
+
```bash
|
|
163
|
+
# Unit + integration tests (46 tests)
|
|
164
|
+
npm test -- --run
|
|
165
|
+
|
|
166
|
+
# Smoke test against live deployment
|
|
167
|
+
npx tsx scripts/smoke-test.ts https://obol-production.up.railway.app
|
|
168
|
+
|
|
169
|
+
# Manual payment test
|
|
170
|
+
npx tsx scripts/test-payment.ts <tx-signature>
|
|
171
|
+
```
|
|
172
|
+
|
|
173
|
+
## Environment
|
|
174
|
+
|
|
175
|
+
See `.env.example`. Key variables:
|
|
176
|
+
|
|
177
|
+
- `PAYMENT_MODE` — `mock` (dev, auto-approve) or `onchain` (production)
|
|
178
|
+
- `PAYMENT_RECIPIENT_ADDRESS` — your Solana wallet that receives USDC
|
|
179
|
+
- `HELIUS_API_KEY` — Helius RPC access
|
|
180
|
+
- `UPSTASH_REDIS_REST_URL` / `TOKEN` — cache layer
|
|
181
|
+
|
|
182
|
+
## Roadmap
|
|
183
|
+
|
|
184
|
+
- [x] 11 paid endpoints (wallet, token, DeFi, LST, P&L)
|
|
185
|
+
- [x] On-chain USDC verification with replay prevention
|
|
186
|
+
- [x] Test suite (46 vitest + 20-point smoke test)
|
|
187
|
+
- [x] Agent client reference implementation
|
|
188
|
+
- [x] GitHub Actions CI
|
|
189
|
+
- [ ] WebSocket subscriptions for wallet monitoring
|
|
190
|
+
- [ ] Dynamic congestion-based pricing
|
|
191
|
+
- [ ] Multi-chain support
|
|
192
|
+
|
|
193
|
+
## License
|
|
194
|
+
|
|
195
|
+
MIT
|
package/dist/app.d.ts
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"app.d.ts","sourceRoot":"","sources":["../src/app.ts"],"names":[],"mappings":"AACA,OAAO,KAAK,EAAE,eAAe,EAAE,MAAM,SAAS,CAAC;AAY/C,wBAAsB,SAAS,IAAI,OAAO,CAAC,eAAe,CAAC,CAuI1D"}
|
package/dist/app.js
ADDED
|
@@ -0,0 +1,130 @@
|
|
|
1
|
+
import Fastify from 'fastify';
|
|
2
|
+
import helmet from '@fastify/helmet';
|
|
3
|
+
import cors from '@fastify/cors';
|
|
4
|
+
import rateLimit from '@fastify/rate-limit';
|
|
5
|
+
import { config } from './config/env.js';
|
|
6
|
+
import { x402PaymentMiddleware } from './middleware/x402.js';
|
|
7
|
+
import { endpointPricing } from './config/pricing.js';
|
|
8
|
+
// Plugins
|
|
9
|
+
import { walletPlugin } from './plugins/wallet/routes.js';
|
|
10
|
+
import { tokenPlugin } from './plugins/token/routes.js';
|
|
11
|
+
import { defiPlugin } from './plugins/defi/routes.js';
|
|
12
|
+
export async function createApp() {
|
|
13
|
+
const app = Fastify({
|
|
14
|
+
logger: config.server.isDevelopment
|
|
15
|
+
? { transport: { target: 'pino-pretty', options: { translateTime: 'HH:MM:ss', ignore: 'pid,hostname' } } }
|
|
16
|
+
: true,
|
|
17
|
+
disableRequestLogging: false,
|
|
18
|
+
requestIdHeader: 'x-request-id',
|
|
19
|
+
trustProxy: true,
|
|
20
|
+
requestTimeout: 30000,
|
|
21
|
+
connectionTimeout: 60000,
|
|
22
|
+
});
|
|
23
|
+
// ── Security ──
|
|
24
|
+
await app.register(helmet, {
|
|
25
|
+
contentSecurityPolicy: { directives: { defaultSrc: ["'none'"] } },
|
|
26
|
+
});
|
|
27
|
+
await app.register(cors, {
|
|
28
|
+
origin: config.server.isDevelopment ? '*'
|
|
29
|
+
: config.security.corsOrigins?.[0] === '*' ? '*'
|
|
30
|
+
: config.security.corsOrigins || false,
|
|
31
|
+
methods: ['GET', 'POST', 'OPTIONS'],
|
|
32
|
+
credentials: true,
|
|
33
|
+
});
|
|
34
|
+
await app.register(rateLimit, {
|
|
35
|
+
global: false,
|
|
36
|
+
max: config.security.rateLimitMaxRequests,
|
|
37
|
+
timeWindow: config.security.rateLimitWindowMs,
|
|
38
|
+
cache: 10000,
|
|
39
|
+
allowList: ['127.0.0.1'],
|
|
40
|
+
keyGenerator: (request) => request.ip || 'unknown',
|
|
41
|
+
});
|
|
42
|
+
// ── x402 Payment Gate (global preHandler) ──
|
|
43
|
+
app.addHook('preHandler', x402PaymentMiddleware);
|
|
44
|
+
// ── Free Endpoints ──
|
|
45
|
+
app.get('/health', async (_request, reply) => {
|
|
46
|
+
return reply.send({
|
|
47
|
+
status: 'ok',
|
|
48
|
+
timestamp: new Date().toISOString(),
|
|
49
|
+
uptime: process.uptime(),
|
|
50
|
+
environment: config.server.isDevelopment ? 'development' : 'production',
|
|
51
|
+
});
|
|
52
|
+
});
|
|
53
|
+
app.get('/', async (_request, reply) => {
|
|
54
|
+
return reply.send({
|
|
55
|
+
name: 'obol',
|
|
56
|
+
version: '2.0.0',
|
|
57
|
+
description: 'Pay the ferryman. Solana agent gateway via x402.',
|
|
58
|
+
paymentMode: config.payment.mode,
|
|
59
|
+
endpoints: Object.fromEntries(Object.entries(endpointPricing).map(([path, info]) => [
|
|
60
|
+
path,
|
|
61
|
+
{ price: `${info.priceUSDC} USDC`, description: info.description },
|
|
62
|
+
])),
|
|
63
|
+
freeEndpoints: ['GET /', 'GET /health', 'POST /api/v1/rpc'],
|
|
64
|
+
protocol: 'x402',
|
|
65
|
+
docs: 'https://github.com/halfkey/obol',
|
|
66
|
+
});
|
|
67
|
+
});
|
|
68
|
+
// ── RPC Proxy (free — for transaction creation) ──
|
|
69
|
+
app.post('/api/v1/rpc', async (request, reply) => {
|
|
70
|
+
const body = request.body;
|
|
71
|
+
const allowed = ['getAccountInfo', 'getLatestBlockhash', 'sendTransaction', 'getSignatureStatuses'];
|
|
72
|
+
if (!body?.method || !allowed.includes(body.method)) {
|
|
73
|
+
return reply.code(403).send({
|
|
74
|
+
error: 'Forbidden',
|
|
75
|
+
message: `RPC method '${body?.method ?? 'undefined'}' not allowed`,
|
|
76
|
+
});
|
|
77
|
+
}
|
|
78
|
+
if (!config.solana.rpcUrl) {
|
|
79
|
+
return reply.code(503).send({ error: 'Service Unavailable', message: 'RPC not configured' });
|
|
80
|
+
}
|
|
81
|
+
try {
|
|
82
|
+
const response = await fetch(config.solana.rpcUrl, {
|
|
83
|
+
method: 'POST',
|
|
84
|
+
headers: { 'Content-Type': 'application/json' },
|
|
85
|
+
body: JSON.stringify({
|
|
86
|
+
jsonrpc: '2.0', id: 1,
|
|
87
|
+
method: body.method,
|
|
88
|
+
params: body.params ?? [],
|
|
89
|
+
}),
|
|
90
|
+
});
|
|
91
|
+
const data = await response.json();
|
|
92
|
+
return reply.send(data);
|
|
93
|
+
}
|
|
94
|
+
catch (error) {
|
|
95
|
+
app.log.error({ error }, 'RPC proxy error');
|
|
96
|
+
return reply.code(500).send({ error: 'Internal Server Error', message: 'RPC proxy failed' });
|
|
97
|
+
}
|
|
98
|
+
});
|
|
99
|
+
// ── Register Endpoint Plugins ──
|
|
100
|
+
await app.register(walletPlugin);
|
|
101
|
+
await app.register(tokenPlugin);
|
|
102
|
+
await app.register(defiPlugin);
|
|
103
|
+
// ── 404 ──
|
|
104
|
+
app.setNotFoundHandler((_request, reply) => {
|
|
105
|
+
return reply.code(404).send({
|
|
106
|
+
error: 'Not Found',
|
|
107
|
+
message: 'Endpoint does not exist',
|
|
108
|
+
});
|
|
109
|
+
});
|
|
110
|
+
// ── Error Handler ──
|
|
111
|
+
app.setErrorHandler((error, _request, reply) => {
|
|
112
|
+
app.log.error(error);
|
|
113
|
+
const statusCode = error.statusCode ?? 500;
|
|
114
|
+
// In production, never leak internal error names or stack traces
|
|
115
|
+
if (config.server.isProduction) {
|
|
116
|
+
return reply.code(statusCode).send({
|
|
117
|
+
error: statusCode >= 500 ? 'Internal Server Error' : 'Error',
|
|
118
|
+
message: statusCode >= 500 ? 'An unexpected error occurred' : error.message,
|
|
119
|
+
statusCode,
|
|
120
|
+
});
|
|
121
|
+
}
|
|
122
|
+
return reply.code(statusCode).send({
|
|
123
|
+
error: error.name || 'InternalServerError',
|
|
124
|
+
message: error.message,
|
|
125
|
+
statusCode,
|
|
126
|
+
});
|
|
127
|
+
});
|
|
128
|
+
return app;
|
|
129
|
+
}
|
|
130
|
+
//# sourceMappingURL=app.js.map
|
package/dist/app.js.map
ADDED
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"app.js","sourceRoot":"","sources":["../src/app.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,47 @@
|
|
|
1
|
+
export declare const env: {
|
|
2
|
+
NODE_ENV: "development" | "test" | "production";
|
|
3
|
+
PORT: number;
|
|
4
|
+
HOST: string;
|
|
5
|
+
SOLANA_NETWORK: "mainnet-beta" | "devnet";
|
|
6
|
+
PAYMENT_MODE: "mock" | "onchain";
|
|
7
|
+
CORS_ORIGINS: string[];
|
|
8
|
+
RATE_LIMIT_WINDOW_MS: number;
|
|
9
|
+
RATE_LIMIT_MAX_REQUESTS: number;
|
|
10
|
+
LOG_LEVEL: "trace" | "debug" | "info" | "warn" | "error" | "fatal";
|
|
11
|
+
HELIUS_API_KEY?: string | undefined;
|
|
12
|
+
PAYMENT_RECIPIENT_ADDRESS?: string | undefined;
|
|
13
|
+
UPSTASH_REDIS_REST_URL?: string | undefined;
|
|
14
|
+
UPSTASH_REDIS_REST_TOKEN?: string | undefined;
|
|
15
|
+
};
|
|
16
|
+
export declare const config: {
|
|
17
|
+
readonly server: {
|
|
18
|
+
readonly port: number;
|
|
19
|
+
readonly host: string;
|
|
20
|
+
readonly isDevelopment: boolean;
|
|
21
|
+
readonly isProduction: boolean;
|
|
22
|
+
readonly isTest: boolean;
|
|
23
|
+
};
|
|
24
|
+
readonly solana: {
|
|
25
|
+
readonly network: "mainnet-beta" | "devnet";
|
|
26
|
+
readonly heliusApiKey: string;
|
|
27
|
+
readonly rpcUrl: string | undefined;
|
|
28
|
+
};
|
|
29
|
+
readonly payment: {
|
|
30
|
+
readonly mode: "mock" | "onchain";
|
|
31
|
+
readonly recipientAddress: string;
|
|
32
|
+
};
|
|
33
|
+
readonly redis: {
|
|
34
|
+
readonly restUrl: string | undefined;
|
|
35
|
+
readonly restToken: string | undefined;
|
|
36
|
+
};
|
|
37
|
+
readonly security: {
|
|
38
|
+
readonly corsOrigins: string[];
|
|
39
|
+
readonly rateLimitWindowMs: number;
|
|
40
|
+
readonly rateLimitMaxRequests: number;
|
|
41
|
+
};
|
|
42
|
+
readonly logging: {
|
|
43
|
+
readonly level: "trace" | "debug" | "info" | "warn" | "error" | "fatal";
|
|
44
|
+
};
|
|
45
|
+
};
|
|
46
|
+
export type Config = typeof config;
|
|
47
|
+
//# sourceMappingURL=env.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"env.d.ts","sourceRoot":"","sources":["../../src/config/env.ts"],"names":[],"mappings":"AAuCA,eAAO,MAAM,GAAG;;;;;;;;;;;;;;CAAc,CAAC;AAE/B,eAAO,MAAM,MAAM;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CA+BT,CAAC;AAEX,MAAM,MAAM,MAAM,GAAG,OAAO,MAAM,CAAC"}
|
|
@@ -0,0 +1,63 @@
|
|
|
1
|
+
import { z } from 'zod';
|
|
2
|
+
import dotenv from 'dotenv';
|
|
3
|
+
dotenv.config();
|
|
4
|
+
const envSchema = z.object({
|
|
5
|
+
// Server
|
|
6
|
+
NODE_ENV: z.enum(['development', 'test', 'production']).default('development'),
|
|
7
|
+
PORT: z.coerce.number().min(1).max(65535).default(3000),
|
|
8
|
+
HOST: z.string().default('0.0.0.0'),
|
|
9
|
+
// Solana
|
|
10
|
+
SOLANA_NETWORK: z.enum(['mainnet-beta', 'devnet']).default('mainnet-beta'),
|
|
11
|
+
HELIUS_API_KEY: z.string().min(1).optional(),
|
|
12
|
+
// x402 Payments
|
|
13
|
+
PAYMENT_MODE: z.enum(['mock', 'onchain']).default('mock'),
|
|
14
|
+
PAYMENT_RECIPIENT_ADDRESS: z.string().min(32).max(64).optional(),
|
|
15
|
+
// Upstash Redis
|
|
16
|
+
UPSTASH_REDIS_REST_URL: z.string().url().optional(),
|
|
17
|
+
UPSTASH_REDIS_REST_TOKEN: z.string().optional(),
|
|
18
|
+
// Security
|
|
19
|
+
CORS_ORIGINS: z.string().optional().transform(val => val ? val.split(',').map(s => s.trim()) : ['*']),
|
|
20
|
+
RATE_LIMIT_WINDOW_MS: z.coerce.number().default(60000),
|
|
21
|
+
RATE_LIMIT_MAX_REQUESTS: z.coerce.number().default(100),
|
|
22
|
+
// Logging
|
|
23
|
+
LOG_LEVEL: z.enum(['trace', 'debug', 'info', 'warn', 'error', 'fatal']).default('info'),
|
|
24
|
+
});
|
|
25
|
+
const parsed = envSchema.safeParse(process.env);
|
|
26
|
+
if (!parsed.success) {
|
|
27
|
+
console.error('Invalid environment variables:', parsed.error.format());
|
|
28
|
+
throw new Error('Invalid environment configuration');
|
|
29
|
+
}
|
|
30
|
+
export const env = parsed.data;
|
|
31
|
+
export const config = {
|
|
32
|
+
server: {
|
|
33
|
+
port: env.PORT,
|
|
34
|
+
host: env.HOST,
|
|
35
|
+
isDevelopment: env.NODE_ENV === 'development',
|
|
36
|
+
isProduction: env.NODE_ENV === 'production',
|
|
37
|
+
isTest: env.NODE_ENV === 'test',
|
|
38
|
+
},
|
|
39
|
+
solana: {
|
|
40
|
+
network: env.SOLANA_NETWORK,
|
|
41
|
+
heliusApiKey: env.HELIUS_API_KEY ?? '',
|
|
42
|
+
rpcUrl: env.HELIUS_API_KEY
|
|
43
|
+
? `https://${env.SOLANA_NETWORK === 'mainnet-beta' ? 'mainnet' : 'devnet'}.helius-rpc.com/?api-key=${env.HELIUS_API_KEY}`
|
|
44
|
+
: undefined,
|
|
45
|
+
},
|
|
46
|
+
payment: {
|
|
47
|
+
mode: env.PAYMENT_MODE,
|
|
48
|
+
recipientAddress: env.PAYMENT_RECIPIENT_ADDRESS ?? '',
|
|
49
|
+
},
|
|
50
|
+
redis: {
|
|
51
|
+
restUrl: env.UPSTASH_REDIS_REST_URL,
|
|
52
|
+
restToken: env.UPSTASH_REDIS_REST_TOKEN,
|
|
53
|
+
},
|
|
54
|
+
security: {
|
|
55
|
+
corsOrigins: env.CORS_ORIGINS,
|
|
56
|
+
rateLimitWindowMs: env.RATE_LIMIT_WINDOW_MS,
|
|
57
|
+
rateLimitMaxRequests: env.RATE_LIMIT_MAX_REQUESTS,
|
|
58
|
+
},
|
|
59
|
+
logging: {
|
|
60
|
+
level: env.LOG_LEVEL,
|
|
61
|
+
},
|
|
62
|
+
};
|
|
63
|
+
//# sourceMappingURL=env.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"env.js","sourceRoot":"","sources":["../../src/config/env.ts"],"names":[],"mappings":"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"}
|
|
@@ -0,0 +1,12 @@
|
|
|
1
|
+
/** Endpoint pricing configuration for x402 payment gates */
|
|
2
|
+
export interface EndpointPrice {
|
|
3
|
+
priceUSDC: number;
|
|
4
|
+
description: string;
|
|
5
|
+
}
|
|
6
|
+
/** Registry of all paid endpoints and their prices */
|
|
7
|
+
export declare const endpointPricing: Record<string, EndpointPrice>;
|
|
8
|
+
/** Get price for an endpoint in USDC. Throws if no pricing exists. */
|
|
9
|
+
export declare function getEndpointPrice(path: string): number;
|
|
10
|
+
/** Check if an endpoint requires payment */
|
|
11
|
+
export declare function requiresPayment(path: string): boolean;
|
|
12
|
+
//# sourceMappingURL=pricing.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pricing.d.ts","sourceRoot":"","sources":["../../src/config/pricing.ts"],"names":[],"mappings":"AAAA,4DAA4D;AAE5D,MAAM,WAAW,aAAa;IAC5B,SAAS,EAAE,MAAM,CAAC;IAClB,WAAW,EAAE,MAAM,CAAC;CACrB;AAED,sDAAsD;AACtD,eAAO,MAAM,eAAe,EAAE,MAAM,CAAC,MAAM,EAAE,aAAa,CAgDzD,CAAC;AAiCF,sEAAsE;AACtE,wBAAgB,gBAAgB,CAAC,IAAI,EAAE,MAAM,GAAG,MAAM,CAOrD;AAED,4CAA4C;AAC5C,wBAAgB,eAAe,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAWrD"}
|
|
@@ -0,0 +1,99 @@
|
|
|
1
|
+
/** Endpoint pricing configuration for x402 payment gates */
|
|
2
|
+
/** Registry of all paid endpoints and their prices */
|
|
3
|
+
export const endpointPricing = {
|
|
4
|
+
// Phase 1: Core
|
|
5
|
+
'/api/v1/wallet/:address/overview': {
|
|
6
|
+
priceUSDC: 0.01,
|
|
7
|
+
description: 'Wallet overview — SOL balance, token count, total value',
|
|
8
|
+
},
|
|
9
|
+
'/api/v1/wallet/:address/portfolio': {
|
|
10
|
+
priceUSDC: 0.05,
|
|
11
|
+
description: 'Full portfolio — tokens with prices, NFTs, breakdown',
|
|
12
|
+
},
|
|
13
|
+
'/api/v1/wallet/:address/activity': {
|
|
14
|
+
priceUSDC: 0.05,
|
|
15
|
+
description: 'Transaction history with categorization',
|
|
16
|
+
},
|
|
17
|
+
'/api/v1/wallet/:address/risk': {
|
|
18
|
+
priceUSDC: 0.10,
|
|
19
|
+
description: 'Multi-factor risk assessment',
|
|
20
|
+
},
|
|
21
|
+
'/api/v1/token/:mint/price': {
|
|
22
|
+
priceUSDC: 0.005,
|
|
23
|
+
description: 'Real-time token price via Jupiter',
|
|
24
|
+
},
|
|
25
|
+
'/api/v1/token/:mint/metadata': {
|
|
26
|
+
priceUSDC: 0.01,
|
|
27
|
+
description: 'Token name, symbol, supply, holder count',
|
|
28
|
+
},
|
|
29
|
+
// Phase 2: DeFi
|
|
30
|
+
'/api/v1/defi/swap/quote': {
|
|
31
|
+
priceUSDC: 0.005,
|
|
32
|
+
description: 'Jupiter swap quote with route planning',
|
|
33
|
+
},
|
|
34
|
+
'/api/v1/defi/swap/execute': {
|
|
35
|
+
priceUSDC: 0.25,
|
|
36
|
+
description: 'Jupiter swap transaction builder — returns serialized tx for signing',
|
|
37
|
+
},
|
|
38
|
+
'/api/v1/defi/positions/:address': {
|
|
39
|
+
priceUSDC: 0.10,
|
|
40
|
+
description: 'DeFi positions — LSTs, LPs, lending, yield',
|
|
41
|
+
},
|
|
42
|
+
'/api/v1/defi/lst/yields': {
|
|
43
|
+
priceUSDC: 0.02,
|
|
44
|
+
description: 'LST yield comparison across Solana liquid staking',
|
|
45
|
+
},
|
|
46
|
+
'/api/v1/wallet/:address/pnl': {
|
|
47
|
+
priceUSDC: 0.15,
|
|
48
|
+
description: 'Wallet P&L — token flows, current values, transaction analysis',
|
|
49
|
+
},
|
|
50
|
+
};
|
|
51
|
+
/** Paths that never require payment */
|
|
52
|
+
const FREE_PATHS = new Set(['/', '/health', '/api/v1/rpc']);
|
|
53
|
+
/** Normalize a real URL path to its route pattern */
|
|
54
|
+
function normalizeToRoute(path) {
|
|
55
|
+
// Strip query string
|
|
56
|
+
const cleanPath = path.split('?')[0] ?? path;
|
|
57
|
+
// /api/v1/wallet/<address>/overview → /api/v1/wallet/:address/overview
|
|
58
|
+
const walletMatch = cleanPath.match(/^\/api\/v1\/wallet\/[^/]+\/(overview|portfolio|activity|risk|pnl)$/);
|
|
59
|
+
if (walletMatch) {
|
|
60
|
+
return `/api/v1/wallet/:address/${walletMatch[1]}`;
|
|
61
|
+
}
|
|
62
|
+
// /api/v1/token/<mint>/price → /api/v1/token/:mint/price
|
|
63
|
+
const tokenMatch = cleanPath.match(/^\/api\/v1\/token\/[^/]+\/(price|metadata)$/);
|
|
64
|
+
if (tokenMatch) {
|
|
65
|
+
return `/api/v1/token/:mint/${tokenMatch[1]}`;
|
|
66
|
+
}
|
|
67
|
+
// /api/v1/defi/positions/<address> → /api/v1/defi/positions/:address
|
|
68
|
+
const defiPositionsMatch = cleanPath.match(/^\/api\/v1\/defi\/positions\/[^/]+$/);
|
|
69
|
+
if (defiPositionsMatch) {
|
|
70
|
+
return '/api/v1/defi/positions/:address';
|
|
71
|
+
}
|
|
72
|
+
// /api/v1/defi/swap/quote and /api/v1/defi/lst/yields are static routes — no normalization needed
|
|
73
|
+
return cleanPath;
|
|
74
|
+
}
|
|
75
|
+
/** Get price for an endpoint in USDC. Throws if no pricing exists. */
|
|
76
|
+
export function getEndpointPrice(path) {
|
|
77
|
+
const route = normalizeToRoute(path);
|
|
78
|
+
const pricing = endpointPricing[route];
|
|
79
|
+
if (!pricing) {
|
|
80
|
+
throw new Error(`No pricing configured for: ${path}`);
|
|
81
|
+
}
|
|
82
|
+
return pricing.priceUSDC;
|
|
83
|
+
}
|
|
84
|
+
/** Check if an endpoint requires payment */
|
|
85
|
+
export function requiresPayment(path) {
|
|
86
|
+
const cleanPath = path.split('?')[0] ?? path;
|
|
87
|
+
if (FREE_PATHS.has(cleanPath))
|
|
88
|
+
return false;
|
|
89
|
+
if (!cleanPath.startsWith('/api/'))
|
|
90
|
+
return false;
|
|
91
|
+
try {
|
|
92
|
+
getEndpointPrice(path);
|
|
93
|
+
return true;
|
|
94
|
+
}
|
|
95
|
+
catch {
|
|
96
|
+
return false;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
//# sourceMappingURL=pricing.js.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"pricing.js","sourceRoot":"","sources":["../../src/config/pricing.ts"],"names":[],"mappings":"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"}
|
package/dist/mcp.d.ts
ADDED
|
@@ -0,0 +1,35 @@
|
|
|
1
|
+
#!/usr/bin/env node
|
|
2
|
+
/**
|
|
3
|
+
* Obol MCP Server
|
|
4
|
+
*
|
|
5
|
+
* Exposes Obol's Solana data endpoints as MCP tools that any
|
|
6
|
+
* AI agent can discover and call. Runs over stdio transport.
|
|
7
|
+
*
|
|
8
|
+
* The MCP server proxies requests to either:
|
|
9
|
+
* - A live Obol instance (default: production)
|
|
10
|
+
* - A local dev instance
|
|
11
|
+
*
|
|
12
|
+
* In MCP mode, the server operates in "proxy" payment mode:
|
|
13
|
+
* - If OBOL_URL points to a mock-mode instance, data is free
|
|
14
|
+
* - If OBOL_URL points to an onchain instance, the MCP server
|
|
15
|
+
* can optionally include a payment signature via AGENT_PRIVATE_KEY
|
|
16
|
+
*
|
|
17
|
+
* Usage:
|
|
18
|
+
* npx tsx src/mcp.ts # stdio transport
|
|
19
|
+
* OBOL_URL=http://localhost:3000 npx tsx src/mcp.ts # local dev
|
|
20
|
+
*
|
|
21
|
+
* Claude Desktop config (~/.claude/claude_desktop_config.json):
|
|
22
|
+
* {
|
|
23
|
+
* "mcpServers": {
|
|
24
|
+
* "obol": {
|
|
25
|
+
* "command": "npx",
|
|
26
|
+
* "args": ["tsx", "/path/to/obol/src/mcp.ts"],
|
|
27
|
+
* "env": {
|
|
28
|
+
* "OBOL_URL": "https://obol-production.up.railway.app"
|
|
29
|
+
* }
|
|
30
|
+
* }
|
|
31
|
+
* }
|
|
32
|
+
* }
|
|
33
|
+
*/
|
|
34
|
+
export {};
|
|
35
|
+
//# sourceMappingURL=mcp.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"mcp.d.ts","sourceRoot":"","sources":["../src/mcp.ts"],"names":[],"mappings":";AACA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;GA+BG"}
|