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 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);
@@ -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.1",
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.0",
37
+ "@eslint/js": "^9.39.1",
33
38
  "@stylistic/eslint-plugin": "^5.5.0",
34
- "@types/node": "24.9.2",
35
- "eslint": "^9.39.0",
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.2",
43
- "vitest": "^4.0.6"
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.20.2",
53
+ "@modelcontextprotocol/sdk": "^1.21.0",
49
54
  "hono": "^4.10.4",
50
- "zod": "^3.25.76"
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
- };
@@ -1,15 +0,0 @@
1
- {
2
- "extends": [
3
- "./tsconfig.json"
4
- ],
5
- "compilerOptions": {
6
- "noEmit": true
7
- },
8
- "include": [
9
- "src",
10
- "*.config.ts"
11
- ],
12
- "exclude": [
13
- "node_modules"
14
- ]
15
- }
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
- }