@sochdb/sochdb 0.4.2 → 0.4.4
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 +356 -14
- package/_bin/aarch64-apple-darwin/libsochdb_storage.dylib +0 -0
- package/_bin/aarch64-apple-darwin/sochdb-bulk +0 -0
- package/_bin/aarch64-apple-darwin/sochdb-grpc-server +0 -0
- package/_bin/aarch64-apple-darwin/sochdb-server +0 -0
- package/_bin/x86_64-pc-windows-msvc/sochdb-bulk.exe +0 -0
- package/_bin/x86_64-pc-windows-msvc/sochdb-grpc-server.exe +0 -0
- package/_bin/x86_64-pc-windows-msvc/sochdb_storage.dll +0 -0
- package/_bin/x86_64-unknown-linux-gnu/libsochdb_storage.so +0 -0
- package/_bin/x86_64-unknown-linux-gnu/sochdb-bulk +0 -0
- package/_bin/x86_64-unknown-linux-gnu/sochdb-grpc-server +0 -0
- package/_bin/x86_64-unknown-linux-gnu/sochdb-server +0 -0
- package/dist/cjs/embedded/database.js +98 -4
- package/dist/cjs/embedded/ffi/bindings.js +46 -8
- package/dist/cjs/index.js +28 -6
- package/dist/cjs/mcp/client.js +115 -0
- package/dist/cjs/mcp/index.js +28 -0
- package/dist/cjs/mcp/server.js +242 -0
- package/dist/cjs/mcp/types.js +32 -0
- package/dist/cjs/namespace.js +147 -19
- package/dist/cjs/policy/index.js +26 -0
- package/dist/cjs/policy/service.js +394 -0
- package/dist/cjs/policy/types.js +8 -0
- package/dist/esm/embedded/database.js +98 -4
- package/dist/esm/embedded/ffi/bindings.js +48 -8
- package/dist/esm/index.js +28 -6
- package/dist/esm/mcp/client.js +116 -0
- package/dist/esm/mcp/index.js +28 -0
- package/dist/esm/mcp/server.js +244 -0
- package/dist/esm/mcp/types.js +34 -0
- package/dist/esm/namespace.js +150 -19
- package/dist/esm/policy/index.js +26 -0
- package/dist/esm/policy/service.js +396 -0
- package/dist/esm/policy/types.js +8 -0
- package/dist/types/embedded/database.d.ts +66 -1
- package/dist/types/embedded/database.d.ts.map +1 -1
- package/dist/types/embedded/ffi/bindings.d.ts +7 -0
- package/dist/types/embedded/ffi/bindings.d.ts.map +1 -1
- package/dist/types/index.d.ts +23 -5
- package/dist/types/index.d.ts.map +1 -1
- package/dist/types/mcp/client.d.ts +69 -0
- package/dist/types/mcp/client.d.ts.map +1 -0
- package/dist/types/mcp/index.d.ts +9 -0
- package/dist/types/mcp/index.d.ts.map +1 -0
- package/dist/types/mcp/server.d.ts +87 -0
- package/dist/types/mcp/server.d.ts.map +1 -0
- package/dist/types/mcp/types.d.ts +124 -0
- package/dist/types/mcp/types.d.ts.map +1 -0
- package/dist/types/namespace.d.ts +13 -0
- package/dist/types/namespace.d.ts.map +1 -1
- package/dist/types/policy/index.d.ts +8 -0
- package/dist/types/policy/index.d.ts.map +1 -0
- package/dist/types/policy/service.d.ts +115 -0
- package/dist/types/policy/service.d.ts.map +1 -0
- package/dist/types/policy/types.d.ts +102 -0
- package/dist/types/policy/types.d.ts.map +1 -0
- package/package.json +2 -2
|
@@ -0,0 +1,28 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* MCP (Model Context Protocol) Module
|
|
4
|
+
*
|
|
5
|
+
* Provides MCP server and client implementations for LLM agent integration.
|
|
6
|
+
*/
|
|
7
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
8
|
+
if (k2 === undefined) k2 = k;
|
|
9
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
10
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
11
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
12
|
+
}
|
|
13
|
+
Object.defineProperty(o, k2, desc);
|
|
14
|
+
}) : (function(o, m, k, k2) {
|
|
15
|
+
if (k2 === undefined) k2 = k;
|
|
16
|
+
o[k2] = m[k];
|
|
17
|
+
}));
|
|
18
|
+
var __exportStar = (this && this.__exportStar) || function(m, exports) {
|
|
19
|
+
for (var p in m) if (p !== "default" && !Object.prototype.hasOwnProperty.call(exports, p)) __createBinding(exports, m, p);
|
|
20
|
+
};
|
|
21
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
22
|
+
exports.McpClient = exports.McpServer = void 0;
|
|
23
|
+
__exportStar(require("./types"), exports);
|
|
24
|
+
var server_1 = require("./server");
|
|
25
|
+
Object.defineProperty(exports, "McpServer", { enumerable: true, get: function () { return server_1.McpServer; } });
|
|
26
|
+
var client_1 = require("./client");
|
|
27
|
+
Object.defineProperty(exports, "McpClient", { enumerable: true, get: function () { return client_1.McpClient; } });
|
|
28
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiaW5kZXguanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbWNwL2luZGV4LnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7OztHQUlHOzs7Ozs7Ozs7Ozs7Ozs7OztBQUVILDBDQUF3QjtBQUN4QixtQ0FBcUM7QUFBNUIsbUdBQUEsU0FBUyxPQUFBO0FBQ2xCLG1DQUFxQztBQUE1QixtR0FBQSxTQUFTLE9BQUEiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIE1DUCAoTW9kZWwgQ29udGV4dCBQcm90b2NvbCkgTW9kdWxlXG4gKiBcbiAqIFByb3ZpZGVzIE1DUCBzZXJ2ZXIgYW5kIGNsaWVudCBpbXBsZW1lbnRhdGlvbnMgZm9yIExMTSBhZ2VudCBpbnRlZ3JhdGlvbi5cbiAqL1xuXG5leHBvcnQgKiBmcm9tICcuL3R5cGVzJztcbmV4cG9ydCB7IE1jcFNlcnZlciB9IGZyb20gJy4vc2VydmVyJztcbmV4cG9ydCB7IE1jcENsaWVudCB9IGZyb20gJy4vY2xpZW50JztcbiJdfQ==
|
|
@@ -0,0 +1,244 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* MCP Server Implementation
|
|
4
|
+
*
|
|
5
|
+
* SochDB MCP Server for exposing database operations to LLM agents.
|
|
6
|
+
*/
|
|
7
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
+
exports.McpServer = void 0;
|
|
9
|
+
/**
|
|
10
|
+
* MCP Server for SochDB
|
|
11
|
+
*
|
|
12
|
+
* Exposes database operations as MCP tools for LLM agents.
|
|
13
|
+
*
|
|
14
|
+
* @example
|
|
15
|
+
* ```typescript
|
|
16
|
+
* import { EmbeddedDatabase, McpServer } from '@sochdb/sochdb';
|
|
17
|
+
*
|
|
18
|
+
* const db = EmbeddedDatabase.open('./mydb');
|
|
19
|
+
* const server = new McpServer(db, {
|
|
20
|
+
* name: 'sochdb-mcp',
|
|
21
|
+
* version: '1.0.0',
|
|
22
|
+
* capabilities: { tools: true, resources: true }
|
|
23
|
+
* });
|
|
24
|
+
*
|
|
25
|
+
* // List available tools
|
|
26
|
+
* const tools = server.listTools();
|
|
27
|
+
*
|
|
28
|
+
* // Execute a tool call
|
|
29
|
+
* const result = await server.callTool({
|
|
30
|
+
* id: 'call_1',
|
|
31
|
+
* name: 'db_get',
|
|
32
|
+
* arguments: { key: 'user:123' }
|
|
33
|
+
* });
|
|
34
|
+
* ```
|
|
35
|
+
*/
|
|
36
|
+
class McpServer {
|
|
37
|
+
db;
|
|
38
|
+
config;
|
|
39
|
+
tools = new Map();
|
|
40
|
+
prompts = new Map();
|
|
41
|
+
customToolHandlers = new Map();
|
|
42
|
+
constructor(db, config) {
|
|
43
|
+
this.db = db;
|
|
44
|
+
this.config = config;
|
|
45
|
+
this.registerBuiltinTools();
|
|
46
|
+
}
|
|
47
|
+
/**
|
|
48
|
+
* Register built-in database tools
|
|
49
|
+
*/
|
|
50
|
+
registerBuiltinTools() {
|
|
51
|
+
// Key-Value operations
|
|
52
|
+
this.registerTool({
|
|
53
|
+
name: 'db_get',
|
|
54
|
+
description: 'Get a value from the database by key',
|
|
55
|
+
inputSchema: {
|
|
56
|
+
type: 'object',
|
|
57
|
+
properties: {
|
|
58
|
+
key: { type: 'string', description: 'The key to retrieve' },
|
|
59
|
+
},
|
|
60
|
+
required: ['key'],
|
|
61
|
+
},
|
|
62
|
+
}, async (args) => {
|
|
63
|
+
const value = await this.db.get(Buffer.from(args.key));
|
|
64
|
+
return value ? value.toString() : null;
|
|
65
|
+
});
|
|
66
|
+
this.registerTool({
|
|
67
|
+
name: 'db_put',
|
|
68
|
+
description: 'Store a value in the database',
|
|
69
|
+
inputSchema: {
|
|
70
|
+
type: 'object',
|
|
71
|
+
properties: {
|
|
72
|
+
key: { type: 'string', description: 'The key to store' },
|
|
73
|
+
value: { type: 'string', description: 'The value to store' },
|
|
74
|
+
},
|
|
75
|
+
required: ['key', 'value'],
|
|
76
|
+
},
|
|
77
|
+
}, async (args) => {
|
|
78
|
+
await this.db.put(Buffer.from(args.key), Buffer.from(args.value));
|
|
79
|
+
return { success: true };
|
|
80
|
+
});
|
|
81
|
+
this.registerTool({
|
|
82
|
+
name: 'db_delete',
|
|
83
|
+
description: 'Delete a key from the database',
|
|
84
|
+
inputSchema: {
|
|
85
|
+
type: 'object',
|
|
86
|
+
properties: {
|
|
87
|
+
key: { type: 'string', description: 'The key to delete' },
|
|
88
|
+
},
|
|
89
|
+
required: ['key'],
|
|
90
|
+
},
|
|
91
|
+
}, async (args) => {
|
|
92
|
+
await this.db.delete(Buffer.from(args.key));
|
|
93
|
+
return { success: true };
|
|
94
|
+
});
|
|
95
|
+
this.registerTool({
|
|
96
|
+
name: 'db_scan',
|
|
97
|
+
description: 'Scan keys with a prefix',
|
|
98
|
+
inputSchema: {
|
|
99
|
+
type: 'object',
|
|
100
|
+
properties: {
|
|
101
|
+
prefix: { type: 'string', description: 'The prefix to scan' },
|
|
102
|
+
limit: { type: 'number', description: 'Maximum results', default: 100 },
|
|
103
|
+
},
|
|
104
|
+
required: ['prefix'],
|
|
105
|
+
},
|
|
106
|
+
}, async (args) => {
|
|
107
|
+
const results = [];
|
|
108
|
+
const limit = args.limit || 100;
|
|
109
|
+
let count = 0;
|
|
110
|
+
for await (const [keyBuffer, valueBuffer] of this.db.scanPrefix(Buffer.from(args.prefix))) {
|
|
111
|
+
if (count >= limit)
|
|
112
|
+
break;
|
|
113
|
+
results.push({
|
|
114
|
+
key: keyBuffer.toString(),
|
|
115
|
+
value: valueBuffer.toString(),
|
|
116
|
+
});
|
|
117
|
+
count++;
|
|
118
|
+
}
|
|
119
|
+
return results;
|
|
120
|
+
});
|
|
121
|
+
this.registerTool({
|
|
122
|
+
name: 'db_stats',
|
|
123
|
+
description: 'Get database statistics',
|
|
124
|
+
inputSchema: {
|
|
125
|
+
type: 'object',
|
|
126
|
+
properties: {},
|
|
127
|
+
},
|
|
128
|
+
}, async () => {
|
|
129
|
+
const stats = await this.db.stats();
|
|
130
|
+
return {
|
|
131
|
+
memtableSizeBytes: stats.memtableSizeBytes.toString(),
|
|
132
|
+
walSizeBytes: stats.walSizeBytes.toString(),
|
|
133
|
+
activeTransactions: stats.activeTransactions,
|
|
134
|
+
};
|
|
135
|
+
});
|
|
136
|
+
}
|
|
137
|
+
/**
|
|
138
|
+
* Register a custom tool
|
|
139
|
+
*/
|
|
140
|
+
registerTool(tool, handler) {
|
|
141
|
+
this.tools.set(tool.name, tool);
|
|
142
|
+
this.customToolHandlers.set(tool.name, handler);
|
|
143
|
+
}
|
|
144
|
+
/**
|
|
145
|
+
* Unregister a tool
|
|
146
|
+
*/
|
|
147
|
+
unregisterTool(name) {
|
|
148
|
+
this.customToolHandlers.delete(name);
|
|
149
|
+
return this.tools.delete(name);
|
|
150
|
+
}
|
|
151
|
+
/**
|
|
152
|
+
* List all available tools
|
|
153
|
+
*/
|
|
154
|
+
listTools() {
|
|
155
|
+
return Array.from(this.tools.values());
|
|
156
|
+
}
|
|
157
|
+
/**
|
|
158
|
+
* Call a tool
|
|
159
|
+
*/
|
|
160
|
+
async callTool(call) {
|
|
161
|
+
const handler = this.customToolHandlers.get(call.name);
|
|
162
|
+
if (!handler) {
|
|
163
|
+
return {
|
|
164
|
+
id: call.id,
|
|
165
|
+
content: null,
|
|
166
|
+
isError: true,
|
|
167
|
+
errorMessage: `Tool not found: ${call.name}`,
|
|
168
|
+
};
|
|
169
|
+
}
|
|
170
|
+
try {
|
|
171
|
+
const content = await handler(call.arguments);
|
|
172
|
+
return {
|
|
173
|
+
id: call.id,
|
|
174
|
+
content,
|
|
175
|
+
isError: false,
|
|
176
|
+
};
|
|
177
|
+
}
|
|
178
|
+
catch (error) {
|
|
179
|
+
return {
|
|
180
|
+
id: call.id,
|
|
181
|
+
content: null,
|
|
182
|
+
isError: true,
|
|
183
|
+
errorMessage: error.message || 'Unknown error',
|
|
184
|
+
};
|
|
185
|
+
}
|
|
186
|
+
}
|
|
187
|
+
/**
|
|
188
|
+
* Register a prompt template
|
|
189
|
+
*/
|
|
190
|
+
registerPrompt(prompt) {
|
|
191
|
+
this.prompts.set(prompt.name, prompt);
|
|
192
|
+
}
|
|
193
|
+
/**
|
|
194
|
+
* List all available prompts
|
|
195
|
+
*/
|
|
196
|
+
listPrompts() {
|
|
197
|
+
return Array.from(this.prompts.values());
|
|
198
|
+
}
|
|
199
|
+
/**
|
|
200
|
+
* Get prompt messages
|
|
201
|
+
*/
|
|
202
|
+
getPrompt(name, args) {
|
|
203
|
+
const prompt = this.prompts.get(name);
|
|
204
|
+
if (!prompt) {
|
|
205
|
+
return null;
|
|
206
|
+
}
|
|
207
|
+
// For now, return empty messages - actual implementation would render the prompt
|
|
208
|
+
return [];
|
|
209
|
+
}
|
|
210
|
+
/**
|
|
211
|
+
* List available resources
|
|
212
|
+
*/
|
|
213
|
+
async listResources() {
|
|
214
|
+
// Return database stats as a resource
|
|
215
|
+
return [{
|
|
216
|
+
uri: 'sochdb://stats',
|
|
217
|
+
name: 'Database Statistics',
|
|
218
|
+
description: 'Current database statistics',
|
|
219
|
+
mimeType: 'application/json',
|
|
220
|
+
}];
|
|
221
|
+
}
|
|
222
|
+
/**
|
|
223
|
+
* Read a resource
|
|
224
|
+
*/
|
|
225
|
+
async readResource(uri) {
|
|
226
|
+
if (uri === 'sochdb://stats') {
|
|
227
|
+
const stats = await this.db.stats();
|
|
228
|
+
return {
|
|
229
|
+
uri,
|
|
230
|
+
mimeType: 'application/json',
|
|
231
|
+
text: JSON.stringify(stats, (_, v) => typeof v === 'bigint' ? v.toString() : v),
|
|
232
|
+
};
|
|
233
|
+
}
|
|
234
|
+
return null;
|
|
235
|
+
}
|
|
236
|
+
/**
|
|
237
|
+
* Get server info
|
|
238
|
+
*/
|
|
239
|
+
getServerInfo() {
|
|
240
|
+
return { ...this.config };
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
exports.McpServer = McpServer;
|
|
244
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"server.js","sourceRoot":"","sources":["../../../src/mcp/server.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * MCP Server Implementation\n * \n * SochDB MCP Server for exposing database operations to LLM agents.\n */\n\nimport { EmbeddedDatabase } from '../embedded';\nimport {\n  McpTool,\n  McpToolCall,\n  McpToolResult,\n  McpResource,\n  McpResourceContent,\n  McpPrompt,\n  McpPromptMessage,\n  McpServerConfig,\n  McpServerCapabilities,\n  McpError,\n  MCP_ERROR_CODES,\n} from './types';\n\n/**\n * MCP Server for SochDB\n * \n * Exposes database operations as MCP tools for LLM agents.\n * \n * @example\n * ```typescript\n * import { EmbeddedDatabase, McpServer } from '@sochdb/sochdb';\n * \n * const db = EmbeddedDatabase.open('./mydb');\n * const server = new McpServer(db, {\n *   name: 'sochdb-mcp',\n *   version: '1.0.0',\n *   capabilities: { tools: true, resources: true }\n * });\n * \n * // List available tools\n * const tools = server.listTools();\n * \n * // Execute a tool call\n * const result = await server.callTool({\n *   id: 'call_1',\n *   name: 'db_get',\n *   arguments: { key: 'user:123' }\n * });\n * ```\n */\nexport class McpServer {\n  private db: EmbeddedDatabase;\n  private config: McpServerConfig;\n  private tools: Map<string, McpTool> = new Map();\n  private prompts: Map<string, McpPrompt> = new Map();\n  private customToolHandlers: Map<string, (args: Record<string, any>) => Promise<any>> = new Map();\n\n  constructor(db: EmbeddedDatabase, config: McpServerConfig) {\n    this.db = db;\n    this.config = config;\n    this.registerBuiltinTools();\n  }\n\n  /**\n   * Register built-in database tools\n   */\n  private registerBuiltinTools(): void {\n    // Key-Value operations\n    this.registerTool({\n      name: 'db_get',\n      description: 'Get a value from the database by key',\n      inputSchema: {\n        type: 'object',\n        properties: {\n          key: { type: 'string', description: 'The key to retrieve' },\n        },\n        required: ['key'],\n      },\n    }, async (args) => {\n      const value = await this.db.get(Buffer.from(args.key));\n      return value ? value.toString() : null;\n    });\n\n    this.registerTool({\n      name: 'db_put',\n      description: 'Store a value in the database',\n      inputSchema: {\n        type: 'object',\n        properties: {\n          key: { type: 'string', description: 'The key to store' },\n          value: { type: 'string', description: 'The value to store' },\n        },\n        required: ['key', 'value'],\n      },\n    }, async (args) => {\n      await this.db.put(Buffer.from(args.key), Buffer.from(args.value));\n      return { success: true };\n    });\n\n    this.registerTool({\n      name: 'db_delete',\n      description: 'Delete a key from the database',\n      inputSchema: {\n        type: 'object',\n        properties: {\n          key: { type: 'string', description: 'The key to delete' },\n        },\n        required: ['key'],\n      },\n    }, async (args) => {\n      await this.db.delete(Buffer.from(args.key));\n      return { success: true };\n    });\n\n    this.registerTool({\n      name: 'db_scan',\n      description: 'Scan keys with a prefix',\n      inputSchema: {\n        type: 'object',\n        properties: {\n          prefix: { type: 'string', description: 'The prefix to scan' },\n          limit: { type: 'number', description: 'Maximum results', default: 100 },\n        },\n        required: ['prefix'],\n      },\n    }, async (args) => {\n      const results: Array<{ key: string; value: string }> = [];\n      const limit = args.limit || 100;\n      let count = 0;\n\n      for await (const [keyBuffer, valueBuffer] of this.db.scanPrefix(Buffer.from(args.prefix))) {\n        if (count >= limit) break;\n        results.push({\n          key: keyBuffer.toString(),\n          value: valueBuffer.toString(),\n        });\n        count++;\n      }\n\n      return results;\n    });\n\n    this.registerTool({\n      name: 'db_stats',\n      description: 'Get database statistics',\n      inputSchema: {\n        type: 'object',\n        properties: {},\n      },\n    }, async () => {\n      const stats = await this.db.stats();\n      return {\n        memtableSizeBytes: stats.memtableSizeBytes.toString(),\n        walSizeBytes: stats.walSizeBytes.toString(),\n        activeTransactions: stats.activeTransactions,\n      };\n    });\n  }\n\n  /**\n   * Register a custom tool\n   */\n  registerTool(\n    tool: McpTool,\n    handler: (args: Record<string, any>) => Promise<any>\n  ): void {\n    this.tools.set(tool.name, tool);\n    this.customToolHandlers.set(tool.name, handler);\n  }\n\n  /**\n   * Unregister a tool\n   */\n  unregisterTool(name: string): boolean {\n    this.customToolHandlers.delete(name);\n    return this.tools.delete(name);\n  }\n\n  /**\n   * List all available tools\n   */\n  listTools(): McpTool[] {\n    return Array.from(this.tools.values());\n  }\n\n  /**\n   * Call a tool\n   */\n  async callTool(call: McpToolCall): Promise<McpToolResult> {\n    const handler = this.customToolHandlers.get(call.name);\n    \n    if (!handler) {\n      return {\n        id: call.id,\n        content: null,\n        isError: true,\n        errorMessage: `Tool not found: ${call.name}`,\n      };\n    }\n\n    try {\n      const content = await handler(call.arguments);\n      return {\n        id: call.id,\n        content,\n        isError: false,\n      };\n    } catch (error: any) {\n      return {\n        id: call.id,\n        content: null,\n        isError: true,\n        errorMessage: error.message || 'Unknown error',\n      };\n    }\n  }\n\n  /**\n   * Register a prompt template\n   */\n  registerPrompt(prompt: McpPrompt): void {\n    this.prompts.set(prompt.name, prompt);\n  }\n\n  /**\n   * List all available prompts\n   */\n  listPrompts(): McpPrompt[] {\n    return Array.from(this.prompts.values());\n  }\n\n  /**\n   * Get prompt messages\n   */\n  getPrompt(name: string, args?: Record<string, any>): McpPromptMessage[] | null {\n    const prompt = this.prompts.get(name);\n    if (!prompt) {\n      return null;\n    }\n\n    // For now, return empty messages - actual implementation would render the prompt\n    return [];\n  }\n\n  /**\n   * List available resources\n   */\n  async listResources(): Promise<McpResource[]> {\n    // Return database stats as a resource\n    return [{\n      uri: 'sochdb://stats',\n      name: 'Database Statistics',\n      description: 'Current database statistics',\n      mimeType: 'application/json',\n    }];\n  }\n\n  /**\n   * Read a resource\n   */\n  async readResource(uri: string): Promise<McpResourceContent | null> {\n    if (uri === 'sochdb://stats') {\n      const stats = await this.db.stats();\n      return {\n        uri,\n        mimeType: 'application/json',\n        text: JSON.stringify(stats, (_, v) => \n          typeof v === 'bigint' ? v.toString() : v\n        ),\n      };\n    }\n    return null;\n  }\n\n  /**\n   * Get server info\n   */\n  getServerInfo(): McpServerConfig {\n    return { ...this.config };\n  }\n}\n"]}
|
|
@@ -0,0 +1,34 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* MCP (Model Context Protocol) Types
|
|
4
|
+
*
|
|
5
|
+
* Type definitions for Model Context Protocol integration with LLM agents.
|
|
6
|
+
*/
|
|
7
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
8
|
+
exports.MCP_ERROR_CODES = exports.McpError = void 0;
|
|
9
|
+
/**
|
|
10
|
+
* MCP Error
|
|
11
|
+
*/
|
|
12
|
+
class McpError extends Error {
|
|
13
|
+
code;
|
|
14
|
+
data;
|
|
15
|
+
constructor(message, code, data) {
|
|
16
|
+
super(message);
|
|
17
|
+
this.name = 'McpError';
|
|
18
|
+
this.code = code;
|
|
19
|
+
this.data = data;
|
|
20
|
+
}
|
|
21
|
+
}
|
|
22
|
+
exports.McpError = McpError;
|
|
23
|
+
// Error codes
|
|
24
|
+
exports.MCP_ERROR_CODES = {
|
|
25
|
+
PARSE_ERROR: -32700,
|
|
26
|
+
INVALID_REQUEST: -32600,
|
|
27
|
+
METHOD_NOT_FOUND: -32601,
|
|
28
|
+
INVALID_PARAMS: -32602,
|
|
29
|
+
INTERNAL_ERROR: -32603,
|
|
30
|
+
TOOL_NOT_FOUND: -32001,
|
|
31
|
+
RESOURCE_NOT_FOUND: -32002,
|
|
32
|
+
PROMPT_NOT_FOUND: -32003,
|
|
33
|
+
};
|
|
34
|
+
//# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidHlwZXMuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi8uLi9zcmMvbWNwL3R5cGVzLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7OztHQUlHOzs7QUFrSEg7O0dBRUc7QUFDSCxNQUFhLFFBQVMsU0FBUSxLQUFLO0lBQ2pDLElBQUksQ0FBUztJQUNiLElBQUksQ0FBTztJQUVYLFlBQVksT0FBZSxFQUFFLElBQVksRUFBRSxJQUFVO1FBQ25ELEtBQUssQ0FBQyxPQUFPLENBQUMsQ0FBQztRQUNmLElBQUksQ0FBQyxJQUFJLEdBQUcsVUFBVSxDQUFDO1FBQ3ZCLElBQUksQ0FBQyxJQUFJLEdBQUcsSUFBSSxDQUFDO1FBQ2pCLElBQUksQ0FBQyxJQUFJLEdBQUcsSUFBSSxDQUFDO0lBQ25CLENBQUM7Q0FDRjtBQVZELDRCQVVDO0FBRUQsY0FBYztBQUNELFFBQUEsZUFBZSxHQUFHO0lBQzdCLFdBQVcsRUFBRSxDQUFDLEtBQUs7SUFDbkIsZUFBZSxFQUFFLENBQUMsS0FBSztJQUN2QixnQkFBZ0IsRUFBRSxDQUFDLEtBQUs7SUFDeEIsY0FBYyxFQUFFLENBQUMsS0FBSztJQUN0QixjQUFjLEVBQUUsQ0FBQyxLQUFLO0lBQ3RCLGNBQWMsRUFBRSxDQUFDLEtBQUs7SUFDdEIsa0JBQWtCLEVBQUUsQ0FBQyxLQUFLO0lBQzFCLGdCQUFnQixFQUFFLENBQUMsS0FBSztDQUNoQixDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBNQ1AgKE1vZGVsIENvbnRleHQgUHJvdG9jb2wpIFR5cGVzXG4gKiBcbiAqIFR5cGUgZGVmaW5pdGlvbnMgZm9yIE1vZGVsIENvbnRleHQgUHJvdG9jb2wgaW50ZWdyYXRpb24gd2l0aCBMTE0gYWdlbnRzLlxuICovXG5cbi8qKlxuICogTUNQIFRvb2wgZGVmaW5pdGlvblxuICovXG5leHBvcnQgaW50ZXJmYWNlIE1jcFRvb2wge1xuICBuYW1lOiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uOiBzdHJpbmc7XG4gIGlucHV0U2NoZW1hOiBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xuICBvdXRwdXRTY2hlbWE/OiBSZWNvcmQ8c3RyaW5nLCBhbnk+O1xufVxuXG4vKipcbiAqIE1DUCBUb29sIGNhbGwgcmVxdWVzdFxuICovXG5leHBvcnQgaW50ZXJmYWNlIE1jcFRvb2xDYWxsIHtcbiAgaWQ6IHN0cmluZztcbiAgbmFtZTogc3RyaW5nO1xuICBhcmd1bWVudHM6IFJlY29yZDxzdHJpbmcsIGFueT47XG59XG5cbi8qKlxuICogTUNQIFRvb2wgY2FsbCByZXN1bHRcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBNY3BUb29sUmVzdWx0IHtcbiAgaWQ6IHN0cmluZztcbiAgY29udGVudDogYW55O1xuICBpc0Vycm9yPzogYm9vbGVhbjtcbiAgZXJyb3JNZXNzYWdlPzogc3RyaW5nO1xufVxuXG4vKipcbiAqIE1DUCBSZXNvdXJjZSBkZWZpbml0aW9uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgTWNwUmVzb3VyY2Uge1xuICB1cmk6IHN0cmluZztcbiAgbmFtZTogc3RyaW5nO1xuICBkZXNjcmlwdGlvbj86IHN0cmluZztcbiAgbWltZVR5cGU/OiBzdHJpbmc7XG59XG5cbi8qKlxuICogTUNQIFJlc291cmNlIGNvbnRlbnRcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBNY3BSZXNvdXJjZUNvbnRlbnQge1xuICB1cmk6IHN0cmluZztcbiAgbWltZVR5cGU6IHN0cmluZztcbiAgdGV4dD86IHN0cmluZztcbiAgYmxvYj86IEJ1ZmZlcjtcbn1cblxuLyoqXG4gKiBNQ1AgUHJvbXB0IGRlZmluaXRpb25cbiAqL1xuZXhwb3J0IGludGVyZmFjZSBNY3BQcm9tcHQge1xuICBuYW1lOiBzdHJpbmc7XG4gIGRlc2NyaXB0aW9uPzogc3RyaW5nO1xuICBhcmd1bWVudHM/OiBNY3BQcm9tcHRBcmd1bWVudFtdO1xufVxuXG4vKipcbiAqIE1DUCBQcm9tcHQgYXJndW1lbnRcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBNY3BQcm9tcHRBcmd1bWVudCB7XG4gIG5hbWU6IHN0cmluZztcbiAgZGVzY3JpcHRpb24/OiBzdHJpbmc7XG4gIHJlcXVpcmVkPzogYm9vbGVhbjtcbn1cblxuLyoqXG4gKiBNQ1AgUHJvbXB0IG1lc3NhZ2VcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBNY3BQcm9tcHRNZXNzYWdlIHtcbiAgcm9sZTogJ3VzZXInIHwgJ2Fzc2lzdGFudCcgfCAnc3lzdGVtJztcbiAgY29udGVudDogc3RyaW5nO1xufVxuXG4vKipcbiAqIE1DUCBTZXJ2ZXIgY2FwYWJpbGl0aWVzXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgTWNwU2VydmVyQ2FwYWJpbGl0aWVzIHtcbiAgdG9vbHM/OiBib29sZWFuO1xuICByZXNvdXJjZXM/OiBib29sZWFuO1xuICBwcm9tcHRzPzogYm9vbGVhbjtcbiAgbG9nZ2luZz86IGJvb2xlYW47XG59XG5cbi8qKlxuICogTUNQIFNlcnZlciBjb25maWd1cmF0aW9uXG4gKi9cbmV4cG9ydCBpbnRlcmZhY2UgTWNwU2VydmVyQ29uZmlnIHtcbiAgbmFtZTogc3RyaW5nO1xuICB2ZXJzaW9uOiBzdHJpbmc7XG4gIGNhcGFiaWxpdGllczogTWNwU2VydmVyQ2FwYWJpbGl0aWVzO1xufVxuXG4vKipcbiAqIE1DUCBDbGllbnQgY29uZmlndXJhdGlvblxuICovXG5leHBvcnQgaW50ZXJmYWNlIE1jcENsaWVudENvbmZpZyB7XG4gIHNlcnZlclVyaT86IHN0cmluZztcbiAgdHJhbnNwb3J0PzogJ3N0ZGlvJyB8ICdzc2UnIHwgJ3dlYnNvY2tldCc7XG4gIHRpbWVvdXQ/OiBudW1iZXI7XG59XG5cbi8qKlxuICogTUNQIFRyYW5zcG9ydCBpbnRlcmZhY2VcbiAqL1xuZXhwb3J0IGludGVyZmFjZSBNY3BUcmFuc3BvcnQge1xuICBzZW5kKG1lc3NhZ2U6IGFueSk6IFByb21pc2U8dm9pZD47XG4gIHJlY2VpdmUoKTogQXN5bmNHZW5lcmF0b3I8YW55PjtcbiAgY2xvc2UoKTogUHJvbWlzZTx2b2lkPjtcbn1cblxuLyoqXG4gKiBNQ1AgRXJyb3JcbiAqL1xuZXhwb3J0IGNsYXNzIE1jcEVycm9yIGV4dGVuZHMgRXJyb3Ige1xuICBjb2RlOiBudW1iZXI7XG4gIGRhdGE/OiBhbnk7XG5cbiAgY29uc3RydWN0b3IobWVzc2FnZTogc3RyaW5nLCBjb2RlOiBudW1iZXIsIGRhdGE/OiBhbnkpIHtcbiAgICBzdXBlcihtZXNzYWdlKTtcbiAgICB0aGlzLm5hbWUgPSAnTWNwRXJyb3InO1xuICAgIHRoaXMuY29kZSA9IGNvZGU7XG4gICAgdGhpcy5kYXRhID0gZGF0YTtcbiAgfVxufVxuXG4vLyBFcnJvciBjb2Rlc1xuZXhwb3J0IGNvbnN0IE1DUF9FUlJPUl9DT0RFUyA9IHtcbiAgUEFSU0VfRVJST1I6IC0zMjcwMCxcbiAgSU5WQUxJRF9SRVFVRVNUOiAtMzI2MDAsXG4gIE1FVEhPRF9OT1RfRk9VTkQ6IC0zMjYwMSxcbiAgSU5WQUxJRF9QQVJBTVM6IC0zMjYwMixcbiAgSU5URVJOQUxfRVJST1I6IC0zMjYwMyxcbiAgVE9PTF9OT1RfRk9VTkQ6IC0zMjAwMSxcbiAgUkVTT1VSQ0VfTk9UX0ZPVU5EOiAtMzIwMDIsXG4gIFBST01QVF9OT1RfRk9VTkQ6IC0zMjAwMyxcbn0gYXMgY29uc3Q7XG4iXX0=
|