aws-lambda-mcp-server 0.0.1 → 0.0.3
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 +26 -20
- package/dist/index.d.ts +31 -0
- package/dist/index.d.ts.map +1 -0
- package/dist/index.js +165 -0
- package/package.json +16 -10
- package/eslint.config.ts +0 -72
- package/src/index.ts +0 -107
- package/tsconfig-eslint.json +0 -15
- package/tsconfig.json +0 -38
package/README.md
CHANGED
|
@@ -1,5 +1,7 @@
|
|
|
1
1
|
# aws-lambda-mcp-server
|
|
2
2
|
|
|
3
|
+
[](https://badge.fury.io/js/aws-lambda-mcp-server)
|
|
4
|
+
|
|
3
5
|
AWS Lambda上でMCP(Model Context Protocol)サーバー機能を提供するためのライブラリです。
|
|
4
6
|
APIエンドポイントの実装や、各種AWSサービスとの連携を簡易化することを目的としています。
|
|
5
7
|
|
|
@@ -19,29 +21,33 @@ TypeScriptでの利用例です。
|
|
|
19
21
|
|
|
20
22
|
```typescript
|
|
21
23
|
import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
|
|
22
|
-
import {
|
|
24
|
+
import { createHonoApp } from 'aws-lambda-mcp-server';
|
|
23
25
|
import { handle } from 'hono/aws-lambda';
|
|
24
|
-
|
|
25
|
-
|
|
26
|
-
|
|
27
|
-
|
|
28
|
-
|
|
29
|
-
|
|
30
|
-
|
|
31
|
-
|
|
32
|
-
|
|
33
|
-
|
|
34
|
-
|
|
35
|
-
|
|
36
|
-
|
|
37
|
-
|
|
38
|
-
|
|
39
|
-
|
|
40
|
-
|
|
41
|
-
|
|
26
|
+
import { z } from 'zod';
|
|
27
|
+
|
|
28
|
+
// MCPサーバーのファクトリ関数を用意
|
|
29
|
+
const createMcpServer = () => {
|
|
30
|
+
const server = new McpServer({
|
|
31
|
+
name: 'my-mcp-server',
|
|
32
|
+
version: '1.0.0',
|
|
33
|
+
});
|
|
34
|
+
|
|
35
|
+
// MCPサーバーのインスタンスにToolsやResourcesなどを設定する
|
|
36
|
+
server.tool(
|
|
37
|
+
'say_hello',
|
|
38
|
+
{ who: z.string() },
|
|
39
|
+
async ({ who }) => ({
|
|
40
|
+
content: [{
|
|
41
|
+
type: 'text',
|
|
42
|
+
text: `${who} さん、こんにちは!`
|
|
43
|
+
}]
|
|
44
|
+
})
|
|
45
|
+
);
|
|
46
|
+
return server;
|
|
47
|
+
};
|
|
42
48
|
|
|
43
49
|
// Hono アプリケーションを作成
|
|
44
|
-
const app = createHonoApp(
|
|
50
|
+
const app = createHonoApp(createMcpServer);
|
|
45
51
|
|
|
46
52
|
// AWS Lambdaのエントリポイントとして利用
|
|
47
53
|
export const handler = handle(app);
|
package/dist/index.d.ts
ADDED
|
@@ -0,0 +1,31 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Model Context Protocol (MCP) サーバーを Hono フレームワーク上で動作させるエントリーポイント。
|
|
3
|
+
*
|
|
4
|
+
* @remarks
|
|
5
|
+
* `createHonoApp` 関数を通じて、/mcp エンドポイントでMCPサーバーを提供します。
|
|
6
|
+
* - POST/GET /mcp: MCPリクエストの受信・処理
|
|
7
|
+
* - その他のHTTPメソッド: 405 Method Not Allowed
|
|
8
|
+
*
|
|
9
|
+
* 内部的にエラーハンドリングやリソースクローズ処理も行います。
|
|
10
|
+
*/
|
|
11
|
+
import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
|
|
12
|
+
import { Hono } from 'hono';
|
|
13
|
+
import { BlankEnv } from 'hono/types';
|
|
14
|
+
/**
|
|
15
|
+
* Honoアプリケーションを生成し、/mcpエンドポイントでMCPサーバーを提供します。
|
|
16
|
+
*
|
|
17
|
+
* @remarks
|
|
18
|
+
* POST/GET /mcp でMCPリクエストを受け付け、他のHTTPメソッドは405を返します。
|
|
19
|
+
*
|
|
20
|
+
* @param createMcpServer MCPサーバーインスタンスを生成するファクトリ関数
|
|
21
|
+
* @returns Honoアプリケーションインスタンス
|
|
22
|
+
*
|
|
23
|
+
* @example
|
|
24
|
+
* ```ts
|
|
25
|
+
* import { createHonoApp } from '...';
|
|
26
|
+
* import { createMcpServer } from './your-mcp-server';
|
|
27
|
+
* const app = createHonoApp(createMcpServer);
|
|
28
|
+
* ```
|
|
29
|
+
*/
|
|
30
|
+
export declare const createHonoApp: (createMcpServer: () => McpServer) => Hono<BlankEnv, import("hono/types").BlankSchema, "/">;
|
|
31
|
+
//# sourceMappingURL=index.d.ts.map
|
|
@@ -0,0 +1 @@
|
|
|
1
|
+
{"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAAA;;;;;;;;;GASG;AAIH,OAAO,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AACpE,OAAO,EAAW,IAAI,EAAE,MAAM,MAAM,CAAC;AACrC,OAAO,EAAE,QAAQ,EAAc,MAAM,YAAY,CAAC;AA0IlD;;;;;;;;;;;;;;;GAeG;AACH,eAAO,MAAM,aAAa,GAAI,iBAAiB,MAAM,SAAS,0DAiB7D,CAAC"}
|
package/dist/index.js
ADDED
|
@@ -0,0 +1,165 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Model Context Protocol (MCP) サーバーを Hono フレームワーク上で動作させるエントリーポイント。
|
|
3
|
+
*
|
|
4
|
+
* @remarks
|
|
5
|
+
* `createHonoApp` 関数を通じて、/mcp エンドポイントでMCPサーバーを提供します。
|
|
6
|
+
* - POST/GET /mcp: MCPリクエストの受信・処理
|
|
7
|
+
* - その他のHTTPメソッド: 405 Method Not Allowed
|
|
8
|
+
*
|
|
9
|
+
* 内部的にエラーハンドリングやリソースクローズ処理も行います。
|
|
10
|
+
*/
|
|
11
|
+
import { Logger } from '@aws-lambda-powertools/logger';
|
|
12
|
+
import { StreamableHTTPTransport } from '@hono/mcp';
|
|
13
|
+
import { Hono } from 'hono';
|
|
14
|
+
/**
|
|
15
|
+
* ロガーインスタンス(AWS Lambda Powertools)。
|
|
16
|
+
*
|
|
17
|
+
* @private
|
|
18
|
+
*/
|
|
19
|
+
const logger = new Logger();
|
|
20
|
+
/**
|
|
21
|
+
* 許可されていないHTTPメソッドに対するハンドラーです。
|
|
22
|
+
*
|
|
23
|
+
* @remarks
|
|
24
|
+
* 405エラーのJSONレスポンスを返します。
|
|
25
|
+
*
|
|
26
|
+
* @param c Honoのコンテキスト
|
|
27
|
+
* @returns 405エラーのJSONレスポンス
|
|
28
|
+
* @private
|
|
29
|
+
*/
|
|
30
|
+
const methodNotAllowedHandler = async (c) => {
|
|
31
|
+
return c.json({
|
|
32
|
+
jsonrpc: '2.0',
|
|
33
|
+
error: {
|
|
34
|
+
code: -32000,
|
|
35
|
+
message: 'メソッドは許可されていません。',
|
|
36
|
+
},
|
|
37
|
+
id: null,
|
|
38
|
+
}, { status: 405 });
|
|
39
|
+
};
|
|
40
|
+
/**
|
|
41
|
+
* サーバーエラー発生時の共通エラーハンドラーです。
|
|
42
|
+
*
|
|
43
|
+
* @remarks
|
|
44
|
+
* エラー内容をロギングし、500エラーのJSONレスポンスを返します。
|
|
45
|
+
*
|
|
46
|
+
* @param c Honoのコンテキスト
|
|
47
|
+
* @param reason エラー理由
|
|
48
|
+
* @param logMessage ログ出力用メッセージ
|
|
49
|
+
* @returns 500エラーのJSONレスポンス
|
|
50
|
+
* @private
|
|
51
|
+
*/
|
|
52
|
+
const handleError = (c, reason, logMessage) => {
|
|
53
|
+
const errorDetails = reason instanceof Error
|
|
54
|
+
? { message: reason.message, stack: reason.stack, name: reason.name }
|
|
55
|
+
: { reason };
|
|
56
|
+
logger.error(logMessage, errorDetails);
|
|
57
|
+
return c.json({
|
|
58
|
+
jsonrpc: '2.0',
|
|
59
|
+
error: {
|
|
60
|
+
code: -32603,
|
|
61
|
+
message: '内部サーバーエラー',
|
|
62
|
+
},
|
|
63
|
+
id: null,
|
|
64
|
+
}, { status: 500 });
|
|
65
|
+
};
|
|
66
|
+
/**
|
|
67
|
+
* MCPサーバーおよびトランスポートのリソースをクローズします。
|
|
68
|
+
*
|
|
69
|
+
* @remarks
|
|
70
|
+
* どちらか一方のクローズに失敗しても、もう一方は必ず実行されます。
|
|
71
|
+
*
|
|
72
|
+
* @param server MCPサーバーインスタンス
|
|
73
|
+
* @param transport トランスポートインスタンス
|
|
74
|
+
* @returns void
|
|
75
|
+
* @private
|
|
76
|
+
*/
|
|
77
|
+
const closeResources = async (server, transport) => {
|
|
78
|
+
// 両方のクローズを確実に実行(片方が失敗してももう片方を実行)
|
|
79
|
+
const closeResults = await Promise.allSettled([
|
|
80
|
+
transport.close(),
|
|
81
|
+
server.close(),
|
|
82
|
+
]);
|
|
83
|
+
// クローズエラーをログ出力
|
|
84
|
+
closeResults.forEach((result, index) => {
|
|
85
|
+
if (result.status === 'rejected') {
|
|
86
|
+
const resourceName = index === 0 ? 'transport' : 'server';
|
|
87
|
+
const error = result.reason;
|
|
88
|
+
const errorDetails = error instanceof Error
|
|
89
|
+
? { message: error.message, stack: error.stack }
|
|
90
|
+
: error;
|
|
91
|
+
logger.error(`Error closing ${resourceName}:`, { error: errorDetails });
|
|
92
|
+
}
|
|
93
|
+
});
|
|
94
|
+
};
|
|
95
|
+
/**
|
|
96
|
+
* MCPリクエストを処理します。
|
|
97
|
+
*
|
|
98
|
+
* @remarks
|
|
99
|
+
* サーバーとトランスポートの接続・リクエスト処理・エラーハンドリングを行います。
|
|
100
|
+
*
|
|
101
|
+
* @param createMcpServer MCPサーバーインスタンスを生成するファクトリ関数
|
|
102
|
+
* @param c Honoのコンテキスト
|
|
103
|
+
* @returns MCPレスポンス
|
|
104
|
+
* @private
|
|
105
|
+
*/
|
|
106
|
+
const handleRequest = async (createMcpServer, c) => {
|
|
107
|
+
const transport = new StreamableHTTPTransport({
|
|
108
|
+
sessionIdGenerator: undefined, // セッションIDを生成しない(ステートレスモード)
|
|
109
|
+
enableJsonResponse: true,
|
|
110
|
+
});
|
|
111
|
+
const server = createMcpServer();
|
|
112
|
+
try {
|
|
113
|
+
await server.connect(transport);
|
|
114
|
+
logger.trace('MCP リクエストを受信');
|
|
115
|
+
return await transport.handleRequest(c);
|
|
116
|
+
}
|
|
117
|
+
catch (error) {
|
|
118
|
+
return handleError(c, error, 'MCP 接続中のエラー:');
|
|
119
|
+
}
|
|
120
|
+
finally {
|
|
121
|
+
// エラーの有無に関わらず必ずリソースをクローズ
|
|
122
|
+
try {
|
|
123
|
+
await closeResources(server, transport);
|
|
124
|
+
}
|
|
125
|
+
catch (closeError) {
|
|
126
|
+
// クローズエラーは既にcloseResources内でログ出力されているため、
|
|
127
|
+
// ここでは追加のエラーハンドリングは不要だが、エラーの詳細を記録
|
|
128
|
+
const errorDetails = closeError instanceof Error
|
|
129
|
+
? { message: closeError.message, stack: closeError.stack }
|
|
130
|
+
: closeError;
|
|
131
|
+
logger.error('リソースクローズ中に追加エラーが発生しましたが、処理を継続します', { closeError: errorDetails });
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
};
|
|
135
|
+
/**
|
|
136
|
+
* Honoアプリケーションを生成し、/mcpエンドポイントでMCPサーバーを提供します。
|
|
137
|
+
*
|
|
138
|
+
* @remarks
|
|
139
|
+
* POST/GET /mcp でMCPリクエストを受け付け、他のHTTPメソッドは405を返します。
|
|
140
|
+
*
|
|
141
|
+
* @param createMcpServer MCPサーバーインスタンスを生成するファクトリ関数
|
|
142
|
+
* @returns Honoアプリケーションインスタンス
|
|
143
|
+
*
|
|
144
|
+
* @example
|
|
145
|
+
* ```ts
|
|
146
|
+
* import { createHonoApp } from '...';
|
|
147
|
+
* import { createMcpServer } from './your-mcp-server';
|
|
148
|
+
* const app = createHonoApp(createMcpServer);
|
|
149
|
+
* ```
|
|
150
|
+
*/
|
|
151
|
+
export const createHonoApp = (createMcpServer) => {
|
|
152
|
+
const app = new Hono();
|
|
153
|
+
app.post('/mcp', async (c) => {
|
|
154
|
+
return await handleRequest(createMcpServer, c);
|
|
155
|
+
});
|
|
156
|
+
app.get('/mcp', async (c) => {
|
|
157
|
+
return await handleRequest(createMcpServer, c);
|
|
158
|
+
});
|
|
159
|
+
app.put('/mcp', methodNotAllowedHandler);
|
|
160
|
+
app.delete('/mcp', methodNotAllowedHandler);
|
|
161
|
+
app.patch('/mcp', methodNotAllowedHandler);
|
|
162
|
+
app.options('/mcp', methodNotAllowedHandler);
|
|
163
|
+
return app;
|
|
164
|
+
};
|
|
165
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Model Context Protocol (MCP) サーバーを Hono フレームワーク上で動作させるエントリーポイント。\n *\n * @remarks\n * `createHonoApp` 関数を通じて、/mcp エンドポイントでMCPサーバーを提供します。\n * - POST/GET /mcp: MCPリクエストの受信・処理\n * - その他のHTTPメソッド: 405 Method Not Allowed\n *\n * 内部的にエラーハンドリングやリソースクローズ処理も行います。\n */\n\nimport { Logger } from '@aws-lambda-powertools/logger';\nimport { StreamableHTTPTransport } from '@hono/mcp';\nimport { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';\nimport { Context, Hono } from 'hono';\nimport { BlankEnv, BlankInput } from 'hono/types';\n\n/**\n * ロガーインスタンス（AWS Lambda Powertools）。\n *\n * @private\n */\nconst logger = new Logger();\n\n/**\n * 許可されていないHTTPメソッドに対するハンドラーです。\n *\n * @remarks\n * 405エラーのJSONレスポンスを返します。\n *\n * @param c Honoのコンテキスト\n * @returns 405エラーのJSONレスポンス\n * @private\n */\nconst methodNotAllowedHandler = async (\n  c: Context<BlankEnv, '/mcp', BlankInput>,\n) => {\n  return c.json(\n    {\n      jsonrpc: '2.0',\n      error: {\n        code: -32000,\n        message: 'メソッドは許可されていません。',\n      },\n      id: null,\n    },\n    { status: 405 },\n  );\n};\n\n/**\n * サーバーエラー発生時の共通エラーハンドラーです。\n *\n * @remarks\n * エラー内容をロギングし、500エラーのJSONレスポンスを返します。\n *\n * @param c Honoのコンテキスト\n * @param reason エラー理由\n * @param logMessage ログ出力用メッセージ\n * @returns 500エラーのJSONレスポンス\n * @private\n */\nconst handleError = (\n  c: Context<BlankEnv, '/mcp', BlankInput>,\n  reason: unknown,\n  logMessage: string,\n) => {\n  const errorDetails = reason instanceof Error\n    ? { message: reason.message, stack: reason.stack, name: reason.name }\n    : { reason };\n  logger.error(logMessage, errorDetails);\n  return c.json(\n    {\n      jsonrpc: '2.0',\n      error: {\n        code: -32603,\n        message: '内部サーバーエラー',\n      },\n      id: null,\n    },\n    { status: 500 },\n  );\n};\n\n/**\n * MCPサーバーおよびトランスポートのリソースをクローズします。\n *\n * @remarks\n * どちらか一方のクローズに失敗しても、もう一方は必ず実行されます。\n *\n * @param server MCPサーバーインスタンス\n * @param transport トランスポートインスタンス\n * @returns void\n * @private\n */\nconst closeResources = async (server: McpServer, transport: StreamableHTTPTransport) => {\n  // 両方のクローズを確実に実行（片方が失敗してももう片方を実行）\n  const closeResults = await Promise.allSettled([\n    transport.close(),\n    server.close(),\n  ]);\n\n  // クローズエラーをログ出力\n  closeResults.forEach((result, index) => {\n    if (result.status === 'rejected') {\n      const resourceName = index === 0 ? 'transport' : 'server';\n      const error = result.reason;\n      const errorDetails = error instanceof Error\n        ? { message: error.message, stack: error.stack }\n        : error;\n      logger.error(`Error closing ${resourceName}:`, { error: errorDetails });\n    }\n  });\n};\n\n/**\n * MCPリクエストを処理します。\n *\n * @remarks\n * サーバーとトランスポートの接続・リクエスト処理・エラーハンドリングを行います。\n *\n * @param createMcpServer MCPサーバーインスタンスを生成するファクトリ関数\n * @param c Honoのコンテキスト\n * @returns MCPレスポンス\n * @private\n */\nconst handleRequest = async (createMcpServer: () => McpServer, c: Context<BlankEnv, '/mcp', BlankInput>) => {\n  const transport = new StreamableHTTPTransport({\n    sessionIdGenerator: undefined, // セッションIDを生成しない（ステートレスモード）\n    enableJsonResponse: true,\n  });\n  const server = createMcpServer();\n  try {\n    await server.connect(transport);\n    logger.trace('MCP リクエストを受信');\n    return await transport.handleRequest(c);\n  } catch (error) {\n    return handleError(c, error, 'MCP 接続中のエラー:');\n  } finally {\n    // エラーの有無に関わらず必ずリソースをクローズ\n    try {\n      await closeResources(server, transport);\n    } catch (closeError) {\n      // クローズエラーは既にcloseResources内でログ出力されているため、\n      // ここでは追加のエラーハンドリングは不要だが、エラーの詳細を記録\n      const errorDetails = closeError instanceof Error\n        ? { message: closeError.message, stack: closeError.stack }\n        : closeError;\n      logger.error('リソースクローズ中に追加エラーが発生しましたが、処理を継続します', { closeError: errorDetails });\n    }\n  }\n};\n\n/**\n * Honoアプリケーションを生成し、/mcpエンドポイントでMCPサーバーを提供します。\n *\n * @remarks\n * POST/GET /mcp でMCPリクエストを受け付け、他のHTTPメソッドは405を返します。\n *\n * @param createMcpServer MCPサーバーインスタンスを生成するファクトリ関数\n * @returns Honoアプリケーションインスタンス\n *\n * @example\n * ```ts\n * import { createHonoApp } from '...';\n * import { createMcpServer } from './your-mcp-server';\n * const app = createHonoApp(createMcpServer);\n * ```\n */\nexport const createHonoApp = (createMcpServer: () => McpServer) => {\n  const app = new Hono();\n\n  app.post('/mcp', async (c) => {\n    return await handleRequest(createMcpServer, c);\n  });\n\n  app.get('/mcp', async (c) => {\n    return await handleRequest(createMcpServer, c);\n  });\n\n  app.put('/mcp', methodNotAllowedHandler);\n  app.delete('/mcp', methodNotAllowedHandler);\n  app.patch('/mcp', methodNotAllowedHandler);\n  app.options('/mcp', methodNotAllowedHandler);\n\n  return app;\n};\n"]}
|
package/package.json
CHANGED
|
@@ -1,6 +1,6 @@
|
|
|
1
1
|
{
|
|
2
2
|
"name": "aws-lambda-mcp-server",
|
|
3
|
-
"version": "0.0.
|
|
3
|
+
"version": "0.0.3",
|
|
4
4
|
"description": "A Hono wrapper for building an MCP (Model Context Protocol) Server that runs on AWS Lambda functions.",
|
|
5
5
|
"type": "module",
|
|
6
6
|
"keywords": [
|
|
@@ -11,6 +11,11 @@
|
|
|
11
11
|
],
|
|
12
12
|
"private": false,
|
|
13
13
|
"sideEffects": false,
|
|
14
|
+
"files": [
|
|
15
|
+
"dist",
|
|
16
|
+
"package.json",
|
|
17
|
+
"README.md"
|
|
18
|
+
],
|
|
14
19
|
"homepage": "https://github.com/poad/aws-lambda-mcp-server#readme",
|
|
15
20
|
"repository": {
|
|
16
21
|
"type": "git",
|
|
@@ -23,31 +28,31 @@
|
|
|
23
28
|
}
|
|
24
29
|
},
|
|
25
30
|
"module": "./dist/index.js",
|
|
26
|
-
"types": "dist/index.d.ts",
|
|
31
|
+
"types": "./dist/index.d.ts",
|
|
27
32
|
"main": "./dist/index.js",
|
|
28
33
|
"author": "poad",
|
|
29
34
|
"license": "ISC",
|
|
30
35
|
"devDependencies": {
|
|
31
36
|
"@eslint/compat": "^1.4.1",
|
|
32
|
-
"@eslint/js": "^9.39.
|
|
37
|
+
"@eslint/js": "^9.39.1",
|
|
33
38
|
"@stylistic/eslint-plugin": "^5.5.0",
|
|
34
|
-
"@types/node": "24.
|
|
35
|
-
"eslint": "^9.39.
|
|
39
|
+
"@types/node": "24.10.0",
|
|
40
|
+
"eslint": "^9.39.1",
|
|
36
41
|
"eslint-import-resolver-typescript": "^4.4.4",
|
|
37
42
|
"eslint-plugin-import": "^2.32.0",
|
|
38
43
|
"eslint-plugin-promise": "^7.2.1",
|
|
39
44
|
"jiti": "^2.6.1",
|
|
40
45
|
"tsx": "^4.20.6",
|
|
41
46
|
"typescript": "^5.9.3",
|
|
42
|
-
"typescript-eslint": "^8.46.
|
|
43
|
-
"vitest": "^4.0.
|
|
47
|
+
"typescript-eslint": "^8.46.3",
|
|
48
|
+
"vitest": "^4.0.7"
|
|
44
49
|
},
|
|
45
50
|
"dependencies": {
|
|
46
51
|
"@aws-lambda-powertools/logger": "^2.28.1",
|
|
47
52
|
"@hono/mcp": "^0.1.5",
|
|
48
|
-
"@modelcontextprotocol/sdk": "^1.
|
|
53
|
+
"@modelcontextprotocol/sdk": "^1.21.0",
|
|
49
54
|
"hono": "^4.10.4",
|
|
50
|
-
"zod": "^
|
|
55
|
+
"zod": "^4.1.12"
|
|
51
56
|
},
|
|
52
57
|
"scripts": {
|
|
53
58
|
"build": "tsc",
|
|
@@ -55,5 +60,6 @@
|
|
|
55
60
|
"test": "vitest run --passWithNoTests",
|
|
56
61
|
"lint": "eslint .",
|
|
57
62
|
"lint-fix": "eslint . --fix"
|
|
58
|
-
}
|
|
63
|
+
},
|
|
64
|
+
"typings": "./dist/index.d.ts"
|
|
59
65
|
}
|
package/eslint.config.ts
DELETED
|
@@ -1,72 +0,0 @@
|
|
|
1
|
-
import { Config, defineConfig } from 'eslint/config';
|
|
2
|
-
import eslint from '@eslint/js';
|
|
3
|
-
import { configs, parser } from 'typescript-eslint';
|
|
4
|
-
import stylistic from '@stylistic/eslint-plugin';
|
|
5
|
-
import importPlugin from 'eslint-plugin-import';
|
|
6
|
-
// @ts-expect-error ignore type errors
|
|
7
|
-
import pluginPromise from 'eslint-plugin-promise';
|
|
8
|
-
|
|
9
|
-
import { includeIgnoreFile } from '@eslint/compat';
|
|
10
|
-
import path from 'node:path';
|
|
11
|
-
import { fileURLToPath } from 'node:url';
|
|
12
|
-
|
|
13
|
-
const __filename = fileURLToPath(import.meta.url);
|
|
14
|
-
const __dirname = path.dirname(__filename);
|
|
15
|
-
const gitignorePath = path.resolve(__dirname, '.gitignore');
|
|
16
|
-
|
|
17
|
-
const eslintConfig: Config[] = defineConfig(
|
|
18
|
-
{
|
|
19
|
-
ignores: [
|
|
20
|
-
...(includeIgnoreFile(gitignorePath).ignores || []),
|
|
21
|
-
'**/*.d.ts',
|
|
22
|
-
'src/tsconfig.json',
|
|
23
|
-
'src/stories',
|
|
24
|
-
'**/*.css',
|
|
25
|
-
'node_modules/**/*',
|
|
26
|
-
'out',
|
|
27
|
-
'cdk.out',
|
|
28
|
-
'dist',
|
|
29
|
-
'app',
|
|
30
|
-
],
|
|
31
|
-
},
|
|
32
|
-
eslint.configs.recommended,
|
|
33
|
-
configs.strict,
|
|
34
|
-
configs.stylistic,
|
|
35
|
-
pluginPromise.configs['flat/recommended'],
|
|
36
|
-
{
|
|
37
|
-
files: ['**/*.ts'],
|
|
38
|
-
plugins: {
|
|
39
|
-
'@stylistic': stylistic,
|
|
40
|
-
},
|
|
41
|
-
languageOptions: {
|
|
42
|
-
ecmaVersion: 'latest',
|
|
43
|
-
sourceType: 'module',
|
|
44
|
-
parser,
|
|
45
|
-
parserOptions: {
|
|
46
|
-
tsconfigRootDir: __dirname,
|
|
47
|
-
project: ['./tsconfig-eslint.json'],
|
|
48
|
-
},
|
|
49
|
-
},
|
|
50
|
-
extends: [
|
|
51
|
-
importPlugin.flatConfigs.recommended,
|
|
52
|
-
importPlugin.flatConfigs.typescript,
|
|
53
|
-
],
|
|
54
|
-
settings: {
|
|
55
|
-
'import/resolver': {
|
|
56
|
-
// You will also need to install and configure the TypeScript resolver
|
|
57
|
-
// See also https://github.com/import-js/eslint-import-resolver-typescript#configuration
|
|
58
|
-
'typescript': true,
|
|
59
|
-
'node': true,
|
|
60
|
-
},
|
|
61
|
-
},
|
|
62
|
-
rules: {
|
|
63
|
-
'@stylistic/semi': ['error', 'always'],
|
|
64
|
-
'@stylistic/indent': ['error', 2],
|
|
65
|
-
'@stylistic/comma-dangle': ['error', 'always-multiline'],
|
|
66
|
-
'@stylistic/arrow-parens': ['error', 'always'],
|
|
67
|
-
'@stylistic/quotes': ['error', 'single'],
|
|
68
|
-
},
|
|
69
|
-
},
|
|
70
|
-
);
|
|
71
|
-
|
|
72
|
-
export default eslintConfig;
|
package/src/index.ts
DELETED
|
@@ -1,107 +0,0 @@
|
|
|
1
|
-
import { Logger } from '@aws-lambda-powertools/logger';
|
|
2
|
-
import { StreamableHTTPTransport } from '@hono/mcp';
|
|
3
|
-
import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
|
|
4
|
-
import { Context, Hono } from 'hono';
|
|
5
|
-
import { BlankEnv, BlankInput } from 'hono/types';
|
|
6
|
-
|
|
7
|
-
const logger = new Logger();
|
|
8
|
-
|
|
9
|
-
const methodNotAllowedHandler = async (
|
|
10
|
-
c: Context<BlankEnv, '/mcp', BlankInput>,
|
|
11
|
-
) => {
|
|
12
|
-
return c.json(
|
|
13
|
-
{
|
|
14
|
-
jsonrpc: '2.0',
|
|
15
|
-
error: {
|
|
16
|
-
code: -32000,
|
|
17
|
-
message: 'メソッドは許可されていません。',
|
|
18
|
-
},
|
|
19
|
-
id: null,
|
|
20
|
-
},
|
|
21
|
-
{ status: 405 },
|
|
22
|
-
);
|
|
23
|
-
};
|
|
24
|
-
|
|
25
|
-
const handleError = (
|
|
26
|
-
c: Context<BlankEnv, '/mcp', BlankInput>,
|
|
27
|
-
reason: unknown,
|
|
28
|
-
logMessage: string,
|
|
29
|
-
) => {
|
|
30
|
-
const errorDetails = reason instanceof Error
|
|
31
|
-
? { message: reason.message, stack: reason.stack, name: reason.name }
|
|
32
|
-
: { reason };
|
|
33
|
-
logger.error(logMessage, errorDetails);
|
|
34
|
-
return c.json(
|
|
35
|
-
{
|
|
36
|
-
jsonrpc: '2.0',
|
|
37
|
-
error: {
|
|
38
|
-
code: -32603,
|
|
39
|
-
message: '内部サーバーエラー',
|
|
40
|
-
},
|
|
41
|
-
id: null,
|
|
42
|
-
},
|
|
43
|
-
{ status: 500 },
|
|
44
|
-
);
|
|
45
|
-
};
|
|
46
|
-
|
|
47
|
-
const closeResources = async (server: McpServer, transport: StreamableHTTPTransport) => {
|
|
48
|
-
// 両方のクローズを確実に実行(片方が失敗してももう片方を実行)
|
|
49
|
-
const closeResults = await Promise.allSettled([
|
|
50
|
-
transport.close(),
|
|
51
|
-
server.close(),
|
|
52
|
-
]);
|
|
53
|
-
|
|
54
|
-
// クローズエラーをログ出力
|
|
55
|
-
closeResults.forEach((result, index) => {
|
|
56
|
-
if (result.status === 'rejected') {
|
|
57
|
-
const resourceName = index === 0 ? 'transport' : 'server';
|
|
58
|
-
const error = result.reason;
|
|
59
|
-
const errorDetails = error instanceof Error
|
|
60
|
-
? { message: error.message, stack: error.stack }
|
|
61
|
-
: error;
|
|
62
|
-
logger.error(`Error closing ${resourceName}:`, { error: errorDetails });
|
|
63
|
-
}
|
|
64
|
-
});
|
|
65
|
-
};
|
|
66
|
-
|
|
67
|
-
const handleRequest = async (server: McpServer, c: Context<BlankEnv, '/mcp', BlankInput>) => {
|
|
68
|
-
const transport = new StreamableHTTPTransport({
|
|
69
|
-
sessionIdGenerator: undefined, // セッションIDを生成しない(ステートレスモード)
|
|
70
|
-
enableJsonResponse: true,
|
|
71
|
-
});
|
|
72
|
-
try {
|
|
73
|
-
await server.connect(transport);
|
|
74
|
-
logger.trace('MCP リクエストを受信');
|
|
75
|
-
return await transport.handleRequest(c);
|
|
76
|
-
} catch (error) {
|
|
77
|
-
try {
|
|
78
|
-
await closeResources(server, transport);
|
|
79
|
-
} catch (closeError) {
|
|
80
|
-
const errorDetails = closeError instanceof Error
|
|
81
|
-
? { message: closeError.message, stack: closeError.stack }
|
|
82
|
-
: closeError;
|
|
83
|
-
logger.error('Transport close failed after connection error:', { closeError: errorDetails });
|
|
84
|
-
}
|
|
85
|
-
return handleError(c, error, 'MCP 接続中のエラー:');
|
|
86
|
-
}
|
|
87
|
-
|
|
88
|
-
};
|
|
89
|
-
|
|
90
|
-
export const createHonoApp = (server: McpServer) => {
|
|
91
|
-
const app = new Hono();
|
|
92
|
-
|
|
93
|
-
app.post('/mcp', async (c) => {
|
|
94
|
-
return await handleRequest(server, c);
|
|
95
|
-
});
|
|
96
|
-
|
|
97
|
-
app.get('/mcp', async (c) => {
|
|
98
|
-
return await handleRequest(server, c);
|
|
99
|
-
});
|
|
100
|
-
|
|
101
|
-
app.put('/mcp', methodNotAllowedHandler);
|
|
102
|
-
app.delete('/mcp', methodNotAllowedHandler);
|
|
103
|
-
app.patch('/mcp', methodNotAllowedHandler);
|
|
104
|
-
app.options('/mcp', methodNotAllowedHandler);
|
|
105
|
-
|
|
106
|
-
return app;
|
|
107
|
-
};
|
package/tsconfig-eslint.json
DELETED
package/tsconfig.json
DELETED
|
@@ -1,38 +0,0 @@
|
|
|
1
|
-
{
|
|
2
|
-
"compilerOptions": {
|
|
3
|
-
"target": "ES2024",
|
|
4
|
-
"module": "NodeNext",
|
|
5
|
-
"moduleResolution": "NodeNext",
|
|
6
|
-
"lib": [
|
|
7
|
-
"es2024"
|
|
8
|
-
],
|
|
9
|
-
"declaration": true,
|
|
10
|
-
"strict": true,
|
|
11
|
-
"noImplicitAny": true,
|
|
12
|
-
"strictNullChecks": true,
|
|
13
|
-
"noImplicitThis": true,
|
|
14
|
-
"alwaysStrict": true,
|
|
15
|
-
"noUnusedLocals": false,
|
|
16
|
-
"noUnusedParameters": false,
|
|
17
|
-
"noImplicitReturns": true,
|
|
18
|
-
"noFallthroughCasesInSwitch": false,
|
|
19
|
-
"inlineSourceMap": true,
|
|
20
|
-
"inlineSources": true,
|
|
21
|
-
"experimentalDecorators": true,
|
|
22
|
-
"strictPropertyInitialization": false,
|
|
23
|
-
"skipLibCheck": true,
|
|
24
|
-
"declarationMap": true,
|
|
25
|
-
"outDir": "dist",
|
|
26
|
-
"typeRoots": [
|
|
27
|
-
"./node_modules/@types"
|
|
28
|
-
],
|
|
29
|
-
"types": [
|
|
30
|
-
"node"
|
|
31
|
-
]
|
|
32
|
-
},
|
|
33
|
-
"exclude": [
|
|
34
|
-
"node_modules",
|
|
35
|
-
"*.config.ts",
|
|
36
|
-
"dist"
|
|
37
|
-
]
|
|
38
|
-
}
|