aws-lambda-mcp-server 0.0.2 → 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 CHANGED
@@ -1,5 +1,7 @@
1
1
  # aws-lambda-mcp-server
2
2
 
3
+ [![npm version](https://badge.fury.io/js/aws-lambda-mcp-server.svg)](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 { handler } from 'aws-lambda-mcp-server';
24
+ import { createHonoApp } from 'aws-lambda-mcp-server';
23
25
  import { handle } from 'hono/aws-lambda';
24
-
25
- // MCPサーバーのインスタンスを作成
26
- const server = new McpServer({
27
- name: 'my-mcp-server',
28
- version: '1.0.0',
29
- });
30
-
31
- // MCPサーバーのインスタンスにToolsやResourcesなどを設定する
32
- server.tool(
33
- "say_hello",
34
- { who: z.string() },
35
- async ({ who }) => ({
36
- content: [{
37
- type: "text",
38
- text: `${who} さん、こんにちは!`
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(server);
50
+ const app = createHonoApp(createMcpServer);
45
51
 
46
52
  // AWS Lambdaのエントリポイントとして利用
47
53
  export const handler = handle(app);
package/dist/index.d.ts CHANGED
@@ -1,5 +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
+ */
1
11
  import { McpServer } from '@modelcontextprotocol/sdk/server/mcp.js';
2
12
  import { Hono } from 'hono';
3
13
  import { BlankEnv } from 'hono/types';
4
- export declare const createHonoApp: (server: McpServer) => Hono<BlankEnv, import("hono/types").BlankSchema, "/">;
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, "/">;
5
31
  //# sourceMappingURL=index.d.ts.map
@@ -1 +1 @@
1
- {"version":3,"file":"index.d.ts","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"AAEA,OAAO,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AACpE,OAAO,EAAW,IAAI,EAAE,MAAM,MAAM,CAAC;AACrC,OAAO,EAAE,QAAQ,EAAc,MAAM,YAAY,CAAC;AAqFlD,eAAO,MAAM,aAAa,GAAI,QAAQ,SAAS,0DAiB9C,CAAC"}
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 CHANGED
@@ -1,7 +1,32 @@
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
+ */
1
11
  import { Logger } from '@aws-lambda-powertools/logger';
2
12
  import { StreamableHTTPTransport } from '@hono/mcp';
3
13
  import { Hono } from 'hono';
14
+ /**
15
+ * ロガーインスタンス(AWS Lambda Powertools)。
16
+ *
17
+ * @private
18
+ */
4
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
+ */
5
30
  const methodNotAllowedHandler = async (c) => {
6
31
  return c.json({
7
32
  jsonrpc: '2.0',
@@ -12,6 +37,18 @@ const methodNotAllowedHandler = async (c) => {
12
37
  id: null,
13
38
  }, { status: 405 });
14
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
+ */
15
52
  const handleError = (c, reason, logMessage) => {
16
53
  const errorDetails = reason instanceof Error
17
54
  ? { message: reason.message, stack: reason.stack, name: reason.name }
@@ -26,6 +63,17 @@ const handleError = (c, reason, logMessage) => {
26
63
  id: null,
27
64
  }, { status: 500 });
28
65
  };
66
+ /**
67
+ * MCPサーバーおよびトランスポートのリソースをクローズします。
68
+ *
69
+ * @remarks
70
+ * どちらか一方のクローズに失敗しても、もう一方は必ず実行されます。
71
+ *
72
+ * @param server MCPサーバーインスタンス
73
+ * @param transport トランスポートインスタンス
74
+ * @returns void
75
+ * @private
76
+ */
29
77
  const closeResources = async (server, transport) => {
30
78
  // 両方のクローズを確実に実行(片方が失敗してももう片方を実行)
31
79
  const closeResults = await Promise.allSettled([
@@ -44,36 +92,69 @@ const closeResources = async (server, transport) => {
44
92
  }
45
93
  });
46
94
  };
47
- const handleRequest = async (server, c) => {
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) => {
48
107
  const transport = new StreamableHTTPTransport({
49
108
  sessionIdGenerator: undefined, // セッションIDを生成しない(ステートレスモード)
50
109
  enableJsonResponse: true,
51
110
  });
111
+ const server = createMcpServer();
52
112
  try {
53
113
  await server.connect(transport);
54
114
  logger.trace('MCP リクエストを受信');
55
115
  return await transport.handleRequest(c);
56
116
  }
57
117
  catch (error) {
118
+ return handleError(c, error, 'MCP 接続中のエラー:');
119
+ }
120
+ finally {
121
+ // エラーの有無に関わらず必ずリソースをクローズ
58
122
  try {
59
123
  await closeResources(server, transport);
60
124
  }
61
125
  catch (closeError) {
126
+ // クローズエラーは既にcloseResources内でログ出力されているため、
127
+ // ここでは追加のエラーハンドリングは不要だが、エラーの詳細を記録
62
128
  const errorDetails = closeError instanceof Error
63
129
  ? { message: closeError.message, stack: closeError.stack }
64
130
  : closeError;
65
- logger.error('Transport close failed after connection error:', { closeError: errorDetails });
131
+ logger.error('リソースクローズ中に追加エラーが発生しましたが、処理を継続します', { closeError: errorDetails });
66
132
  }
67
- return handleError(c, error, 'MCP 接続中のエラー:');
68
133
  }
69
134
  };
70
- export const createHonoApp = (server) => {
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) => {
71
152
  const app = new Hono();
72
153
  app.post('/mcp', async (c) => {
73
- return await handleRequest(server, c);
154
+ return await handleRequest(createMcpServer, c);
74
155
  });
75
156
  app.get('/mcp', async (c) => {
76
- return await handleRequest(server, c);
157
+ return await handleRequest(createMcpServer, c);
77
158
  });
78
159
  app.put('/mcp', methodNotAllowedHandler);
79
160
  app.delete('/mcp', methodNotAllowedHandler);
@@ -81,4 +162,4 @@ export const createHonoApp = (server) => {
81
162
  app.options('/mcp', methodNotAllowedHandler);
82
163
  return app;
83
164
  };
84
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../src/index.ts"],"names":[],"mappings":"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","sourcesContent":["import { 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\nconst logger = new Logger();\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\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\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\nconst handleRequest = async (server: McpServer, c: Context<BlankEnv, '/mcp', BlankInput>) => {\n  const transport = new StreamableHTTPTransport({\n    sessionIdGenerator: undefined, // セッションIDを生成しない（ステートレスモード）\n    enableJsonResponse: true,\n  });\n  try {\n    await server.connect(transport);\n    logger.trace('MCP リクエストを受信');\n    return await transport.handleRequest(c);\n  } catch (error) {\n    try {\n      await closeResources(server, transport);\n    } catch (closeError) {\n      const errorDetails = closeError instanceof Error\n        ? { message: closeError.message, stack: closeError.stack }\n        : closeError;\n      logger.error('Transport close failed after connection error:', { closeError: errorDetails });\n    }\n    return handleError(c, error, 'MCP 接続中のエラー:');\n  }\n\n};\n\nexport const createHonoApp = (server: McpServer) => {\n  const app = new Hono();\n\n  app.post('/mcp', async (c) => {\n    return await handleRequest(server, c);\n  });\n\n  app.get('/mcp', async (c) => {\n    return await handleRequest(server, 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"]}
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.2",
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": [
@@ -34,25 +34,25 @@
34
34
  "license": "ISC",
35
35
  "devDependencies": {
36
36
  "@eslint/compat": "^1.4.1",
37
- "@eslint/js": "^9.39.0",
37
+ "@eslint/js": "^9.39.1",
38
38
  "@stylistic/eslint-plugin": "^5.5.0",
39
- "@types/node": "24.9.2",
40
- "eslint": "^9.39.0",
39
+ "@types/node": "24.10.0",
40
+ "eslint": "^9.39.1",
41
41
  "eslint-import-resolver-typescript": "^4.4.4",
42
42
  "eslint-plugin-import": "^2.32.0",
43
43
  "eslint-plugin-promise": "^7.2.1",
44
44
  "jiti": "^2.6.1",
45
45
  "tsx": "^4.20.6",
46
46
  "typescript": "^5.9.3",
47
- "typescript-eslint": "^8.46.2",
48
- "vitest": "^4.0.6"
47
+ "typescript-eslint": "^8.46.3",
48
+ "vitest": "^4.0.7"
49
49
  },
50
50
  "dependencies": {
51
51
  "@aws-lambda-powertools/logger": "^2.28.1",
52
52
  "@hono/mcp": "^0.1.5",
53
- "@modelcontextprotocol/sdk": "^1.20.2",
53
+ "@modelcontextprotocol/sdk": "^1.21.0",
54
54
  "hono": "^4.10.4",
55
- "zod": "^3.25.76"
55
+ "zod": "^4.1.12"
56
56
  },
57
57
  "scripts": {
58
58
  "build": "tsc",