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.
Files changed (58) hide show
  1. package/README.md +195 -0
  2. package/dist/app.d.ts +3 -0
  3. package/dist/app.d.ts.map +1 -0
  4. package/dist/app.js +130 -0
  5. package/dist/app.js.map +1 -0
  6. package/dist/config/env.d.ts +47 -0
  7. package/dist/config/env.d.ts.map +1 -0
  8. package/dist/config/env.js +63 -0
  9. package/dist/config/env.js.map +1 -0
  10. package/dist/config/pricing.d.ts +12 -0
  11. package/dist/config/pricing.d.ts.map +1 -0
  12. package/dist/config/pricing.js +99 -0
  13. package/dist/config/pricing.js.map +1 -0
  14. package/dist/mcp.d.ts +35 -0
  15. package/dist/mcp.d.ts.map +1 -0
  16. package/dist/mcp.js +150 -0
  17. package/dist/mcp.js.map +1 -0
  18. package/dist/middleware/validation.d.ts +6 -0
  19. package/dist/middleware/validation.d.ts.map +1 -0
  20. package/dist/middleware/validation.js +40 -0
  21. package/dist/middleware/validation.js.map +1 -0
  22. package/dist/middleware/x402.d.ts +33 -0
  23. package/dist/middleware/x402.d.ts.map +1 -0
  24. package/dist/middleware/x402.js +294 -0
  25. package/dist/middleware/x402.js.map +1 -0
  26. package/dist/plugins/defi/routes.d.ts +12 -0
  27. package/dist/plugins/defi/routes.d.ts.map +1 -0
  28. package/dist/plugins/defi/routes.js +430 -0
  29. package/dist/plugins/defi/routes.js.map +1 -0
  30. package/dist/plugins/token/routes.d.ts +10 -0
  31. package/dist/plugins/token/routes.d.ts.map +1 -0
  32. package/dist/plugins/token/routes.js +111 -0
  33. package/dist/plugins/token/routes.js.map +1 -0
  34. package/dist/plugins/wallet/routes.d.ts +13 -0
  35. package/dist/plugins/wallet/routes.d.ts.map +1 -0
  36. package/dist/plugins/wallet/routes.js +235 -0
  37. package/dist/plugins/wallet/routes.js.map +1 -0
  38. package/dist/server.d.ts +2 -0
  39. package/dist/server.d.ts.map +1 -0
  40. package/dist/server.js +38 -0
  41. package/dist/server.js.map +1 -0
  42. package/dist/services/cache.d.ts +25 -0
  43. package/dist/services/cache.d.ts.map +1 -0
  44. package/dist/services/cache.js +82 -0
  45. package/dist/services/cache.js.map +1 -0
  46. package/dist/services/helius.d.ts +102 -0
  47. package/dist/services/helius.d.ts.map +1 -0
  48. package/dist/services/helius.js +456 -0
  49. package/dist/services/helius.js.map +1 -0
  50. package/dist/utils/logger.d.ts +3 -0
  51. package/dist/utils/logger.d.ts.map +1 -0
  52. package/dist/utils/logger.js +15 -0
  53. package/dist/utils/logger.js.map +1 -0
  54. package/dist/utils/retry.d.ts +9 -0
  55. package/dist/utils/retry.d.ts.map +1 -0
  56. package/dist/utils/retry.js +45 -0
  57. package/dist/utils/retry.js.map +1 -0
  58. 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,3 @@
1
+ import type { FastifyInstance } from 'fastify';
2
+ export declare function createApp(): Promise<FastifyInstance>;
3
+ //# sourceMappingURL=app.d.ts.map
@@ -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
@@ -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"}