@kajidog/connpass-mcp-server 0.3.0 → 0.4.1

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 (62) hide show
  1. package/README.md +72 -149
  2. package/dist/index.d.ts +0 -2
  3. package/dist/index.js +4749 -162
  4. package/dist/index.js.map +1 -1
  5. package/dist/mcp-app.html +139 -0
  6. package/dist/stdio.d.ts +1 -0
  7. package/dist/stdio.js +2140 -0
  8. package/dist/stdio.js.map +1 -0
  9. package/package.json +18 -18
  10. package/dist/apps-sdk.d.ts +0 -8
  11. package/dist/apps-sdk.d.ts.map +0 -1
  12. package/dist/apps-sdk.js +0 -93
  13. package/dist/apps-sdk.js.map +0 -1
  14. package/dist/config.d.ts +0 -7
  15. package/dist/config.d.ts.map +0 -1
  16. package/dist/config.js +0 -82
  17. package/dist/config.js.map +0 -1
  18. package/dist/index.d.ts.map +0 -1
  19. package/dist/tools/events.d.ts +0 -74
  20. package/dist/tools/events.d.ts.map +0 -1
  21. package/dist/tools/events.js +0 -413
  22. package/dist/tools/events.js.map +0 -1
  23. package/dist/tools/formatting.d.ts +0 -74
  24. package/dist/tools/formatting.d.ts.map +0 -1
  25. package/dist/tools/formatting.js +0 -222
  26. package/dist/tools/formatting.js.map +0 -1
  27. package/dist/tools/groups.d.ts +0 -48
  28. package/dist/tools/groups.d.ts.map +0 -1
  29. package/dist/tools/groups.js +0 -106
  30. package/dist/tools/groups.js.map +0 -1
  31. package/dist/tools/index.d.ts +0 -54
  32. package/dist/tools/index.d.ts.map +0 -1
  33. package/dist/tools/index.js +0 -21
  34. package/dist/tools/index.js.map +0 -1
  35. package/dist/tools/shared.d.ts +0 -24
  36. package/dist/tools/shared.d.ts.map +0 -1
  37. package/dist/tools/shared.js +0 -113
  38. package/dist/tools/shared.js.map +0 -1
  39. package/dist/tools/users.d.ts +0 -51
  40. package/dist/tools/users.d.ts.map +0 -1
  41. package/dist/tools/users.js +0 -239
  42. package/dist/tools/users.js.map +0 -1
  43. package/dist/transports/http.d.ts +0 -17
  44. package/dist/transports/http.d.ts.map +0 -1
  45. package/dist/transports/http.js +0 -185
  46. package/dist/transports/http.js.map +0 -1
  47. package/dist/transports/sse.d.ts +0 -21
  48. package/dist/transports/sse.d.ts.map +0 -1
  49. package/dist/transports/sse.js +0 -161
  50. package/dist/transports/sse.js.map +0 -1
  51. package/dist/widgets/connpass-events.d.ts +0 -41
  52. package/dist/widgets/connpass-events.d.ts.map +0 -1
  53. package/dist/widgets/connpass-events.js +0 -68
  54. package/dist/widgets/connpass-events.js.map +0 -1
  55. package/dist/widgets/connpass-schedule.d.ts +0 -41
  56. package/dist/widgets/connpass-schedule.d.ts.map +0 -1
  57. package/dist/widgets/connpass-schedule.js +0 -68
  58. package/dist/widgets/connpass-schedule.js.map +0 -1
  59. package/dist/widgets/index.d.ts +0 -6
  60. package/dist/widgets/index.d.ts.map +0 -1
  61. package/dist/widgets/index.js +0 -39
  62. package/dist/widgets/index.js.map +0 -1
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../../../packages/mcp-core/src/config-schema.ts","../../../packages/mcp-core/src/config.ts","../../../packages/mcp-core/src/http.ts","../../../node_modules/.pnpm/hono@4.11.3/node_modules/hono/dist/request/constants.js","../../../node_modules/.pnpm/hono@4.11.3/node_modules/hono/dist/utils/body.js","../../../node_modules/.pnpm/hono@4.11.3/node_modules/hono/dist/utils/url.js","../../../node_modules/.pnpm/hono@4.11.3/node_modules/hono/dist/request.js","../../../node_modules/.pnpm/hono@4.11.3/node_modules/hono/dist/router/reg-exp-router/node.js","../../../packages/mcp-core/src/stdio.ts","../src/server.ts","../src/config.ts","../src/tools/events.ts","../src/tools/prefectures.ts","../src/tools/utils/registration.ts","../src/tools/utils/formatting.ts","../src/tools/utils/resource.ts","../src/tools/utils/shared.ts","../src/tools/groups.ts","../src/tools/schedule.ts","../src/tools/ui-tools/detail.ts","../src/tools/ui-tools/schedule.ts","../src/tools/ui-tools/search.ts","../src/tools/ui-tools/index.ts","../src/tools/users.ts","../src/tools/index.ts","../src/tools/utils/searchSessionStore.ts","../src/stdio.ts"],"sourcesContent":["/**\n * 宣言的な設定スキーマヘルパー\n *\n * 設定オプションをメタデータ付きで定義し、\n * CLI引数パース・環境変数パース・help文生成を自動化する。\n */\n\nimport { z } from \"zod/v4\";\n\n// オプションの型\ntype OptionType = \"string\" | \"number\" | \"boolean\" | \"string[]\";\n\n/** 設定オプション定義 */\nexport interface OptionDef {\n /** CLIフラグ名 (例: \"--port\") */\n cli: string;\n /** 環境変数名 (例: \"MCP_HTTP_PORT\") */\n env?: string;\n /** help表示用の説明文 */\n description: string;\n /** helpのグループ名 (例: \"Server Options\") */\n group: string;\n /** オプションの型 */\n type: OptionType;\n /** デフォルト値 */\n default?: unknown;\n /** CLIで値を取る引数の表示名 (例: \"<port>\") */\n valueName?: string;\n}\n\n/** 設定定義のマップ */\nexport type ConfigDefs = Record<string, OptionDef>;\n\n/**\n * CLI引数を設定定義からパースする\n */\nexport function parseCliFromDefs(\n defs: ConfigDefs,\n argv: string[],\n): Record<string, unknown> {\n const config: Record<string, unknown> = {};\n\n // CLIフラグ → 設定キーのマッピングを構築\n const flagMap = new Map<string, { key: string; def: OptionDef }>();\n const negationMap = new Map<string, string>();\n\n for (const [key, def] of Object.entries(defs)) {\n flagMap.set(def.cli, { key, def });\n\n // boolean型は自動で --no-* フラグを生成\n if (def.type === \"boolean\") {\n const baseName = def.cli.replace(/^--/, \"\");\n negationMap.set(`--no-${baseName}`, key);\n }\n }\n\n for (let i = 0; i < argv.length; i++) {\n const arg = argv[i];\n const nextArg = argv[i + 1];\n\n // --no-* 否定フラグ(booleanのみ)\n const negKey = negationMap.get(arg);\n if (negKey !== undefined) {\n config[negKey] = false;\n continue;\n }\n\n const entry = flagMap.get(arg);\n if (!entry) continue;\n\n const { key, def } = entry;\n\n switch (def.type) {\n case \"boolean\":\n config[key] = true;\n break;\n\n case \"string\":\n if (nextArg && !nextArg.startsWith(\"-\")) {\n config[key] = nextArg;\n i++;\n }\n break;\n\n case \"number\":\n if (nextArg && !nextArg.startsWith(\"-\")) {\n const num = Number(nextArg);\n if (Number.isFinite(num)) {\n config[key] = num;\n }\n i++;\n }\n break;\n\n case \"string[]\":\n if (nextArg && !nextArg.startsWith(\"-\")) {\n config[key] = nextArg.split(\",\").map((s) => s.trim());\n i++;\n }\n break;\n }\n }\n\n return config;\n}\n\n/**\n * 環境変数を設定定義からパースする\n */\nexport function parseEnvFromDefs(\n defs: ConfigDefs,\n env: Record<string, string | undefined>,\n): Record<string, unknown> {\n const config: Record<string, unknown> = {};\n\n for (const [key, def] of Object.entries(defs)) {\n if (!def.env) continue;\n const val = env[def.env];\n if (val === undefined) continue;\n\n switch (def.type) {\n case \"boolean\":\n // 統一: 'true' → true, 'false' → false, それ以外は設定しない\n if (val === \"true\") config[key] = true;\n else if (val === \"false\") config[key] = false;\n break;\n\n case \"number\": {\n if (val === \"\") break;\n const num = Number(val);\n if (Number.isFinite(num)) config[key] = num;\n break;\n }\n\n case \"string\":\n if (val) config[key] = val;\n break;\n\n case \"string[]\":\n if (val) config[key] = val.split(\",\").map((s) => s.trim());\n break;\n }\n }\n\n return config;\n}\n\n/**\n * 設定ファイルの内容を設定定義のキー名に正規化する\n *\n * 設定ファイルではCLIフラグ名のキー(\"--\" なし、kebab-case → camelCase)を使えるようにする。\n * 例: { \"speaker\": 3, \"use-streaming\": true } → { defaultSpeaker: 3, useStreaming: true }\n */\nexport function parseConfigFileFromDefs(\n defs: ConfigDefs,\n fileContent: Record<string, unknown>,\n): Record<string, unknown> {\n const config: Record<string, unknown> = {};\n\n // CLIフラグ名(-- なし)→ 設定キーのマッピングを構築\n const cliNameMap = new Map<string, { key: string; def: OptionDef }>();\n for (const [key, def] of Object.entries(defs)) {\n const cliName = def.cli.replace(/^--/, \"\");\n cliNameMap.set(cliName, { key, def });\n // camelCase バージョンも登録\n const camelCase = cliName.replace(/-([a-z])/g, (_, c) => c.toUpperCase());\n if (camelCase !== cliName) {\n cliNameMap.set(camelCase, { key, def });\n }\n // 設定キー名そのままも登録\n cliNameMap.set(key, { key, def });\n }\n\n for (const [fileKey, val] of Object.entries(fileContent)) {\n const entry = cliNameMap.get(fileKey);\n if (!entry) continue;\n\n const { key, def } = entry;\n\n switch (def.type) {\n case \"boolean\":\n if (typeof val === \"boolean\") config[key] = val;\n break;\n case \"number\":\n if (typeof val === \"number\" && Number.isFinite(val)) config[key] = val;\n break;\n case \"string\":\n if (typeof val === \"string\" && val !== \"\") config[key] = val;\n break;\n case \"string[]\":\n if (Array.isArray(val))\n config[key] = val.filter((v): v is string => typeof v === \"string\");\n else if (typeof val === \"string\")\n config[key] = val.split(\",\").map((s) => s.trim());\n break;\n }\n }\n\n return config;\n}\n\n/**\n * 設定定義からデフォルト値を抽出する\n */\nexport function getDefaultsFromDefs(defs: ConfigDefs): Record<string, unknown> {\n const defaults: Record<string, unknown> = {};\n for (const [key, def] of Object.entries(defs)) {\n if (def.default !== undefined) {\n defaults[key] = def.default;\n }\n }\n return defaults;\n}\n\n/**\n * 設定定義からhelp文を生成する\n */\nexport function generateHelp(\n defs: ConfigDefs,\n opts?: { usage?: string; examples?: string[] },\n): string {\n const lines: string[] = [];\n\n if (opts?.usage) {\n lines.push(`Usage: ${opts.usage}`);\n lines.push(\"\");\n }\n\n lines.push(\"Options:\");\n lines.push(\" --help, -h Show this help message\");\n lines.push(\" --version, -v Show version number\");\n lines.push(\n \" --init Generate .voicevoxrc.json with default settings\",\n );\n lines.push(\"\");\n\n // グループごとに整理\n const groups = new Map<string, { key: string; def: OptionDef }[]>();\n for (const [key, def] of Object.entries(defs)) {\n const group = def.group;\n if (!groups.has(group)) groups.set(group, []);\n groups.get(group)!.push({ key, def });\n }\n\n for (const [groupName, entries] of groups) {\n lines.push(` ${groupName}:`);\n\n for (const { def } of entries) {\n const flag =\n def.type === \"boolean\"\n ? def.cli\n : `${def.cli} ${def.valueName || \"<value>\"}`;\n\n const defaultStr =\n def.default !== undefined\n ? ` (default: ${Array.isArray(def.default) ? def.default.join(\",\") : def.default})`\n : \"\";\n\n const line = ` ${flag.padEnd(28)}${def.description}${defaultStr}`;\n lines.push(line);\n\n // boolean型は --no-* も表示\n if (def.type === \"boolean\") {\n const baseName = def.cli.replace(/^--/, \"\");\n const negFlag = `--no-${baseName}`;\n lines.push(\n ` ${negFlag.padEnd(28)}Disable ${def.description.toLowerCase().replace(/^enable /, \"\")}`,\n );\n }\n }\n\n lines.push(\"\");\n }\n\n if (opts?.examples?.length) {\n lines.push(\"Examples:\");\n for (const ex of opts.examples) {\n lines.push(` ${ex}`);\n }\n lines.push(\"\");\n }\n\n return lines.join(\"\\n\");\n}\n\n/**\n * Zodスキーマで設定値をバリデーションする\n */\nexport function validateConfig<T>(\n schema: z.ZodType<T>,\n config: Record<string, unknown>,\n): T {\n return schema.parse(config);\n}\n\n/**\n * 設定定義から設定ファイルのテンプレートJSONを生成する\n *\n * CLIフラグ名(-- なし)をキーとして、デフォルト値と説明コメントを含む。\n * 内部用のオプション(configFile等)は除外する。\n */\nexport function generateConfigTemplate(\n defs: ConfigDefs,\n opts?: { exclude?: string[] },\n): Record<string, unknown> {\n const template: Record<string, unknown> = {};\n const excludeSet = new Set(opts?.exclude ?? []);\n\n for (const [key, def] of Object.entries(defs)) {\n if (excludeSet.has(key)) continue;\n // default が未定義のオプションはテンプレートに含めない\n // (ランタイムデフォルトを持つパスオプション等を空値で上書きしないため)\n if (def.default === undefined) continue;\n\n const cliName = def.cli.replace(/^--/, \"\");\n template[cliName] = def.default;\n }\n\n return template;\n}\n\n/**\n * undefinedのプロパティを除去する\n */\nexport function filterUndefined<T extends object>(obj: T): Partial<T> {\n return Object.fromEntries(\n Object.entries(obj).filter(([_, v]) => v !== undefined),\n ) as Partial<T>;\n}\n","/**\n * MCP Core 基本設定モジュール\n *\n * HTTP/Stdioサーバーの共通設定を管理\n * 優先順位: CLI引数 > 環境変数 > デフォルト値\n */\n\nimport {\n type ConfigDefs,\n filterUndefined,\n getDefaultsFromDefs,\n parseCliFromDefs,\n parseEnvFromDefs,\n} from \"./config-schema.js\";\n\n// 基本設定の宣言的定義\nexport const baseConfigDefs: ConfigDefs = {\n httpMode: {\n cli: \"--http\",\n env: \"MCP_HTTP_MODE\",\n description: \"Enable HTTP server mode\",\n group: \"Server Options\",\n type: \"boolean\",\n default: false,\n },\n httpPort: {\n cli: \"--port\",\n env: \"MCP_HTTP_PORT\",\n description: \"HTTP server port\",\n group: \"Server Options\",\n type: \"number\",\n default: 3000,\n valueName: \"<port>\",\n },\n httpHost: {\n cli: \"--host\",\n env: \"MCP_HTTP_HOST\",\n description: \"HTTP server host\",\n group: \"Server Options\",\n type: \"string\",\n default: \"0.0.0.0\",\n valueName: \"<host>\",\n },\n allowedHosts: {\n cli: \"--allowed-hosts\",\n env: \"MCP_ALLOWED_HOSTS\",\n description: \"Comma-separated list of allowed hosts\",\n group: \"Server Options\",\n type: \"string[]\",\n default: [\"localhost\", \"127.0.0.1\", \"[::1]\"],\n valueName: \"<hosts>\",\n },\n allowedOrigins: {\n cli: \"--allowed-origins\",\n env: \"MCP_ALLOWED_ORIGINS\",\n description: \"Comma-separated list of allowed origins\",\n group: \"Server Options\",\n type: \"string[]\",\n default: [\n \"http://localhost\",\n \"http://127.0.0.1\",\n \"https://localhost\",\n \"https://127.0.0.1\",\n ],\n valueName: \"<origins>\",\n },\n apiKey: {\n cli: \"--api-key\",\n env: \"MCP_API_KEY\",\n description:\n \"Require matching API key via X-API-Key or Authorization: Bearer\",\n group: \"Server Options\",\n type: \"string\",\n valueName: \"<key>\",\n },\n};\n\n// 基本設定型定義(HTTP/サーバー関連のみ)\nexport interface BaseServerConfig {\n httpMode: boolean;\n httpPort: number;\n httpHost: string;\n allowedHosts: string[];\n allowedOrigins: string[];\n apiKey?: string;\n}\n\n// デフォルト設定\nexport const defaultBaseConfig = getDefaultsFromDefs(\n baseConfigDefs,\n) as unknown as BaseServerConfig;\n\n/**\n * CLI引数から基本設定をパースする\n */\nexport function parseBaseCliArgs(\n argv: string[] = process.argv.slice(2),\n): Partial<BaseServerConfig> {\n return parseCliFromDefs(baseConfigDefs, argv) as Partial<BaseServerConfig>;\n}\n\n/**\n * 環境変数から基本設定を読み込む\n */\nexport function parseBaseEnvVars(\n env: NodeJS.ProcessEnv = process.env,\n): Partial<BaseServerConfig> {\n return parseEnvFromDefs(baseConfigDefs, env) as Partial<BaseServerConfig>;\n}\n\nexport { filterUndefined };\n","import { randomUUID } from \"node:crypto\";\nimport type { McpServer } from \"@modelcontextprotocol/sdk/server/mcp.js\";\nimport { WebStandardStreamableHTTPServerTransport } from \"@modelcontextprotocol/sdk/server/webStandardStreamableHttp.js\";\nimport { isInitializeRequest } from \"@modelcontextprotocol/sdk/types.js\";\nimport { type Context, Hono, type Next } from \"hono\";\nimport { cors } from \"hono/cors\";\n\nimport type { BaseServerConfig } from \"./config.js\";\nimport { deleteSessionConfig } from \"./session.js\";\n\n// 型定義\ninterface ErrorResponse {\n jsonrpc: \"2.0\";\n error: {\n code: number;\n message: string;\n };\n id: null;\n}\n\ninterface HealthCheckResponse {\n status: \"ok\";\n transports: number;\n timestamp: string;\n}\n\nexport interface CreateHttpAppOptions {\n server: McpServer;\n config: BaseServerConfig;\n /** セッションごとに新しい McpServer を生成するファクトリ関数(HTTPモード用) */\n serverFactory?: () => McpServer;\n /** 追加のCORSヘッダー(例: 'X-Voicevox-Speaker') */\n extraCorsHeaders?: string[];\n /** セッション初期化時のコールバック(ヘッダーからの設定読み取り等に使用) */\n onSessionInitialized?: (sessionId: string, request: Request) => void;\n /** セッション終了時のコールバック */\n onSessionClosed?: (sessionId: string) => void;\n}\n\n/**\n * JSONRPCエラーレスポンスを生成するヘルパー関数\n */\nfunction badRequestError(\n message = \"Bad Request: No valid session ID provided\",\n): ErrorResponse {\n return {\n jsonrpc: \"2.0\",\n error: { code: -32000, message },\n id: null,\n };\n}\n\nfunction internalServerError(): ErrorResponse {\n return {\n jsonrpc: \"2.0\",\n error: { code: -32603, message: \"Internal server error\" },\n id: null,\n };\n}\n\nfunction forbiddenError(message: string): ErrorResponse {\n return {\n jsonrpc: \"2.0\",\n error: { code: -32000, message },\n id: null,\n };\n}\n\nfunction unauthorizedError(message: string): ErrorResponse {\n return {\n jsonrpc: \"2.0\",\n error: { code: -32001, message },\n id: null,\n };\n}\n\nfunction extractHostnameFromHostHeader(host: string): string {\n const trimmedHost = host.trim();\n\n if (trimmedHost.startsWith(\"[\")) {\n const bracketEnd = trimmedHost.indexOf(\"]\");\n if (bracketEnd !== -1) {\n return trimmedHost.slice(0, bracketEnd + 1);\n }\n }\n\n const colonIndex = trimmedHost.indexOf(\":\");\n if (colonIndex === -1) {\n return trimmedHost;\n }\n\n return trimmedHost.slice(0, colonIndex);\n}\n\n/**\n * Origin検証ミドルウェア\n */\nfunction validateOrigin(config: BaseServerConfig) {\n return async (c: Context, next: Next) => {\n const origin = c.req.header(\"Origin\");\n\n if (!origin) {\n return next();\n }\n\n try {\n const originUrl = new URL(origin);\n const originWithoutPort = `${originUrl.protocol}//${originUrl.hostname}`;\n\n const isAllowed = config.allowedOrigins.some((allowed) => {\n try {\n const allowedUrl = new URL(allowed);\n return (\n originWithoutPort ===\n `${allowedUrl.protocol}//${allowedUrl.hostname}`\n );\n } catch {\n return false;\n }\n });\n\n if (!isAllowed) {\n console.log(\n `Rejected request with invalid Origin: ${origin} (allowed: ${config.allowedOrigins.join(\", \")})`,\n );\n return c.json(forbiddenError(\"Forbidden: Invalid Origin header\"), {\n status: 403,\n });\n }\n } catch {\n console.log(`Rejected request with malformed Origin: ${origin}`);\n return c.json(forbiddenError(\"Forbidden: Malformed Origin header\"), {\n status: 403,\n });\n }\n\n return next();\n };\n}\n\n/**\n * Host検証ミドルウェア\n */\nfunction validateHost(config: BaseServerConfig) {\n return async (c: Context, next: Next) => {\n const host = c.req.header(\"Host\");\n\n if (!host) {\n return next();\n }\n\n const hostname = extractHostnameFromHostHeader(host);\n\n if (!config.allowedHosts.includes(hostname)) {\n console.log(\n `Rejected request with invalid Host: ${host} (allowed: ${config.allowedHosts.join(\", \")})`,\n );\n return c.json(forbiddenError(\"Forbidden: Invalid Host header\"), {\n status: 403,\n });\n }\n\n return next();\n };\n}\n\n/**\n * APIキー検証ミドルウェア\n */\nfunction validateApiKey(config: BaseServerConfig) {\n return async (c: Context, next: Next) => {\n if (!config.apiKey || c.req.method === \"OPTIONS\") {\n return next();\n }\n\n const xApiKey = c.req.header(\"X-API-Key\");\n const authorization = c.req.header(\"Authorization\");\n const bearerToken = authorization?.startsWith(\"Bearer \")\n ? authorization.slice(7).trim()\n : undefined;\n const providedKey = xApiKey ?? bearerToken;\n\n if (providedKey !== config.apiKey) {\n console.log(\"Rejected request with invalid API key\");\n return c.json(unauthorizedError(\"Unauthorized: Invalid API key\"), {\n status: 401,\n });\n }\n\n return next();\n };\n}\n\n/**\n * MCP HTTP アプリケーションを作成する\n *\n * @param options - HTTPアプリの設定オプション\n * @returns 設定済みのHonoアプリケーション\n */\nexport function createHttpApp(options: CreateHttpAppOptions): Hono {\n const {\n server,\n config,\n serverFactory,\n extraCorsHeaders = [],\n onSessionInitialized,\n onSessionClosed,\n } = options;\n\n // セッションごとのtransportを管理\n const transports: Map<string, WebStandardStreamableHTTPServerTransport> =\n new Map();\n\n /**\n * MCP エンドポイントハンドラー\n */\n async function handleMCP(c: Context): Promise<Response> {\n console.log(`Received ${c.req.method} request for MCP`);\n\n const sessionId = c.req.header(\"mcp-session-id\");\n\n try {\n // 既存セッションの再利用\n if (sessionId && transports.has(sessionId)) {\n console.log(`Reusing existing session: ${sessionId}`);\n const transport = transports.get(sessionId)!;\n return transport.handleRequest(c.req.raw);\n }\n\n // 新しいセッションの初期化(POSTリクエストのみ)\n if (c.req.method === \"POST\") {\n let body: unknown;\n try {\n body = await c.req.json();\n } catch {\n return c.json(badRequestError(\"Invalid JSON\"), { status: 400 });\n }\n\n // initializeリクエストの場合のみ新しいtransportを作成\n if (isInitializeRequest(body)) {\n console.log(\"Creating new WebStandard session\");\n\n // コールバック用にリクエストを保持\n const rawRequest = c.req.raw;\n\n const transport = new WebStandardStreamableHTTPServerTransport({\n sessionIdGenerator: () => randomUUID(),\n onsessioninitialized: (newSessionId) => {\n console.log(`Session initialized: ${newSessionId}`);\n transports.set(newSessionId, transport);\n\n // アプリ固有の初期化処理\n onSessionInitialized?.(newSessionId, rawRequest);\n },\n });\n\n // クリーンアップハンドラー\n transport.onclose = () => {\n const sid = transport.sessionId;\n if (sid) {\n console.log(`Transport closed for session: ${sid}`);\n transports.delete(sid);\n deleteSessionConfig(sid);\n\n // アプリ固有のクリーンアップ処理\n onSessionClosed?.(sid);\n }\n };\n\n // セッションごとに新しいサーバーインスタンスを使用\n const sessionServer = serverFactory ? serverFactory() : server;\n await sessionServer.connect(transport);\n\n // リクエスト処理(parsedBodyを渡す)\n return transport.handleRequest(c.req.raw, { parsedBody: body });\n }\n }\n\n // セッションIDがなく、initializeリクエストでもない場合\n console.log(\n \"Invalid request - no session ID and not an initialize request\",\n );\n return c.json(badRequestError(), { status: 400 });\n } catch (e) {\n console.error(\"MCP connection error:\", e);\n return c.json(internalServerError(), { status: 500 });\n }\n }\n\n /**\n * ヘルスチェックエンドポイントハンドラー\n */\n function handleHealth(c: Context): Response {\n const response: HealthCheckResponse = {\n status: \"ok\",\n transports: transports.size,\n timestamp: new Date().toISOString(),\n };\n return c.json(response);\n }\n\n // アプリケーションのセットアップ\n const app: Hono = new Hono();\n\n // CORSを設定\n const allowHeaders = [\n \"Content-Type\",\n \"mcp-session-id\",\n \"Last-Event-ID\",\n \"mcp-protocol-version\",\n \"X-API-Key\",\n \"Authorization\",\n ...extraCorsHeaders,\n ];\n\n app.use(\n \"/mcp\",\n cors({\n origin: \"*\",\n allowMethods: [\"GET\", \"POST\", \"DELETE\", \"OPTIONS\"],\n allowHeaders,\n exposeHeaders: [\"mcp-session-id\", \"mcp-protocol-version\"],\n }),\n );\n\n // セキュリティミドルウェアを適用\n app.use(\"/mcp\", validateOrigin(config));\n app.use(\"/mcp\", validateHost(config));\n app.use(\"/mcp\", validateApiKey(config));\n\n // ルート定義\n app.all(\"/mcp\", handleMCP);\n app.get(\"/health\", handleHealth);\n\n return app;\n}\n","// src/request/constants.ts\nvar GET_MATCH_RESULT = /* @__PURE__ */ Symbol();\nexport {\n GET_MATCH_RESULT\n};\n","// src/utils/body.ts\nimport { HonoRequest } from \"../request.js\";\nvar parseBody = async (request, options = /* @__PURE__ */ Object.create(null)) => {\n const { all = false, dot = false } = options;\n const headers = request instanceof HonoRequest ? request.raw.headers : request.headers;\n const contentType = headers.get(\"Content-Type\");\n if (contentType?.startsWith(\"multipart/form-data\") || contentType?.startsWith(\"application/x-www-form-urlencoded\")) {\n return parseFormData(request, { all, dot });\n }\n return {};\n};\nasync function parseFormData(request, options) {\n const formData = await request.formData();\n if (formData) {\n return convertFormDataToBodyData(formData, options);\n }\n return {};\n}\nfunction convertFormDataToBodyData(formData, options) {\n const form = /* @__PURE__ */ Object.create(null);\n formData.forEach((value, key) => {\n const shouldParseAllValues = options.all || key.endsWith(\"[]\");\n if (!shouldParseAllValues) {\n form[key] = value;\n } else {\n handleParsingAllValues(form, key, value);\n }\n });\n if (options.dot) {\n Object.entries(form).forEach(([key, value]) => {\n const shouldParseDotValues = key.includes(\".\");\n if (shouldParseDotValues) {\n handleParsingNestedValues(form, key, value);\n delete form[key];\n }\n });\n }\n return form;\n}\nvar handleParsingAllValues = (form, key, value) => {\n if (form[key] !== void 0) {\n if (Array.isArray(form[key])) {\n ;\n form[key].push(value);\n } else {\n form[key] = [form[key], value];\n }\n } else {\n if (!key.endsWith(\"[]\")) {\n form[key] = value;\n } else {\n form[key] = [value];\n }\n }\n};\nvar handleParsingNestedValues = (form, key, value) => {\n let nestedForm = form;\n const keys = key.split(\".\");\n keys.forEach((key2, index) => {\n if (index === keys.length - 1) {\n nestedForm[key2] = value;\n } else {\n if (!nestedForm[key2] || typeof nestedForm[key2] !== \"object\" || Array.isArray(nestedForm[key2]) || nestedForm[key2] instanceof File) {\n nestedForm[key2] = /* @__PURE__ */ Object.create(null);\n }\n nestedForm = nestedForm[key2];\n }\n });\n};\nexport {\n parseBody\n};\n","// src/utils/url.ts\nvar splitPath = (path) => {\n const paths = path.split(\"/\");\n if (paths[0] === \"\") {\n paths.shift();\n }\n return paths;\n};\nvar splitRoutingPath = (routePath) => {\n const { groups, path } = extractGroupsFromPath(routePath);\n const paths = splitPath(path);\n return replaceGroupMarks(paths, groups);\n};\nvar extractGroupsFromPath = (path) => {\n const groups = [];\n path = path.replace(/\\{[^}]+\\}/g, (match, index) => {\n const mark = `@${index}`;\n groups.push([mark, match]);\n return mark;\n });\n return { groups, path };\n};\nvar replaceGroupMarks = (paths, groups) => {\n for (let i = groups.length - 1; i >= 0; i--) {\n const [mark] = groups[i];\n for (let j = paths.length - 1; j >= 0; j--) {\n if (paths[j].includes(mark)) {\n paths[j] = paths[j].replace(mark, groups[i][1]);\n break;\n }\n }\n }\n return paths;\n};\nvar patternCache = {};\nvar getPattern = (label, next) => {\n if (label === \"*\") {\n return \"*\";\n }\n const match = label.match(/^\\:([^\\{\\}]+)(?:\\{(.+)\\})?$/);\n if (match) {\n const cacheKey = `${label}#${next}`;\n if (!patternCache[cacheKey]) {\n if (match[2]) {\n patternCache[cacheKey] = next && next[0] !== \":\" && next[0] !== \"*\" ? [cacheKey, match[1], new RegExp(`^${match[2]}(?=/${next})`)] : [label, match[1], new RegExp(`^${match[2]}$`)];\n } else {\n patternCache[cacheKey] = [label, match[1], true];\n }\n }\n return patternCache[cacheKey];\n }\n return null;\n};\nvar tryDecode = (str, decoder) => {\n try {\n return decoder(str);\n } catch {\n return str.replace(/(?:%[0-9A-Fa-f]{2})+/g, (match) => {\n try {\n return decoder(match);\n } catch {\n return match;\n }\n });\n }\n};\nvar tryDecodeURI = (str) => tryDecode(str, decodeURI);\nvar getPath = (request) => {\n const url = request.url;\n const start = url.indexOf(\"/\", url.indexOf(\":\") + 4);\n let i = start;\n for (; i < url.length; i++) {\n const charCode = url.charCodeAt(i);\n if (charCode === 37) {\n const queryIndex = url.indexOf(\"?\", i);\n const path = url.slice(start, queryIndex === -1 ? void 0 : queryIndex);\n return tryDecodeURI(path.includes(\"%25\") ? path.replace(/%25/g, \"%2525\") : path);\n } else if (charCode === 63) {\n break;\n }\n }\n return url.slice(start, i);\n};\nvar getQueryStrings = (url) => {\n const queryIndex = url.indexOf(\"?\", 8);\n return queryIndex === -1 ? \"\" : \"?\" + url.slice(queryIndex + 1);\n};\nvar getPathNoStrict = (request) => {\n const result = getPath(request);\n return result.length > 1 && result.at(-1) === \"/\" ? result.slice(0, -1) : result;\n};\nvar mergePath = (base, sub, ...rest) => {\n if (rest.length) {\n sub = mergePath(sub, ...rest);\n }\n return `${base?.[0] === \"/\" ? \"\" : \"/\"}${base}${sub === \"/\" ? \"\" : `${base?.at(-1) === \"/\" ? \"\" : \"/\"}${sub?.[0] === \"/\" ? sub.slice(1) : sub}`}`;\n};\nvar checkOptionalParameter = (path) => {\n if (path.charCodeAt(path.length - 1) !== 63 || !path.includes(\":\")) {\n return null;\n }\n const segments = path.split(\"/\");\n const results = [];\n let basePath = \"\";\n segments.forEach((segment) => {\n if (segment !== \"\" && !/\\:/.test(segment)) {\n basePath += \"/\" + segment;\n } else if (/\\:/.test(segment)) {\n if (/\\?/.test(segment)) {\n if (results.length === 0 && basePath === \"\") {\n results.push(\"/\");\n } else {\n results.push(basePath);\n }\n const optionalSegment = segment.replace(\"?\", \"\");\n basePath += \"/\" + optionalSegment;\n results.push(basePath);\n } else {\n basePath += \"/\" + segment;\n }\n }\n });\n return results.filter((v, i, a) => a.indexOf(v) === i);\n};\nvar _decodeURI = (value) => {\n if (!/[%+]/.test(value)) {\n return value;\n }\n if (value.indexOf(\"+\") !== -1) {\n value = value.replace(/\\+/g, \" \");\n }\n return value.indexOf(\"%\") !== -1 ? tryDecode(value, decodeURIComponent_) : value;\n};\nvar _getQueryParam = (url, key, multiple) => {\n let encoded;\n if (!multiple && key && !/[%+]/.test(key)) {\n let keyIndex2 = url.indexOf(\"?\", 8);\n if (keyIndex2 === -1) {\n return void 0;\n }\n if (!url.startsWith(key, keyIndex2 + 1)) {\n keyIndex2 = url.indexOf(`&${key}`, keyIndex2 + 1);\n }\n while (keyIndex2 !== -1) {\n const trailingKeyCode = url.charCodeAt(keyIndex2 + key.length + 1);\n if (trailingKeyCode === 61) {\n const valueIndex = keyIndex2 + key.length + 2;\n const endIndex = url.indexOf(\"&\", valueIndex);\n return _decodeURI(url.slice(valueIndex, endIndex === -1 ? void 0 : endIndex));\n } else if (trailingKeyCode == 38 || isNaN(trailingKeyCode)) {\n return \"\";\n }\n keyIndex2 = url.indexOf(`&${key}`, keyIndex2 + 1);\n }\n encoded = /[%+]/.test(url);\n if (!encoded) {\n return void 0;\n }\n }\n const results = {};\n encoded ??= /[%+]/.test(url);\n let keyIndex = url.indexOf(\"?\", 8);\n while (keyIndex !== -1) {\n const nextKeyIndex = url.indexOf(\"&\", keyIndex + 1);\n let valueIndex = url.indexOf(\"=\", keyIndex);\n if (valueIndex > nextKeyIndex && nextKeyIndex !== -1) {\n valueIndex = -1;\n }\n let name = url.slice(\n keyIndex + 1,\n valueIndex === -1 ? nextKeyIndex === -1 ? void 0 : nextKeyIndex : valueIndex\n );\n if (encoded) {\n name = _decodeURI(name);\n }\n keyIndex = nextKeyIndex;\n if (name === \"\") {\n continue;\n }\n let value;\n if (valueIndex === -1) {\n value = \"\";\n } else {\n value = url.slice(valueIndex + 1, nextKeyIndex === -1 ? void 0 : nextKeyIndex);\n if (encoded) {\n value = _decodeURI(value);\n }\n }\n if (multiple) {\n if (!(results[name] && Array.isArray(results[name]))) {\n results[name] = [];\n }\n ;\n results[name].push(value);\n } else {\n results[name] ??= value;\n }\n }\n return key ? results[key] : results;\n};\nvar getQueryParam = _getQueryParam;\nvar getQueryParams = (url, key) => {\n return _getQueryParam(url, key, true);\n};\nvar decodeURIComponent_ = decodeURIComponent;\nexport {\n checkOptionalParameter,\n decodeURIComponent_,\n getPath,\n getPathNoStrict,\n getPattern,\n getQueryParam,\n getQueryParams,\n getQueryStrings,\n mergePath,\n splitPath,\n splitRoutingPath,\n tryDecode\n};\n","// src/request.ts\nimport { HTTPException } from \"./http-exception.js\";\nimport { GET_MATCH_RESULT } from \"./request/constants.js\";\nimport { parseBody } from \"./utils/body.js\";\nimport { decodeURIComponent_, getQueryParam, getQueryParams, tryDecode } from \"./utils/url.js\";\nvar tryDecodeURIComponent = (str) => tryDecode(str, decodeURIComponent_);\nvar HonoRequest = class {\n /**\n * `.raw` can get the raw Request object.\n *\n * @see {@link https://hono.dev/docs/api/request#raw}\n *\n * @example\n * ```ts\n * // For Cloudflare Workers\n * app.post('/', async (c) => {\n * const metadata = c.req.raw.cf?.hostMetadata?\n * ...\n * })\n * ```\n */\n raw;\n #validatedData;\n // Short name of validatedData\n #matchResult;\n routeIndex = 0;\n /**\n * `.path` can get the pathname of the request.\n *\n * @see {@link https://hono.dev/docs/api/request#path}\n *\n * @example\n * ```ts\n * app.get('/about/me', (c) => {\n * const pathname = c.req.path // `/about/me`\n * })\n * ```\n */\n path;\n bodyCache = {};\n constructor(request, path = \"/\", matchResult = [[]]) {\n this.raw = request;\n this.path = path;\n this.#matchResult = matchResult;\n this.#validatedData = {};\n }\n param(key) {\n return key ? this.#getDecodedParam(key) : this.#getAllDecodedParams();\n }\n #getDecodedParam(key) {\n const paramKey = this.#matchResult[0][this.routeIndex][1][key];\n const param = this.#getParamValue(paramKey);\n return param && /\\%/.test(param) ? tryDecodeURIComponent(param) : param;\n }\n #getAllDecodedParams() {\n const decoded = {};\n const keys = Object.keys(this.#matchResult[0][this.routeIndex][1]);\n for (const key of keys) {\n const value = this.#getParamValue(this.#matchResult[0][this.routeIndex][1][key]);\n if (value !== void 0) {\n decoded[key] = /\\%/.test(value) ? tryDecodeURIComponent(value) : value;\n }\n }\n return decoded;\n }\n #getParamValue(paramKey) {\n return this.#matchResult[1] ? this.#matchResult[1][paramKey] : paramKey;\n }\n query(key) {\n return getQueryParam(this.url, key);\n }\n queries(key) {\n return getQueryParams(this.url, key);\n }\n header(name) {\n if (name) {\n return this.raw.headers.get(name) ?? void 0;\n }\n const headerData = {};\n this.raw.headers.forEach((value, key) => {\n headerData[key] = value;\n });\n return headerData;\n }\n async parseBody(options) {\n return this.bodyCache.parsedBody ??= await parseBody(this, options);\n }\n #cachedBody = (key) => {\n const { bodyCache, raw } = this;\n const cachedBody = bodyCache[key];\n if (cachedBody) {\n return cachedBody;\n }\n const anyCachedKey = Object.keys(bodyCache)[0];\n if (anyCachedKey) {\n return bodyCache[anyCachedKey].then((body) => {\n if (anyCachedKey === \"json\") {\n body = JSON.stringify(body);\n }\n return new Response(body)[key]();\n });\n }\n return bodyCache[key] = raw[key]();\n };\n /**\n * `.json()` can parse Request body of type `application/json`\n *\n * @see {@link https://hono.dev/docs/api/request#json}\n *\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.json()\n * })\n * ```\n */\n json() {\n return this.#cachedBody(\"text\").then((text) => JSON.parse(text));\n }\n /**\n * `.text()` can parse Request body of type `text/plain`\n *\n * @see {@link https://hono.dev/docs/api/request#text}\n *\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.text()\n * })\n * ```\n */\n text() {\n return this.#cachedBody(\"text\");\n }\n /**\n * `.arrayBuffer()` parse Request body as an `ArrayBuffer`\n *\n * @see {@link https://hono.dev/docs/api/request#arraybuffer}\n *\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.arrayBuffer()\n * })\n * ```\n */\n arrayBuffer() {\n return this.#cachedBody(\"arrayBuffer\");\n }\n /**\n * Parses the request body as a `Blob`.\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.blob();\n * });\n * ```\n * @see https://hono.dev/docs/api/request#blob\n */\n blob() {\n return this.#cachedBody(\"blob\");\n }\n /**\n * Parses the request body as `FormData`.\n * @example\n * ```ts\n * app.post('/entry', async (c) => {\n * const body = await c.req.formData();\n * });\n * ```\n * @see https://hono.dev/docs/api/request#formdata\n */\n formData() {\n return this.#cachedBody(\"formData\");\n }\n /**\n * Adds validated data to the request.\n *\n * @param target - The target of the validation.\n * @param data - The validated data to add.\n */\n addValidatedData(target, data) {\n this.#validatedData[target] = data;\n }\n valid(target) {\n return this.#validatedData[target];\n }\n /**\n * `.url()` can get the request url strings.\n *\n * @see {@link https://hono.dev/docs/api/request#url}\n *\n * @example\n * ```ts\n * app.get('/about/me', (c) => {\n * const url = c.req.url // `http://localhost:8787/about/me`\n * ...\n * })\n * ```\n */\n get url() {\n return this.raw.url;\n }\n /**\n * `.method()` can get the method name of the request.\n *\n * @see {@link https://hono.dev/docs/api/request#method}\n *\n * @example\n * ```ts\n * app.get('/about/me', (c) => {\n * const method = c.req.method // `GET`\n * })\n * ```\n */\n get method() {\n return this.raw.method;\n }\n get [GET_MATCH_RESULT]() {\n return this.#matchResult;\n }\n /**\n * `.matchedRoutes()` can return a matched route in the handler\n *\n * @deprecated\n *\n * Use matchedRoutes helper defined in \"hono/route\" instead.\n *\n * @see {@link https://hono.dev/docs/api/request#matchedroutes}\n *\n * @example\n * ```ts\n * app.use('*', async function logger(c, next) {\n * await next()\n * c.req.matchedRoutes.forEach(({ handler, method, path }, i) => {\n * const name = handler.name || (handler.length < 2 ? '[handler]' : '[middleware]')\n * console.log(\n * method,\n * ' ',\n * path,\n * ' '.repeat(Math.max(10 - path.length, 0)),\n * name,\n * i === c.req.routeIndex ? '<- respond from here' : ''\n * )\n * })\n * })\n * ```\n */\n get matchedRoutes() {\n return this.#matchResult[0].map(([[, route]]) => route);\n }\n /**\n * `routePath()` can retrieve the path registered within the handler\n *\n * @deprecated\n *\n * Use routePath helper defined in \"hono/route\" instead.\n *\n * @see {@link https://hono.dev/docs/api/request#routepath}\n *\n * @example\n * ```ts\n * app.get('/posts/:id', (c) => {\n * return c.json({ path: c.req.routePath })\n * })\n * ```\n */\n get routePath() {\n return this.#matchResult[0].map(([[, route]]) => route)[this.routeIndex].path;\n }\n};\nvar cloneRawRequest = async (req) => {\n if (!req.raw.bodyUsed) {\n return req.raw.clone();\n }\n const cacheKey = Object.keys(req.bodyCache)[0];\n if (!cacheKey) {\n throw new HTTPException(500, {\n message: \"Cannot clone request: body was already consumed and not cached. Please use HonoRequest methods (e.g., req.json(), req.text()) instead of consuming req.raw directly.\"\n });\n }\n const requestInit = {\n body: await req[cacheKey](),\n cache: req.raw.cache,\n credentials: req.raw.credentials,\n headers: req.header(),\n integrity: req.raw.integrity,\n keepalive: req.raw.keepalive,\n method: req.method,\n mode: req.raw.mode,\n redirect: req.raw.redirect,\n referrer: req.raw.referrer,\n referrerPolicy: req.raw.referrerPolicy,\n signal: req.raw.signal\n };\n return new Request(req.url, requestInit);\n};\nexport {\n HonoRequest,\n cloneRawRequest\n};\n","// src/router/reg-exp-router/node.ts\nvar LABEL_REG_EXP_STR = \"[^/]+\";\nvar ONLY_WILDCARD_REG_EXP_STR = \".*\";\nvar TAIL_WILDCARD_REG_EXP_STR = \"(?:|/.*)\";\nvar PATH_ERROR = /* @__PURE__ */ Symbol();\nvar regExpMetaChars = new Set(\".\\\\+*[^]$()\");\nfunction compareKey(a, b) {\n if (a.length === 1) {\n return b.length === 1 ? a < b ? -1 : 1 : -1;\n }\n if (b.length === 1) {\n return 1;\n }\n if (a === ONLY_WILDCARD_REG_EXP_STR || a === TAIL_WILDCARD_REG_EXP_STR) {\n return 1;\n } else if (b === ONLY_WILDCARD_REG_EXP_STR || b === TAIL_WILDCARD_REG_EXP_STR) {\n return -1;\n }\n if (a === LABEL_REG_EXP_STR) {\n return 1;\n } else if (b === LABEL_REG_EXP_STR) {\n return -1;\n }\n return a.length === b.length ? a < b ? -1 : 1 : b.length - a.length;\n}\nvar Node = class _Node {\n #index;\n #varIndex;\n #children = /* @__PURE__ */ Object.create(null);\n insert(tokens, index, paramMap, context, pathErrorCheckOnly) {\n if (tokens.length === 0) {\n if (this.#index !== void 0) {\n throw PATH_ERROR;\n }\n if (pathErrorCheckOnly) {\n return;\n }\n this.#index = index;\n return;\n }\n const [token, ...restTokens] = tokens;\n const pattern = token === \"*\" ? restTokens.length === 0 ? [\"\", \"\", ONLY_WILDCARD_REG_EXP_STR] : [\"\", \"\", LABEL_REG_EXP_STR] : token === \"/*\" ? [\"\", \"\", TAIL_WILDCARD_REG_EXP_STR] : token.match(/^\\:([^\\{\\}]+)(?:\\{(.+)\\})?$/);\n let node;\n if (pattern) {\n const name = pattern[1];\n let regexpStr = pattern[2] || LABEL_REG_EXP_STR;\n if (name && pattern[2]) {\n if (regexpStr === \".*\") {\n throw PATH_ERROR;\n }\n regexpStr = regexpStr.replace(/^\\((?!\\?:)(?=[^)]+\\)$)/, \"(?:\");\n if (/\\((?!\\?:)/.test(regexpStr)) {\n throw PATH_ERROR;\n }\n }\n node = this.#children[regexpStr];\n if (!node) {\n if (Object.keys(this.#children).some(\n (k) => k !== ONLY_WILDCARD_REG_EXP_STR && k !== TAIL_WILDCARD_REG_EXP_STR\n )) {\n throw PATH_ERROR;\n }\n if (pathErrorCheckOnly) {\n return;\n }\n node = this.#children[regexpStr] = new _Node();\n if (name !== \"\") {\n node.#varIndex = context.varIndex++;\n }\n }\n if (!pathErrorCheckOnly && name !== \"\") {\n paramMap.push([name, node.#varIndex]);\n }\n } else {\n node = this.#children[token];\n if (!node) {\n if (Object.keys(this.#children).some(\n (k) => k.length > 1 && k !== ONLY_WILDCARD_REG_EXP_STR && k !== TAIL_WILDCARD_REG_EXP_STR\n )) {\n throw PATH_ERROR;\n }\n if (pathErrorCheckOnly) {\n return;\n }\n node = this.#children[token] = new _Node();\n }\n }\n node.insert(restTokens, index, paramMap, context, pathErrorCheckOnly);\n }\n buildRegExpStr() {\n const childKeys = Object.keys(this.#children).sort(compareKey);\n const strList = childKeys.map((k) => {\n const c = this.#children[k];\n return (typeof c.#varIndex === \"number\" ? `(${k})@${c.#varIndex}` : regExpMetaChars.has(k) ? `\\\\${k}` : k) + c.buildRegExpStr();\n });\n if (typeof this.#index === \"number\") {\n strList.unshift(`#${this.#index}`);\n }\n if (strList.length === 0) {\n return \"\";\n }\n if (strList.length === 1) {\n return strList[0];\n }\n return \"(?:\" + strList.join(\"|\") + \")\";\n }\n};\nexport {\n Node,\n PATH_ERROR\n};\n","import type { McpServer } from \"@modelcontextprotocol/sdk/server/mcp.js\";\nimport { StdioServerTransport } from \"@modelcontextprotocol/sdk/server/stdio.js\";\n\n/**\n * Stdio transport でMCPサーバーに接続する\n */\nexport async function connectStdio(server: McpServer): Promise<void> {\n await server.connect(new StdioServerTransport());\n}\n","import { ConnpassClient } from \"@kajidog/connpass-api-client\";\nimport { McpServer } from \"@modelcontextprotocol/sdk/server/mcp.js\";\nimport { getConfig } from \"./config.js\";\nimport { registerAllTools } from \"./tools/index.js\";\nimport { SearchSessionStore } from \"./tools/utils/searchSessionStore.js\";\nimport type { ToolDeps } from \"./tools/utils/types.js\";\n\nconst config = getConfig();\nlet cachedServer: McpServer | null = null;\n\nfunction createUnavailableConnpassClient(): ConnpassClient {\n const message =\n \"CONNPASS_API_KEY is required to use Connpass API tools. Set the environment variable or pass --connpass-api-key.\";\n\n return new Proxy({} as ConnpassClient, {\n get(_target, property) {\n if (property === \"then\") {\n return undefined;\n }\n\n if (typeof property === \"symbol\") {\n return undefined;\n }\n\n return async () => {\n throw new Error(message);\n };\n },\n });\n}\n\nfunction createConnpassClient(): ConnpassClient {\n if (!config.connpassApiKey?.trim()) {\n return createUnavailableConnpassClient();\n }\n\n return new ConnpassClient({\n apiKey: config.connpassApiKey,\n rateLimitEnabled: config.rateLimitEnabled,\n rateLimitDelay: config.rateLimitDelayMs,\n });\n}\n\n/**\n * McpServer を作成しツールを登録するファクトリ関数\n * HTTPモードではセッションごとに新しいインスタンスが必要\n */\nexport function createServer(): McpServer {\n const server = new McpServer({\n name: \"connpass-mcp-server\",\n version: \"0.3.0\",\n description: \"Connpass event search and browsing MCP server with Apps UI\",\n });\n\n const connpassClient = createConnpassClient();\n const searchSessionStore = new SearchSessionStore();\n\n const deps: ToolDeps = {\n server,\n connpassClient,\n config,\n searchSessionStore,\n };\n\n registerAllTools(deps);\n\n return server;\n}\n\nexport function getServer(): McpServer {\n if (!cachedServer) {\n cachedServer = createServer();\n }\n\n return cachedServer;\n}\n\nexport const server = getServer();\nexport { config };\n","/**\n * Connpass MCP Apps サーバー設定\n *\n * 優先順位: CLI引数 > 環境変数 > デフォルト値\n */\n\nimport {\n type BaseServerConfig,\n type ConfigDefs,\n baseConfigDefs,\n filterUndefined,\n generateHelp,\n getDefaultsFromDefs,\n parseCliFromDefs,\n parseEnvFromDefs,\n} from \"@kajidog/mcp-core\";\n\nconst connpassConfigDefs: ConfigDefs = {\n connpassApiKey: {\n cli: \"--connpass-api-key\",\n env: \"CONNPASS_API_KEY\",\n description: \"Connpass API key\",\n group: \"Connpass Configuration\",\n type: \"string\",\n valueName: \"<key>\",\n },\n defaultUserId: {\n cli: \"--default-user-id\",\n env: \"CONNPASS_DEFAULT_USER_ID\",\n description: \"Default Connpass user ID for schedule search\",\n group: \"Connpass Configuration\",\n type: \"number\",\n valueName: \"<id>\",\n },\n rateLimitEnabled: {\n cli: \"--rate-limit\",\n env: \"CONNPASS_RATE_LIMIT_ENABLED\",\n description: \"Enable API rate limiting\",\n group: \"Connpass Configuration\",\n type: \"boolean\",\n default: true,\n },\n rateLimitDelayMs: {\n cli: \"--rate-limit-delay\",\n env: \"CONNPASS_RATE_LIMIT_DELAY_MS\",\n description: \"Rate limit delay in milliseconds\",\n group: \"Connpass Configuration\",\n type: \"number\",\n default: 1000,\n valueName: \"<ms>\",\n },\n};\n\nexport const allConfigDefs: ConfigDefs = {\n ...connpassConfigDefs,\n ...baseConfigDefs,\n};\n\nexport interface ServerConfig extends BaseServerConfig {\n connpassApiKey?: string;\n defaultUserId?: number;\n rateLimitEnabled: boolean;\n rateLimitDelayMs: number;\n}\n\nfunction createDefaultConfig(): ServerConfig {\n const schemaDefs = getDefaultsFromDefs(allConfigDefs) as Record<\n string,\n unknown\n >;\n return schemaDefs as unknown as ServerConfig;\n}\n\nexport function getConfig(\n argv?: string[],\n env?: NodeJS.ProcessEnv,\n): ServerConfig {\n const cliConfig = parseCliFromDefs(\n allConfigDefs,\n argv ?? process.argv.slice(2),\n ) as Partial<ServerConfig>;\n const envConfig = parseEnvFromDefs(\n allConfigDefs,\n env ?? process.env,\n ) as Partial<ServerConfig>;\n\n const defaultConfig = createDefaultConfig();\n const merged: ServerConfig = {\n ...defaultConfig,\n ...filterUndefined(envConfig),\n ...filterUndefined(cliConfig),\n };\n\n return merged;\n}\n\nexport function getHelpText(): string {\n return generateHelp(allConfigDefs, {\n usage: \"connpass-mcp-apps [options]\",\n examples: [\n \"CONNPASS_API_KEY=xxx connpass-mcp-apps\",\n \"connpass-mcp-apps --http --port 8080\",\n \"connpass-mcp-apps --connpass-api-key xxx --default-user-id 12345\",\n ],\n });\n}\n\nlet cachedConfig: ServerConfig | null = null;\n\nexport function getCachedConfig(): ServerConfig {\n if (!cachedConfig) {\n cachedConfig = getConfig();\n }\n return cachedConfig;\n}\n","import { z } from \"zod\";\nimport { resolvePrefectureInputs } from \"./prefectures.js\";\nimport {\n FORMAT_PRESETS,\n type FormatEventOptions,\n formatEvent,\n formatEventsResponse,\n formatPresentationsResponse,\n summarizeEventsResponse,\n} from \"./utils/formatting.js\";\nimport { registerAppToolIfEnabled } from \"./utils/registration.js\";\nimport { connpassResourceUri } from \"./utils/resource.js\";\nimport {\n EVENT_SORT_KEYS,\n EVENT_SORT_MAP,\n applyPagination,\n normalizeKeywordOr,\n normalizeStringArray,\n parseHyphenatedDate,\n toYmdArray,\n} from \"./utils/shared.js\";\nimport type { ToolDeps } from \"./utils/types.js\";\n\nconst EventSearchInputSchema = z.object({\n anyQuery: z\n .string()\n .min(1)\n .describe(\"Search keywords (comma-separated, matches ANY keyword)\")\n .optional(),\n on: z\n .union([z.string().min(1), z.array(z.string().min(1))])\n .describe(\"Specific date(s) in YYYY-MM-DD or YYYYMMDD format\")\n .optional(),\n from: z\n .string()\n .min(1)\n .describe(\"Start of date range (inclusive). Format: YYYY-MM-DD\")\n .optional(),\n to: z\n .string()\n .min(1)\n .describe(\"End of date range (inclusive). Format: YYYY-MM-DD\")\n .optional(),\n participantNickname: z\n .string()\n .min(1)\n .describe(\"Filter by participant nickname\")\n .optional(),\n hostNickname: z\n .string()\n .min(1)\n .describe(\"Filter by host/owner nickname\")\n .optional(),\n groupIds: z\n .array(z.number())\n .describe(\"Limit results to specific group IDs\")\n .optional(),\n prefectures: z\n .union([z.string().min(1), z.array(z.string().min(1))])\n .describe(\"Prefecture name(s) to filter by\")\n .optional(),\n page: z.number().int().min(1).describe(\"1-based page number\").optional(),\n pageSize: z\n .number()\n .int()\n .min(1)\n .max(100)\n .describe(\"Events per page (default 20, max 100)\")\n .optional(),\n sort: z.enum(EVENT_SORT_KEYS).describe(\"Sort order\").optional(),\n includeDetails: z\n .boolean()\n .describe(\n \"Include event description (up to 200 chars). Use when you need content details for recommendations.\",\n )\n .default(false)\n .optional(),\n});\n\ntype EventSearchInput = z.infer<typeof EventSearchInputSchema>;\n\nconst EventPresentationsInputSchema = z.object({\n eventId: z\n .union([z.number().int().positive(), z.string().min(1)])\n .describe(\"Connpass event ID\")\n .transform((value) => Number(value))\n .refine((value) => Number.isFinite(value), \"eventId must be a number\"),\n});\n\nfunction buildEventSearchParams(input: EventSearchInput) {\n const pagination = applyPagination(input.page, input.pageSize);\n const resolved = resolvePrefectureInputs(input.prefectures);\n if (\"response\" in resolved) {\n return resolved;\n }\n\n return {\n keywordOr: normalizeKeywordOr(input.anyQuery),\n ymd: toYmdArray(input.on),\n ymdFrom: input.from ? parseHyphenatedDate(input.from) : undefined,\n ymdTo: input.to ? parseHyphenatedDate(input.to) : undefined,\n nickname: input.participantNickname,\n ownerNickname: input.hostNickname,\n groupId: input.groupIds,\n prefecture: resolved.prefectures ?? normalizeStringArray(input.prefectures),\n order: input.sort ? EVENT_SORT_MAP[input.sort] : undefined,\n ...pagination,\n };\n}\n\nconst LIST_FORMAT_OPTIONS: FormatEventOptions = {\n descriptionLimit: 0,\n catchPhraseLimit: 100,\n};\n\nexport function registerEventTools(deps: ToolDeps): void {\n const { server, connpassClient, searchSessionStore } = deps;\n\n const searchEventsHandler = async (args: Record<string, unknown>) => {\n const params = EventSearchInputSchema.parse(args ?? {});\n const searchParams = buildEventSearchParams(params);\n if (\"response\" in searchParams) {\n return {\n ...searchParams.response,\n isError: true,\n };\n }\n const response = await connpassClient.searchEvents(searchParams);\n const formatOptions = params.includeDetails\n ? FORMAT_PRESETS.detailed\n : FORMAT_PRESETS.default;\n const formatted = formatEventsResponse(response, formatOptions);\n const browseFormatted = formatEventsResponse(response, LIST_FORMAT_OPTIONS);\n const searchSessionId = searchSessionStore.save(browseFormatted);\n return {\n content: [\n {\n type: \"text\" as const,\n text: summarizeEventsResponse(formatted, \"events\", {\n searchSessionId,\n }),\n },\n ],\n structuredContent: {\n kind: \"events\",\n searchSessionId,\n data: formatted,\n },\n };\n };\n\n // search_events - 公開ツール(UI なし、モデル向け)\n registerAppToolIfEnabled(\n server,\n \"search_events\",\n {\n title: \"Search Events\",\n description:\n \"Search Connpass events and return results as text for reasoning and recommendations. Use this whenever the user asks about events.\",\n inputSchema: EventSearchInputSchema,\n },\n searchEventsHandler,\n );\n\n // browse_events - 公開ツール(UI あり、対話ブラウズ向け)\n registerAppToolIfEnabled(\n server,\n \"browse_events\",\n {\n title: \"Browse Events\",\n description:\n \"Display previously searched events in the interactive event browser UI. Use this proactively when the user wants to browse event options, scan many candidates, compare events, or inspect results visually. The UI lets the user refine and re-run searches directly, so prefer this for event exploration. Call this ONCE with the searchSessionId returned by search_events. Do not use this to search again.\",\n inputSchema: z.object({\n searchSessionId: z\n .uuid()\n .describe(\"Session ID returned by search_events\"),\n }),\n _meta: {\n ui: { resourceUri: connpassResourceUri },\n },\n },\n async (args: Record<string, unknown>) => {\n const { searchSessionId } = z\n .object({\n searchSessionId: z.uuid(),\n })\n .parse(args ?? {});\n const formatted = searchSessionStore.get(searchSessionId);\n if (!formatted) {\n return {\n content: [\n {\n type: \"text\" as const,\n text: \"Search session not found or expired. Run search_events again, then call browse_events once with the returned searchSessionId.\",\n },\n ],\n isError: true,\n };\n }\n\n return {\n content: [\n {\n type: \"text\" as const,\n text: `Displayed ${formatted.events.length} of ${formatted.available} events in the browser UI.`,\n },\n ],\n structuredContent: {\n kind: \"events\",\n searchSessionId,\n data: formatted,\n },\n };\n },\n );\n\n // get_event_presentations - 公開ツール\n registerAppToolIfEnabled(\n server,\n \"get_event_presentations\",\n {\n title: \"Get Event Presentations\",\n description: \"Look up presentation details for a specific event\",\n inputSchema: EventPresentationsInputSchema,\n },\n async (args: Record<string, unknown>) => {\n const { eventId } = EventPresentationsInputSchema.parse(args ?? {});\n const response = await connpassClient.getEventPresentations(eventId);\n const formatted = formatPresentationsResponse(response);\n return {\n content: [{ type: \"text\" as const, text: JSON.stringify(formatted) }],\n };\n },\n );\n\n // get_event_detail - 公開ツール\n registerAppToolIfEnabled(\n server,\n \"get_event_detail\",\n {\n title: \"Get Event Detail\",\n description:\n \"Get full details of a specific event by ID, including complete description and presentations\",\n inputSchema: EventPresentationsInputSchema,\n },\n async (args: Record<string, unknown>) => {\n const { eventId } = EventPresentationsInputSchema.parse(args ?? {});\n\n const [eventsResponse, presentationsResponse] = await Promise.all([\n connpassClient.searchEvents({ eventId: [eventId], count: 1 }),\n connpassClient.getEventPresentations(eventId).catch(() => undefined),\n ]);\n\n const event = eventsResponse.events[0];\n if (!event) {\n throw new Error(`Event with ID ${eventId} not found.`);\n }\n\n const formatted = formatEvent(event, FORMAT_PRESETS.full);\n const presentations = presentationsResponse\n ? formatPresentationsResponse(presentationsResponse)\n : undefined;\n\n return {\n content: [\n {\n type: \"text\" as const,\n text: JSON.stringify({ event: formatted, presentations }),\n },\n ],\n };\n },\n );\n}\n","import {\n getAllPrefectures,\n normalizePrefecture,\n} from \"@kajidog/connpass-api-client\";\nimport { z } from \"zod\";\nimport { registerAppToolIfEnabled } from \"./utils/registration.js\";\nimport type { ToolDeps } from \"./utils/types.js\";\n\nconst EmptyInputSchema = z.object({});\n\nfunction buildPrefectureListResult(invalid: string[] = []) {\n const prefectures = getAllPrefectures();\n const payload = {\n invalid,\n prefectures,\n };\n\n const invalidText =\n invalid.length > 0\n ? `指定された都道府県が見つかりません: ${invalid.join(\", \")}\\n`\n : \"\";\n\n return {\n content: [\n {\n type: \"text\" as const,\n text: `${invalidText}利用可能な都道府県一覧: ${prefectures.map((item) => `${item.name} (${item.code})`).join(\", \")}`,\n },\n ],\n structuredContent: {\n kind: \"prefectures\",\n data: payload,\n },\n };\n}\n\nexport function resolvePrefectureInputs(\n values?: string | string[],\n):\n | { prefectures?: string[] }\n | { response: ReturnType<typeof buildPrefectureListResult> } {\n if (!values) return {};\n\n const inputs = Array.isArray(values) ? values : [values];\n const prefectures: string[] = [];\n const invalid: string[] = [];\n\n for (const value of inputs) {\n const normalized = normalizePrefecture(value);\n if (!normalized) {\n invalid.push(value);\n continue;\n }\n prefectures.push(normalized);\n }\n\n if (invalid.length > 0) {\n return { response: buildPrefectureListResult(invalid) };\n }\n\n return { prefectures };\n}\n\nexport function registerPrefectureTools(deps: ToolDeps): void {\n const { server } = deps;\n\n registerAppToolIfEnabled(\n server,\n \"list_prefectures\",\n {\n title: \"List Prefectures\",\n description: \"List supported prefectures and region codes for filtering\",\n inputSchema: EmptyInputSchema,\n },\n async () => buildPrefectureListResult(),\n );\n\n registerAppToolIfEnabled(\n server,\n \"_get_prefectures\",\n {\n title: \"Get Prefectures (UI)\",\n description: \"Internal: list supported prefectures for the UI\",\n inputSchema: EmptyInputSchema,\n _meta: {\n ui: {\n visibility: [\"app\"],\n },\n },\n },\n async () => buildPrefectureListResult(),\n );\n}\n","import { registerAppTool } from \"@modelcontextprotocol/ext-apps/server\";\nimport type {\n McpUiAppToolConfig,\n ToolCallback,\n ToolConfig,\n} from \"@modelcontextprotocol/ext-apps/server\";\nimport type { McpServer } from \"@modelcontextprotocol/sdk/server/mcp.js\";\nimport type {\n AnySchema,\n ZodRawShapeCompat,\n} from \"@modelcontextprotocol/sdk/server/zod-compat.js\";\n\n/**\n * MCP Apps Extension 対応のツール登録ヘルパー\n */\nexport function registerAppToolIfEnabled(\n server: McpServer,\n name: string,\n ...args: [\n config: ToolConfig & {\n inputSchema?: undefined | ZodRawShapeCompat | AnySchema;\n },\n cb: ToolCallback<undefined | ZodRawShapeCompat | AnySchema>,\n ]\n) {\n const [config, cb] = args;\n const normalizedConfig = {\n ...config,\n _meta: config._meta && typeof config._meta === \"object\" ? config._meta : {},\n } as McpUiAppToolConfig & {\n inputSchema?: undefined | ZodRawShapeCompat | AnySchema;\n };\n\n registerAppTool(server, name, normalizedConfig, cb);\n}\n","import type {\n Event,\n EventsResponse,\n Group,\n GroupsResponse,\n Presentation,\n PresentationsResponse,\n User,\n UsersResponse,\n} from \"@kajidog/connpass-api-client\";\n\nconst HTML_ENTITY_MAP: Record<string, string> = {\n \"&nbsp;\": \" \",\n \"&amp;\": \"&\",\n \"&lt;\": \"<\",\n \"&gt;\": \">\",\n \"&quot;\": '\"',\n \"&#34;\": '\"',\n \"&#39;\": \"'\",\n \"&#x27;\": \"'\",\n \"&#x60;\": \"`\",\n};\n\nexport interface FormatEventOptions {\n descriptionLimit?: number;\n catchPhraseLimit?: number;\n presentationDescriptionLimit?: number;\n}\n\nexport const FORMAT_PRESETS = {\n default: {\n descriptionLimit: 0,\n catchPhraseLimit: undefined,\n } as FormatEventOptions,\n detailed: {\n descriptionLimit: 200,\n catchPhraseLimit: undefined,\n } as FormatEventOptions,\n full: {\n descriptionLimit: undefined,\n catchPhraseLimit: undefined,\n } as FormatEventOptions,\n};\n\nexport interface FormattedPresentation {\n id: number;\n title: string;\n speaker: string;\n summary?: string;\n links?: {\n url?: string;\n slideshare?: string;\n youtube?: string;\n twitter?: string;\n };\n order: number;\n updatedAt: string;\n}\n\nexport interface FormattedEvent {\n id: number;\n title: string;\n catchPhrase?: string;\n summary?: string;\n url: string;\n hashTag?: string;\n imageUrl?: string;\n schedule: { start: string; end: string };\n location?: { place?: string; address?: string };\n owner: { nickname: string; displayName: string };\n participants: { accepted: number; waiting: number; limit?: number };\n group?: { id?: number; title?: string; url?: string };\n updatedAt: string;\n presentations?: FormattedPresentation[];\n}\n\nexport interface FormattedEventsResponse {\n returned: number;\n available: number;\n start: number;\n events: FormattedEvent[];\n}\n\nexport interface FormattedPresentationsResponse {\n returned: number;\n presentations: FormattedPresentation[];\n}\n\nfunction compactDateLabel(isoString?: string): string | undefined {\n if (!isoString) return undefined;\n const date = new Date(isoString);\n if (Number.isNaN(date.getTime())) return undefined;\n const year = date.getFullYear();\n const month = String(date.getMonth() + 1).padStart(2, \"0\");\n const day = String(date.getDate()).padStart(2, \"0\");\n return `${year}-${month}-${day}`;\n}\n\nfunction summarizeEventLine(event: FormattedEvent): string {\n const date = compactDateLabel(event.schedule.start);\n const place = event.location?.place || event.location?.address;\n const { accepted, waiting, limit } = event.participants;\n const capacityParts: string[] = [`${accepted}人参加`];\n if (waiting > 0) capacityParts.push(`待ち${waiting}`);\n if (typeof limit === \"number\") capacityParts.push(`定員${limit}`);\n const capacity = capacityParts.join(\"/\");\n\n const fragments = [\n `[id:${event.id}]`,\n date,\n event.title,\n place ? `@ ${place}` : undefined,\n `(${capacity})`,\n ].filter(Boolean);\n\n return `- ${fragments.join(\" \")}`;\n}\n\nfunction summarizeGroupLine(group: Group): string {\n const place = [group.prefecture, group.place].filter(Boolean).join(\" \");\n const fragments = [\n group.title?.trim(),\n typeof group.id === \"number\" ? `id:${group.id}` : undefined,\n place || undefined,\n ].filter(Boolean);\n\n return `- ${fragments.join(\" / \")}`;\n}\n\nfunction summarizeUserLine(user: User): string {\n const fragments = [\n user.nickname?.trim(),\n typeof user.id === \"number\" ? `id:${user.id}` : undefined,\n user.displayName?.trim() ? `name:${user.displayName.trim()}` : undefined,\n ].filter(Boolean);\n\n return `- ${fragments.join(\" / \")}`;\n}\n\nexport function truncateText(text: string, limit: number): string {\n if (!text) return \"\";\n if (text.length <= limit) return text;\n if (limit <= 3) return text.slice(0, limit);\n return `${text.slice(0, limit - 3).trimEnd()}...`;\n}\n\nfunction decodeHtmlEntities(input: string): string {\n return input\n .replace(/&(#x?[0-9a-fA-F]+|[a-zA-Z]+);/g, (entity) => {\n const mapped = HTML_ENTITY_MAP[entity];\n if (mapped) return mapped;\n\n const numericMatch = entity.match(/^&#(x?[0-9a-fA-F]+);$/);\n if (!numericMatch) return entity;\n\n const value = numericMatch[1];\n const codePoint =\n value.startsWith(\"x\") || value.startsWith(\"X\")\n ? Number.parseInt(value.slice(1), 16)\n : Number.parseInt(value, 10);\n\n if (!Number.isFinite(codePoint)) return entity;\n\n try {\n return String.fromCodePoint(codePoint);\n } catch {\n return entity;\n }\n })\n .replace(/\\u00a0/gi, \" \");\n}\n\nexport function stripHtml(input: string): string {\n if (!input) return \"\";\n\n const withoutScripts = input.replace(/<script[\\s\\S]*?<\\/script>/gi, \"\");\n const withoutStyles = withoutScripts.replace(/<style[\\s\\S]*?<\\/style>/gi, \"\");\n\n // Replace <br> inside table cells with space to preserve row structure\n const withCellBrNormalized = withoutStyles.replace(\n /(<(?:td|th)\\b[^>]*>)([\\s\\S]*?)(<\\/(?:td|th)>)/gi,\n (_, open, content, close) =>\n `${open}${content.replace(/<br\\s*\\/?\\s*>/gi, \" \")}${close}`,\n );\n\n const normalizedTableCells = withCellBrNormalized\n .replace(/<\\/(td|th)>\\s*<(td|th)/gi, \"</$1>\\t<$2\")\n .replace(/<\\/(tr)>\\s*<tr/gi, \"</$1>\\n<tr\");\n\n const withLineBreaks = normalizedTableCells\n .replace(/<br\\s*\\/?\\s*>/gi, \"\\n\")\n .replace(/<\\/(p|div|section|article|header|footer|li)>/gi, \"\\n\")\n .replace(/<\\/(td|th)>/gi, \"\\t\")\n .replace(/<li[^>]*>/gi, \"- \")\n .replace(/<h[1-6]\\b[^>]*>/gi, \"\\n## \")\n .replace(/<\\/(h[1-6]|tr)>/gi, \"\\n\");\n\n const withoutTags = withLineBreaks.replace(/<[^>]+>/g, \"\");\n return decodeHtmlEntities(withoutTags);\n}\n\nexport function sanitizeRichText(input: string): string {\n if (!input) return \"\";\n\n const stripped = stripHtml(input);\n const normalizedWhitespace = stripped\n .replace(/\\r/g, \"\\n\")\n .split(/\\n+/)\n .map((line) =>\n line\n .split(/\\t+/)\n .map((cell) => cell.trim())\n .filter(Boolean)\n .join(\"\\t\"),\n )\n .filter(Boolean)\n .join(\"\\n\");\n\n return normalizedWhitespace\n .replace(/ {2,}/g, \" \")\n .replace(/ *\\t */g, \"\\t\")\n .trim();\n}\n\nexport function formatPresentation(\n presentation: Presentation,\n descriptionLimit?: number,\n): FormattedPresentation {\n const summary = sanitizeRichText(presentation.description);\n const title = String(presentation.title ?? \"\").trim();\n const speaker = String(presentation.speakerName ?? \"\").trim();\n\n const formatted: FormattedPresentation = {\n id: presentation.id,\n title: title || \"Untitled presentation\",\n speaker: speaker || \"Speaker unknown\",\n order: presentation.order,\n updatedAt: presentation.updatedAt,\n };\n\n const processedSummary =\n typeof descriptionLimit === \"number\" && descriptionLimit > 0\n ? truncateText(summary, descriptionLimit)\n : summary;\n\n if (processedSummary) {\n formatted.summary = processedSummary;\n }\n\n const links: FormattedPresentation[\"links\"] = {\n url: presentation.url,\n slideshare: presentation.slideshareUrl,\n youtube: presentation.youtubeUrl,\n twitter: presentation.twitterUrl,\n };\n\n if (links.url || links.slideshare || links.youtube || links.twitter) {\n formatted.links = links;\n }\n\n return formatted;\n}\n\nexport function formatPresentationsResponse(\n response: PresentationsResponse,\n options?: Pick<FormatEventOptions, \"presentationDescriptionLimit\">,\n): FormattedPresentationsResponse {\n const descriptionLimit = options?.presentationDescriptionLimit;\n\n return {\n returned: response.presentationsReturned,\n presentations: response.presentations.map((presentation) =>\n formatPresentation(presentation, descriptionLimit),\n ),\n };\n}\n\nexport function formatEvent(\n event: Event,\n options?: FormatEventOptions,\n): FormattedEvent {\n const descriptionLimit = options?.descriptionLimit;\n const catchPhraseLimit = options?.catchPhraseLimit;\n const presentationDescriptionLimit = options?.presentationDescriptionLimit;\n\n const catchPhrase = sanitizeRichText(event.catchPhrase);\n const description = sanitizeRichText(event.description);\n\n const participants: FormattedEvent[\"participants\"] = {\n accepted: event.participantCount,\n waiting: event.waitingCount,\n };\n\n if (typeof event.limit === \"number\") {\n participants.limit = event.limit;\n }\n\n const formatted: FormattedEvent = {\n id: event.id,\n title: event.title.trim(),\n url: event.url,\n schedule: {\n start: event.startedAt,\n end: event.endedAt,\n },\n owner: {\n nickname: event.ownerNickname,\n displayName: event.ownerDisplayName,\n },\n participants,\n updatedAt: event.updatedAt,\n };\n\n if (event.hashTag) formatted.hashTag = event.hashTag;\n if (event.imageUrl) formatted.imageUrl = event.imageUrl;\n\n const processedCatchPhrase =\n typeof catchPhraseLimit === \"number\" && catchPhraseLimit > 0\n ? truncateText(catchPhrase, catchPhraseLimit)\n : catchPhrase;\n if (processedCatchPhrase) formatted.catchPhrase = processedCatchPhrase;\n\n const processedDescription =\n typeof descriptionLimit === \"number\"\n ? descriptionLimit > 0\n ? truncateText(description, descriptionLimit)\n : \"\"\n : description;\n if (processedDescription) formatted.summary = processedDescription;\n\n if (event.place || event.address) {\n const location: FormattedEvent[\"location\"] = {};\n if (event.place) location.place = event.place;\n if (event.address) location.address = event.address;\n if (Object.keys(location).length > 0) formatted.location = location;\n }\n\n if (event.groupId || event.groupTitle || event.groupUrl) {\n const group: FormattedEvent[\"group\"] = {};\n if (typeof event.groupId === \"number\") group.id = event.groupId;\n if (event.groupTitle) group.title = event.groupTitle;\n if (event.groupUrl) group.url = event.groupUrl;\n if (Object.keys(group).length > 0) formatted.group = group;\n }\n\n const eventWithPresentations = event as Event & {\n presentations?: PresentationsResponse[\"presentations\"];\n };\n\n if (eventWithPresentations.presentations?.length) {\n formatted.presentations = eventWithPresentations.presentations.map(\n (presentation) =>\n formatPresentation(presentation, presentationDescriptionLimit),\n );\n }\n\n return formatted;\n}\n\nexport function formatEventsResponse(\n response: EventsResponse,\n options?: FormatEventOptions,\n): FormattedEventsResponse {\n return {\n returned: response.eventsReturned,\n available: response.eventsAvailable,\n start: response.eventsStart,\n events: response.events.map((event) => formatEvent(event, options)),\n };\n}\n\nfunction summarizeEventBlock(event: FormattedEvent): string {\n const headline = summarizeEventLine(event);\n const extras: string[] = [];\n if (event.catchPhrase) extras.push(` ${event.catchPhrase}`);\n if (event.group?.title) extras.push(` group: ${event.group.title}`);\n if (extras.length === 0) return headline;\n return [headline, ...extras].join(\"\\n\");\n}\n\nexport function summarizeEventsResponse(\n response: FormattedEventsResponse,\n label = \"events\",\n options?: { searchSessionId?: string },\n): string {\n const lines = [\n `${label}: ${response.returned} returned / ${response.available} available`,\n ];\n\n if (options?.searchSessionId) {\n lines.push(`searchSessionId: ${options.searchSessionId}`);\n }\n\n lines.push(...response.events.map(summarizeEventBlock));\n\n return lines.join(\"\\n\");\n}\n\nexport function summarizeGroupsResponse(response: GroupsResponse): string {\n const lines = [\n `groups: ${response.groupsReturned} returned / ${response.groupsAvailable} available`,\n ...response.groups.slice(0, 5).map(summarizeGroupLine),\n ];\n\n if (response.groups.length > 5) {\n lines.push(`- ...and ${response.groups.length - 5} more`);\n }\n\n return lines.join(\"\\n\");\n}\n\nexport function summarizeUsersResponse(response: UsersResponse): string {\n const lines = [\n `users: ${response.usersReturned} returned / ${response.usersAvailable} available`,\n ...response.users.slice(0, 5).map(summarizeUserLine),\n ];\n\n if (response.users.length > 5) {\n lines.push(`- ...and ${response.users.length - 5} more`);\n }\n\n return lines.join(\"\\n\");\n}\n\nexport function formatEventList(\n events: (Event & {\n presentations?: PresentationsResponse[\"presentations\"];\n })[],\n options?: FormatEventOptions,\n): FormattedEvent[] {\n return events.map((event) => formatEvent(event, options));\n}\n","import { readFileSync } from \"node:fs\";\nimport { dirname, join } from \"node:path\";\nimport { fileURLToPath } from \"node:url\";\nimport {\n RESOURCE_MIME_TYPE,\n registerAppResource,\n} from \"@modelcontextprotocol/ext-apps/server\";\nimport type { ReadResourceResult } from \"@modelcontextprotocol/sdk/types.js\";\nimport type { ToolDeps } from \"./types.js\";\n\nconst __dirname =\n typeof import.meta.dirname === \"string\"\n ? import.meta.dirname\n : dirname(fileURLToPath(import.meta.url));\n\nexport const connpassResourceUri = \"ui://connpass/app.html\";\n\nfunction loadConnpassHtml(): string {\n try {\n // 本番ビルド: dist 配下に同梱された HTML を読む\n const htmlPath = join(__dirname, \"mcp-app.html\");\n return readFileSync(htmlPath, \"utf-8\");\n } catch {\n try {\n // 開発時: node_modules の connpass-ui ビルド成果物を参照する\n const htmlPath = join(\n __dirname,\n \"..\",\n \"..\",\n \"node_modules\",\n \"@kajidog\",\n \"connpass-ui\",\n \"dist\",\n \"mcp-app.html\",\n );\n return readFileSync(htmlPath, \"utf-8\");\n } catch {\n console.error(\n \"Warning: connpass-ui HTML not found. Please build @kajidog/connpass-ui first.\",\n );\n return \"<html><body><p>Connpass UI not available. Please build @kajidog/connpass-ui.</p></body></html>\";\n }\n }\n}\n\nconst connpassHtml = loadConnpassHtml();\n\nexport function registerConnpassResource(deps: ToolDeps): void {\n const { server } = deps;\n registerAppResource(\n server,\n \"Connpass App\",\n connpassResourceUri,\n {\n description: \"Interactive Connpass event browser\",\n mimeType: RESOURCE_MIME_TYPE,\n },\n async (): Promise<ReadResourceResult> => ({\n contents: [\n {\n uri: connpassResourceUri,\n mimeType: RESOURCE_MIME_TYPE,\n text: connpassHtml,\n _meta: {\n ui: {\n csp: {\n resourceDomains: [\n \"https://connpass.com\",\n \"https://media.connpass.com\",\n ],\n },\n },\n },\n },\n ],\n }),\n );\n}\n","export const DEFAULT_PAGE_SIZE = 20;\n\nexport const EVENT_SORT_KEYS = [\n \"start-date-asc\",\n \"start-date-desc\",\n \"newly-added\",\n] as const;\nexport type EventSortKey = (typeof EVENT_SORT_KEYS)[number];\nexport const EVENT_SORT_MAP: Record<EventSortKey, 1 | 2 | 3> = {\n \"start-date-asc\": 1,\n \"start-date-desc\": 2,\n \"newly-added\": 3,\n};\n\nexport const GROUP_SORT_KEYS = [\n \"most-events\",\n \"most-members\",\n \"newly-added\",\n] as const;\nexport type GroupSortKey = (typeof GROUP_SORT_KEYS)[number];\nexport const GROUP_SORT_MAP: Record<GroupSortKey, 1 | 2 | 3> = {\n \"most-events\": 1,\n \"most-members\": 2,\n \"newly-added\": 3,\n};\n\nexport const USER_SORT_KEYS = [\n \"most-events\",\n \"most-followers\",\n \"newly-added\",\n] as const;\nexport type UserSortKey = (typeof USER_SORT_KEYS)[number];\nexport const USER_SORT_MAP: Record<UserSortKey, 1 | 2 | 3> = {\n \"most-events\": 1,\n \"most-followers\": 2,\n \"newly-added\": 3,\n};\n\nfunction formatAsCompactYmd(date: Date): string {\n const year = date.getFullYear();\n const month = String(date.getMonth() + 1).padStart(2, \"0\");\n const day = String(date.getDate()).padStart(2, \"0\");\n return `${year}${month}${day}`;\n}\n\nfunction formatAsHyphenatedYmd(date: Date): string {\n const year = date.getFullYear();\n const month = String(date.getMonth() + 1).padStart(2, \"0\");\n const day = String(date.getDate()).padStart(2, \"0\");\n return `${year}-${month}-${day}`;\n}\n\nexport function parseDateInput(\n input: string,\n options?: { style?: \"compact\" | \"hyphenated\" },\n): string {\n const normalized = input.trim();\n const hyphenFree = normalized.replace(/[-/.]/g, \"\");\n\n if (/^\\d{8}$/.test(hyphenFree)) {\n const year = Number(hyphenFree.slice(0, 4));\n const month = Number(hyphenFree.slice(4, 6)) - 1;\n const day = Number(hyphenFree.slice(6, 8));\n const candidate = new Date(Date.UTC(year, month, day));\n if (!Number.isNaN(candidate.getTime())) {\n return options?.style === \"hyphenated\"\n ? formatAsHyphenatedYmd(candidate)\n : formatAsCompactYmd(candidate);\n }\n }\n\n throw new Error(\n `Invalid date format: ${input}. Expected YYYY-MM-DD or YYYYMMDD format.`,\n );\n}\n\nexport function toYmdArray(value?: string | string[]): string[] | undefined {\n if (!value) return undefined;\n const inputs = Array.isArray(value) ? value : [value];\n return inputs.map((item) => parseDateInput(item));\n}\n\nexport function parseHyphenatedDate(input: string): string {\n const parsed = parseDateInput(input, { style: \"hyphenated\" });\n if (/^\\d{4}-\\d{2}-\\d{2}$/.test(parsed)) return parsed;\n\n const digitsOnly = parsed.replace(/[^0-9]/g, \"\");\n if (digitsOnly.length === 8) {\n return `${digitsOnly.slice(0, 4)}-${digitsOnly.slice(4, 6)}-${digitsOnly.slice(6, 8)}`;\n }\n\n throw new Error(`Could not convert date input to YYYY-MM-DD: ${input}`);\n}\n\nexport function normalizeStringArray(\n value?: string | string[],\n): string[] | undefined {\n if (!value) return undefined;\n return Array.isArray(value) ? value : [value];\n}\n\nexport function normalizeKeywordOr(value?: string): string | undefined {\n if (!value) return undefined;\n\n const tokens = value\n .split(/[、,]+/)\n .map((item) => item.trim())\n .filter(Boolean);\n\n if (tokens.length === 0) return undefined;\n return tokens.join(\",\");\n}\n\nexport type PaginationParams = { start?: number; count?: number };\n\nexport function applyPagination(\n page: number | undefined,\n pageSize: number | undefined,\n options?: { includePagination?: boolean },\n): PaginationParams {\n const includePagination = options?.includePagination ?? true;\n if (!includePagination) return {};\n\n const effectivePageSize = pageSize ?? DEFAULT_PAGE_SIZE;\n const pagination: PaginationParams = {};\n\n if (page) {\n pagination.start = 1 + (page - 1) * effectivePageSize;\n pagination.count = effectivePageSize;\n } else if (pageSize) {\n pagination.count = effectivePageSize;\n }\n\n return pagination;\n}\n","import { z } from \"zod\";\nimport { resolvePrefectureInputs } from \"./prefectures.js\";\nimport { summarizeGroupsResponse } from \"./utils/formatting.js\";\nimport { registerAppToolIfEnabled } from \"./utils/registration.js\";\nimport {\n GROUP_SORT_KEYS,\n GROUP_SORT_MAP,\n applyPagination,\n} from \"./utils/shared.js\";\nimport type { ToolDeps } from \"./utils/types.js\";\n\nconst GroupSearchInputSchema = z.object({\n query: z\n .string()\n .min(1)\n .describe(\"Keywords matching group title or description\")\n .optional(),\n groupIds: z\n .array(z.number())\n .describe(\"Limit to specific group IDs\")\n .optional(),\n country: z.string().min(1).describe(\"ISO country code, e.g. 'JP'\").optional(),\n prefecture: z\n .string()\n .min(1)\n .describe(\"Prefecture name to filter by\")\n .optional(),\n page: z.number().int().min(1).describe(\"1-based page number\").optional(),\n pageSize: z\n .number()\n .int()\n .min(1)\n .max(100)\n .describe(\"Groups per page (default 20)\")\n .optional(),\n sort: z\n .enum(GROUP_SORT_KEYS)\n .describe(\"Ranking by activity, members, or recency\")\n .optional(),\n});\n\ntype GroupSearchInput = z.infer<typeof GroupSearchInputSchema>;\n\nfunction buildGroupSearchParams(input: GroupSearchInput) {\n const pagination = applyPagination(input.page, input.pageSize);\n const resolved = resolvePrefectureInputs(input.prefecture);\n if (\"response\" in resolved) {\n return resolved;\n }\n\n return {\n keyword: input.query,\n groupId: input.groupIds,\n countryCode: input.country,\n prefecture: resolved.prefectures?.[0] ?? input.prefecture,\n order: input.sort ? GROUP_SORT_MAP[input.sort] : undefined,\n ...pagination,\n };\n}\n\nexport function registerGroupTools(deps: ToolDeps): void {\n const { server, connpassClient } = deps;\n\n registerAppToolIfEnabled(\n server,\n \"search_groups\",\n {\n title: \"Search Groups\",\n description: \"Find Connpass groups with simple filters\",\n inputSchema: GroupSearchInputSchema,\n },\n async (args: Record<string, unknown>) => {\n const params = GroupSearchInputSchema.parse(args ?? {});\n const searchParams = buildGroupSearchParams(params);\n if (\"response\" in searchParams) {\n return {\n ...searchParams.response,\n isError: true,\n };\n }\n const response = await connpassClient.searchGroups(searchParams);\n return {\n content: [\n { type: \"text\" as const, text: summarizeGroupsResponse(response) },\n ],\n };\n },\n );\n}\n","import type { Event } from \"@kajidog/connpass-api-client\";\nimport { z } from \"zod\";\nimport {\n FORMAT_PRESETS,\n formatEventList,\n summarizeEventsResponse,\n} from \"./utils/formatting.js\";\nimport { registerAppToolIfEnabled } from \"./utils/registration.js\";\nimport { connpassResourceUri } from \"./utils/resource.js\";\nimport { EVENT_SORT_MAP, parseHyphenatedDate } from \"./utils/shared.js\";\nimport type { ToolDeps } from \"./utils/types.js\";\n\nconst ScheduleInputSchema = z.object({\n userId: z.number().int().positive().describe(\"Connpass user ID\").optional(),\n nickname: z.string().min(1).describe(\"Connpass user nickname\").optional(),\n fromDate: z\n .string()\n .min(1)\n .describe(\"Start date (YYYY-MM-DD). Defaults to today.\")\n .optional(),\n toDate: z\n .string()\n .min(1)\n .describe(\"End date (YYYY-MM-DD). Defaults to +7 days.\")\n .optional(),\n maxEvents: z\n .number()\n .int()\n .min(1)\n .max(100)\n .describe(\"Maximum events to check (default 30)\")\n .optional(),\n includeDetails: z\n .boolean()\n .describe(\"Include event description (up to 200 chars)\")\n .default(false)\n .optional(),\n});\n\nfunction startOfDay(date: Date): Date {\n return new Date(date.getFullYear(), date.getMonth(), date.getDate());\n}\n\nfunction formatDateLabel(date: Date): string {\n const year = date.getFullYear();\n const month = String(date.getMonth() + 1).padStart(2, \"0\");\n const day = String(date.getDate()).padStart(2, \"0\");\n return `${year}-${month}-${day}`;\n}\n\nexport function registerScheduleTools(deps: ToolDeps): void {\n const { server, connpassClient, config } = deps;\n\n registerAppToolIfEnabled(\n server,\n \"search_schedule\",\n {\n title: \"Search Schedule\",\n description: \"Search user's schedule - events within a date range\",\n inputSchema: ScheduleInputSchema,\n _meta: {\n ui: { resourceUri: connpassResourceUri },\n },\n },\n async (args: Record<string, unknown>) => {\n const parsed = ScheduleInputSchema.parse(args ?? {});\n\n let resolvedUserId: number | undefined =\n parsed.userId ?? config.defaultUserId;\n let userNickname: string | undefined;\n\n if (parsed.nickname) {\n const userSearchResponse = await connpassClient.searchUsers({\n nickname: parsed.nickname,\n });\n if (userSearchResponse.users.length === 0) {\n throw new Error(`User with nickname \"${parsed.nickname}\" not found.`);\n }\n resolvedUserId = userSearchResponse.users[0].id;\n userNickname = userSearchResponse.users[0].nickname;\n }\n\n if (!resolvedUserId) {\n throw new Error(\n \"User ID or nickname is required. Pass userId, nickname, or set CONNPASS_DEFAULT_USER_ID.\",\n );\n }\n\n const maxEventsToFetch = parsed.maxEvents ?? 30;\n\n const today = startOfDay(new Date());\n const rangeStart = parsed.fromDate\n ? startOfDay(new Date(parseHyphenatedDate(parsed.fromDate)))\n : today;\n const rangeEnd = parsed.toDate\n ? startOfDay(new Date(parseHyphenatedDate(parsed.toDate)))\n : (() => {\n const defaultEnd = new Date(rangeStart);\n defaultEnd.setDate(defaultEnd.getDate() + 7);\n return defaultEnd;\n })();\n rangeEnd.setHours(23, 59, 59, 999);\n\n if (!userNickname) {\n const userResponse = await connpassClient.searchUsers({\n userId: [resolvedUserId],\n });\n if (userResponse.users.length === 0) {\n throw new Error(`User with ID ${resolvedUserId} not found.`);\n }\n userNickname = userResponse.users.find(\n (u) => u.id === resolvedUserId,\n )?.nickname;\n if (!userNickname)\n throw new Error(`User with ID ${resolvedUserId} not found.`);\n }\n\n const searchResponse = await connpassClient.searchEvents({\n nickname: userNickname,\n ymdFrom: formatDateLabel(rangeStart),\n ymdTo: formatDateLabel(rangeEnd),\n order: EVENT_SORT_MAP[\"start-date-asc\"],\n count: maxEventsToFetch,\n });\n\n const eventsByDate = new Map<string, Event[]>();\n for (const event of searchResponse.events) {\n const eventDate = formatDateLabel(\n startOfDay(new Date(event.startedAt)),\n );\n if (!eventsByDate.has(eventDate)) {\n eventsByDate.set(eventDate, []);\n }\n eventsByDate.get(eventDate)!.push(event);\n }\n\n const sortedDates = Array.from(eventsByDate.keys()).sort();\n const sections = sortedDates.map((date) => ({\n date,\n events: formatEventList(\n eventsByDate.get(date)!,\n parsed.includeDetails\n ? FORMAT_PRESETS.detailed\n : FORMAT_PRESETS.default,\n ),\n }));\n\n const result = {\n userId: resolvedUserId,\n sections,\n metadata: {\n fromDate: formatDateLabel(rangeStart),\n toDate: formatDateLabel(rangeEnd),\n inspected: searchResponse.eventsReturned,\n limit: maxEventsToFetch,\n },\n };\n\n return {\n content: [\n {\n type: \"text\" as const,\n text: summarizeEventsResponse(\n {\n returned: sections.reduce(\n (count, section) => count + section.events.length,\n 0,\n ),\n available: sections.reduce(\n (count, section) => count + section.events.length,\n 0,\n ),\n start: 1,\n events: sections.flatMap((section) => section.events),\n },\n \"schedule\",\n ),\n },\n ],\n structuredContent: {\n kind: \"schedule\",\n data: result,\n },\n };\n },\n );\n}\n","import { z } from \"zod\";\nimport {\n formatEvent,\n formatPresentationsResponse,\n} from \"../utils/formatting.js\";\nimport { registerAppToolIfEnabled } from \"../utils/registration.js\";\nimport { connpassResourceUri } from \"../utils/resource.js\";\nimport type { ToolDeps } from \"../utils/types.js\";\n\nconst UIEventDetailInputSchema = z.object({\n eventId: z.number().int().positive(),\n});\n\nexport function registerUIEventDetailTool(deps: ToolDeps): void {\n const { server, connpassClient } = deps;\n\n registerAppToolIfEnabled(\n server,\n \"_get_event_detail\",\n {\n title: \"Get Event Detail (UI)\",\n description: \"Internal: fetch full event details for the detail view\",\n inputSchema: UIEventDetailInputSchema,\n _meta: {\n ui: {\n resourceUri: connpassResourceUri,\n visibility: [\"app\"],\n },\n },\n },\n async (args: Record<string, unknown>) => {\n const { eventId } = UIEventDetailInputSchema.parse(args ?? {});\n\n const [eventsResponse, presentationsResponse] = await Promise.all([\n connpassClient.searchEvents({\n eventId: [eventId],\n count: 1,\n }),\n connpassClient.getEventPresentations(eventId).catch(() => undefined),\n ]);\n\n const event = eventsResponse.events[0];\n if (!event) {\n throw new Error(`Event with ID ${eventId} not found.`);\n }\n\n const formatted = formatEvent(event);\n const presentations = presentationsResponse\n ? formatPresentationsResponse(presentationsResponse)\n : undefined;\n const detailEvent = presentations?.presentations?.length\n ? { ...formatted, presentations: presentations.presentations }\n : formatted;\n\n return {\n content: [{ type: \"text\" as const, text: JSON.stringify(detailEvent) }],\n structuredContent: {\n kind: \"event-detail\",\n data: {\n event: detailEvent,\n presentations,\n },\n },\n };\n },\n );\n}\n","import type { Event } from \"@kajidog/connpass-api-client\";\nimport { z } from \"zod\";\nimport { formatEventList } from \"../utils/formatting.js\";\nimport { registerAppToolIfEnabled } from \"../utils/registration.js\";\nimport { connpassResourceUri } from \"../utils/resource.js\";\nimport { EVENT_SORT_MAP, parseHyphenatedDate } from \"../utils/shared.js\";\nimport type { ToolDeps } from \"../utils/types.js\";\n\nconst UIScheduleInputSchema = z.object({\n userId: z.number().int().positive().optional(),\n nickname: z.string().min(1).optional(),\n fromDate: z.string().min(1).optional(),\n toDate: z.string().min(1).optional(),\n maxEvents: z.number().int().min(1).max(100).optional(),\n});\n\nfunction startOfDay(date: Date): Date {\n return new Date(date.getFullYear(), date.getMonth(), date.getDate());\n}\n\nfunction formatDateLabel(date: Date): string {\n const year = date.getFullYear();\n const month = String(date.getMonth() + 1).padStart(2, \"0\");\n const day = String(date.getDate()).padStart(2, \"0\");\n return `${year}-${month}-${day}`;\n}\n\nconst SCHEDULE_FORMAT_OPTIONS = {\n descriptionLimit: 0 as const,\n catchPhraseLimit: 100,\n};\n\n// userId -> nickname キャッシュ\nconst userNicknameCache = new Map<number, string>();\n\nexport function registerUIScheduleTool(deps: ToolDeps): void {\n const { server, connpassClient, config } = deps;\n\n registerAppToolIfEnabled(\n server,\n \"_search_schedule\",\n {\n title: \"Search Schedule (UI)\",\n description: \"Internal: re-search schedule from UI\",\n inputSchema: UIScheduleInputSchema,\n _meta: {\n ui: {\n resourceUri: connpassResourceUri,\n visibility: [\"app\"],\n },\n },\n },\n async (args: Record<string, unknown>) => {\n const parsed = UIScheduleInputSchema.parse(args ?? {});\n\n let resolvedUserId: number | undefined =\n parsed.userId ?? config.defaultUserId;\n let userNickname: string | undefined;\n\n if (parsed.nickname) {\n const userSearchResponse = await connpassClient.searchUsers({\n nickname: parsed.nickname,\n });\n if (userSearchResponse.users.length === 0) {\n throw new Error(`User with nickname \"${parsed.nickname}\" not found.`);\n }\n resolvedUserId = userSearchResponse.users[0].id;\n userNickname = userSearchResponse.users[0].nickname;\n userNicknameCache.set(resolvedUserId, userNickname);\n }\n\n if (!resolvedUserId) {\n throw new Error(\"User ID or nickname is required.\");\n }\n\n const maxEventsToFetch = parsed.maxEvents ?? 30;\n\n const today = startOfDay(new Date());\n const rangeStart = parsed.fromDate\n ? startOfDay(new Date(parseHyphenatedDate(parsed.fromDate)))\n : today;\n const rangeEnd = parsed.toDate\n ? startOfDay(new Date(parseHyphenatedDate(parsed.toDate)))\n : (() => {\n const defaultEnd = new Date(rangeStart);\n defaultEnd.setDate(defaultEnd.getDate() + 7);\n return defaultEnd;\n })();\n rangeEnd.setHours(23, 59, 59, 999);\n\n if (!userNickname) {\n // キャッシュからニックネームを取得\n userNickname = userNicknameCache.get(resolvedUserId);\n if (!userNickname) {\n const userResponse = await connpassClient.searchUsers({\n userId: [resolvedUserId],\n });\n if (userResponse.users.length === 0)\n throw new Error(`User with ID ${resolvedUserId} not found.`);\n userNickname = userResponse.users.find(\n (u) => u.id === resolvedUserId,\n )?.nickname;\n if (!userNickname)\n throw new Error(`User with ID ${resolvedUserId} not found.`);\n userNicknameCache.set(resolvedUserId, userNickname);\n }\n }\n\n const searchResponse = await connpassClient.searchEvents({\n nickname: userNickname,\n ymdFrom: formatDateLabel(rangeStart),\n ymdTo: formatDateLabel(rangeEnd),\n order: EVENT_SORT_MAP[\"start-date-asc\"],\n count: maxEventsToFetch,\n });\n\n const eventsByDate = new Map<string, Event[]>();\n for (const event of searchResponse.events) {\n const eventDate = formatDateLabel(\n startOfDay(new Date(event.startedAt)),\n );\n if (!eventsByDate.has(eventDate)) eventsByDate.set(eventDate, []);\n eventsByDate.get(eventDate)!.push(event);\n }\n\n const sortedDates = Array.from(eventsByDate.keys()).sort();\n const sections = sortedDates.map((date) => ({\n date,\n events: formatEventList(\n eventsByDate.get(date)!,\n SCHEDULE_FORMAT_OPTIONS,\n ),\n }));\n\n const result = {\n userId: resolvedUserId,\n sections,\n metadata: {\n fromDate: formatDateLabel(rangeStart),\n toDate: formatDateLabel(rangeEnd),\n inspected: searchResponse.eventsReturned,\n limit: maxEventsToFetch,\n },\n };\n\n return {\n content: [{ type: \"text\" as const, text: JSON.stringify(result) }],\n structuredContent: {\n kind: \"schedule\",\n data: result,\n },\n };\n },\n );\n}\n","import type { Event } from \"@kajidog/connpass-api-client\";\nimport { z } from \"zod\";\nimport { resolvePrefectureInputs } from \"../prefectures.js\";\nimport { formatEvent, formatEventsResponse } from \"../utils/formatting.js\";\nimport { registerAppToolIfEnabled } from \"../utils/registration.js\";\nimport { connpassResourceUri } from \"../utils/resource.js\";\nimport {\n EVENT_SORT_MAP,\n applyPagination,\n normalizeKeywordOr,\n normalizeStringArray,\n parseHyphenatedDate,\n toYmdArray,\n} from \"../utils/shared.js\";\nimport type { ToolDeps } from \"../utils/types.js\";\n\nconst UI_LIST_FORMAT_OPTIONS = {\n descriptionLimit: 0,\n catchPhraseLimit: 100,\n} as const;\n\nconst UIEventSearchInputSchema = z.object({\n query: z.string().min(1).optional(),\n anyQuery: z.string().min(1).optional(),\n on: z.union([z.string().min(1), z.array(z.string().min(1))]).optional(),\n from: z.string().min(1).optional(),\n to: z.string().min(1).optional(),\n participantNickname: z.string().min(1).optional(),\n hostNickname: z.string().min(1).optional(),\n groupIds: z.array(z.number()).optional(),\n prefectures: z\n .union([z.string().min(1), z.array(z.string().min(1))])\n .optional(),\n companyQuery: z.string().min(1).optional(),\n minAccepted: z.number().int().min(0).optional(),\n maxAccepted: z.number().int().min(0).optional(),\n minCapacity: z.number().int().min(0).optional(),\n maxCapacity: z.number().int().min(0).optional(),\n page: z.number().int().min(1).optional(),\n pageSize: z.number().int().min(1).max(100).optional(),\n sort: z\n .enum([\"start-date-asc\", \"participant-count-desc\", \"title-asc\"])\n .optional(),\n});\n\nfunction includesNormalized(\n targets: Array<string | undefined>,\n query?: string,\n): boolean {\n if (!query) return true;\n const normalizedQuery = query.trim().toLowerCase();\n if (!normalizedQuery) return true;\n return targets.some((value) =>\n value?.toLowerCase().includes(normalizedQuery),\n );\n}\n\nfunction applyLocalFilters(\n events: Event[],\n params: z.infer<typeof UIEventSearchInputSchema>,\n) {\n return events.filter((event) => {\n if (\n !includesNormalized(\n [event.ownerDisplayName, event.groupTitle],\n params.companyQuery,\n )\n ) {\n return false;\n }\n if (\n typeof params.minAccepted === \"number\" &&\n event.participantCount < params.minAccepted\n ) {\n return false;\n }\n if (\n typeof params.maxAccepted === \"number\" &&\n event.participantCount > params.maxAccepted\n ) {\n return false;\n }\n if (typeof params.minCapacity === \"number\") {\n const limit = typeof event.limit === \"number\" ? event.limit : 0;\n if (limit < params.minCapacity) return false;\n }\n if (typeof params.maxCapacity === \"number\") {\n const limit = typeof event.limit === \"number\" ? event.limit : 0;\n if (limit > params.maxCapacity) return false;\n }\n return true;\n });\n}\n\nfunction sortEvents(\n events: Event[],\n sort: z.infer<typeof UIEventSearchInputSchema>[\"sort\"],\n) {\n const copied = [...events];\n copied.sort((a, b) => {\n if (sort === \"participant-count-desc\") {\n return (\n b.participantCount - a.participantCount ||\n a.startedAt.localeCompare(b.startedAt)\n );\n }\n if (sort === \"title-asc\") {\n return (\n a.title.localeCompare(b.title, \"ja\") ||\n a.startedAt.localeCompare(b.startedAt)\n );\n }\n return a.startedAt.localeCompare(b.startedAt);\n });\n return copied;\n}\n\nexport function registerUISearchTool(deps: ToolDeps): void {\n const { server, connpassClient } = deps;\n\n registerAppToolIfEnabled(\n server,\n \"_search_events\",\n {\n title: \"Search Events (UI)\",\n description: \"Internal: re-search events from UI\",\n inputSchema: UIEventSearchInputSchema,\n _meta: {\n ui: {\n resourceUri: connpassResourceUri,\n visibility: [\"app\"],\n },\n },\n },\n async (args: Record<string, unknown>) => {\n const params = UIEventSearchInputSchema.parse(args ?? {});\n const useLocalFiltering =\n Boolean(params.companyQuery) ||\n typeof params.minAccepted === \"number\" ||\n typeof params.maxAccepted === \"number\" ||\n typeof params.minCapacity === \"number\" ||\n typeof params.maxCapacity === \"number\" ||\n params.sort === \"participant-count-desc\" ||\n params.sort === \"title-asc\";\n\n const pagination = useLocalFiltering\n ? {}\n : applyPagination(params.page, params.pageSize);\n const resolved = resolvePrefectureInputs(params.prefectures);\n if (\"response\" in resolved) {\n return {\n ...resolved.response,\n isError: true,\n };\n }\n const searchParams = {\n keyword: params.query,\n keywordOr: normalizeKeywordOr(params.anyQuery),\n ymd: toYmdArray(params.on),\n ymdFrom: params.from ? parseHyphenatedDate(params.from) : undefined,\n ymdTo: params.to ? parseHyphenatedDate(params.to) : undefined,\n nickname: params.participantNickname,\n ownerNickname: params.hostNickname,\n groupId: params.groupIds,\n prefecture:\n resolved.prefectures ?? normalizeStringArray(params.prefectures),\n order: EVENT_SORT_MAP[\"start-date-asc\"],\n ...pagination,\n };\n\n if (!useLocalFiltering) {\n const response = await connpassClient.searchEvents(searchParams);\n const formatted = formatEventsResponse(\n response,\n UI_LIST_FORMAT_OPTIONS,\n );\n return {\n content: [{ type: \"text\" as const, text: JSON.stringify(formatted) }],\n structuredContent: {\n kind: \"events\",\n data: formatted,\n },\n };\n }\n\n const response = await connpassClient.getAllEvents(searchParams);\n const filtered = sortEvents(\n applyLocalFilters(response.events, params),\n params.sort,\n );\n const pageSize = params.pageSize ?? 20;\n const page = params.page ?? 1;\n const sliceStart = (page - 1) * pageSize;\n const paged = filtered.slice(sliceStart, sliceStart + pageSize);\n const formatted = {\n returned: paged.length,\n available: filtered.length,\n start: filtered.length === 0 ? 0 : sliceStart + 1,\n events: paged.map((event) =>\n formatEvent(event, UI_LIST_FORMAT_OPTIONS),\n ),\n };\n return {\n content: [{ type: \"text\" as const, text: JSON.stringify(formatted) }],\n structuredContent: {\n kind: \"events\",\n data: formatted,\n },\n };\n },\n );\n}\n","import type { ToolDeps } from \"../utils/types.js\";\nimport { registerUIEventDetailTool } from \"./detail.js\";\nimport { registerUIScheduleTool } from \"./schedule.js\";\nimport { registerUISearchTool } from \"./search.js\";\n\nexport function registerUITools(deps: ToolDeps): void {\n registerUIEventDetailTool(deps);\n registerUISearchTool(deps);\n registerUIScheduleTool(deps);\n}\n","import { z } from \"zod\";\nimport {\n FORMAT_PRESETS,\n formatEventsResponse,\n summarizeEventsResponse,\n summarizeGroupsResponse,\n summarizeUsersResponse,\n} from \"./utils/formatting.js\";\nimport { registerAppToolIfEnabled } from \"./utils/registration.js\";\nimport {\n EVENT_SORT_KEYS,\n EVENT_SORT_MAP,\n USER_SORT_KEYS,\n USER_SORT_MAP,\n applyPagination,\n} from \"./utils/shared.js\";\nimport type { ToolDeps } from \"./utils/types.js\";\n\nconst UserSearchInputSchema = z.object({\n nickname: z\n .string()\n .min(1)\n .describe(\"Match users by nickname (substring search)\")\n .optional(),\n userIds: z\n .array(z.number())\n .describe(\"Limit to specific user IDs\")\n .optional(),\n page: z.number().int().min(1).describe(\"1-based page number\").optional(),\n pageSize: z\n .number()\n .int()\n .min(1)\n .max(100)\n .describe(\"Users per page (default 20)\")\n .optional(),\n sort: z\n .enum(USER_SORT_KEYS)\n .describe(\"Ranking by activity, followers, or recency\")\n .optional(),\n});\n\nconst UserGroupsInputSchema = z.object({\n userId: z.number().int().positive().describe(\"Connpass user ID\"),\n limit: z\n .number()\n .int()\n .min(1)\n .max(100)\n .describe(\"How many to return (default 20)\")\n .optional(),\n page: z.number().int().min(1).describe(\"1-based page number\").optional(),\n});\n\nconst UserRelationshipInputSchema = z.object({\n userId: z.number().int().positive().describe(\"Connpass user ID\"),\n limit: z\n .number()\n .int()\n .min(1)\n .max(100)\n .describe(\"How many to return (default 20)\")\n .optional(),\n page: z.number().int().min(1).describe(\"1-based page number\").optional(),\n sort: z\n .enum(EVENT_SORT_KEYS)\n .describe(\"Sort events by schedule or recency\")\n .optional(),\n includeDetails: z\n .boolean()\n .describe(\n \"Include event description (up to 200 chars). Use when you need content details for recommendations.\",\n )\n .default(false)\n .optional(),\n});\n\ntype UserRelationshipInput = z.infer<typeof UserRelationshipInputSchema>;\n\nfunction buildUserRelationshipParams(input: UserRelationshipInput) {\n const pagination = applyPagination(input.page, input.limit, {\n includePagination: true,\n });\n return {\n pagination,\n order: input.sort ? EVENT_SORT_MAP[input.sort] : undefined,\n };\n}\n\nexport function registerUserTools(deps: ToolDeps): void {\n const { server, connpassClient } = deps;\n\n registerAppToolIfEnabled(\n server,\n \"search_users\",\n {\n title: \"Search Users\",\n description: \"Discover Connpass users\",\n inputSchema: UserSearchInputSchema,\n },\n async (args: Record<string, unknown>) => {\n const params = UserSearchInputSchema.parse(args ?? {});\n const pagination = applyPagination(params.page, params.pageSize);\n const response = await connpassClient.searchUsers({\n nickname: params.nickname,\n userId: params.userIds,\n order: params.sort ? USER_SORT_MAP[params.sort] : undefined,\n ...pagination,\n });\n return {\n content: [\n { type: \"text\" as const, text: summarizeUsersResponse(response) },\n ],\n };\n },\n );\n\n registerAppToolIfEnabled(\n server,\n \"get_user_groups\",\n {\n title: \"Get User Groups\",\n description: \"List the groups a user belongs to\",\n inputSchema: UserGroupsInputSchema,\n },\n async (args: Record<string, unknown>) => {\n const { userId, limit, page } = UserGroupsInputSchema.parse(args ?? {});\n const pagination = applyPagination(page, limit);\n const response = await connpassClient.getUserGroups(userId, pagination);\n return {\n content: [\n { type: \"text\" as const, text: summarizeGroupsResponse(response) },\n ],\n };\n },\n );\n\n registerAppToolIfEnabled(\n server,\n \"get_user_attended_events\",\n {\n title: \"Get User Attended Events\",\n description: \"List events that a user has attended\",\n inputSchema: UserRelationshipInputSchema,\n },\n async (args: Record<string, unknown>) => {\n const parsed = UserRelationshipInputSchema.parse(args ?? {});\n const { pagination, order } = buildUserRelationshipParams(parsed);\n const response = await connpassClient.getUserAttendedEvents(\n parsed.userId,\n { ...pagination, order },\n );\n const formatOptions = parsed.includeDetails\n ? FORMAT_PRESETS.detailed\n : FORMAT_PRESETS.default;\n const formatted = formatEventsResponse(response, formatOptions);\n return {\n content: [\n {\n type: \"text\" as const,\n text: summarizeEventsResponse(formatted, \"attended events\"),\n },\n ],\n };\n },\n );\n\n registerAppToolIfEnabled(\n server,\n \"get_user_presenter_events\",\n {\n title: \"Get User Presenter Events\",\n description: \"List events where the user presented\",\n inputSchema: UserRelationshipInputSchema,\n },\n async (args: Record<string, unknown>) => {\n const parsed = UserRelationshipInputSchema.parse(args ?? {});\n const { pagination, order } = buildUserRelationshipParams(parsed);\n const response = await connpassClient.getUserPresenterEvents(\n parsed.userId,\n { ...pagination, order },\n );\n const formatOptions = parsed.includeDetails\n ? FORMAT_PRESETS.detailed\n : FORMAT_PRESETS.default;\n const formatted = formatEventsResponse(response, formatOptions);\n return {\n content: [\n {\n type: \"text\" as const,\n text: summarizeEventsResponse(formatted, \"presenter events\"),\n },\n ],\n };\n },\n );\n}\n","import { registerEventTools } from \"./events.js\";\nimport { registerGroupTools } from \"./groups.js\";\nimport { registerPrefectureTools } from \"./prefectures.js\";\nimport { registerScheduleTools } from \"./schedule.js\";\nimport { registerUITools } from \"./ui-tools/index.js\";\nimport { registerUserTools } from \"./users.js\";\nimport { registerConnpassResource } from \"./utils/resource.js\";\nimport type { ToolDeps } from \"./utils/types.js\";\n\nexport function registerAllTools(deps: ToolDeps): void {\n // UIリソース登録\n registerConnpassResource(deps);\n\n // 公開ツール登録\n registerEventTools(deps);\n registerScheduleTools(deps);\n registerGroupTools(deps);\n registerPrefectureTools(deps);\n registerUserTools(deps);\n\n // UI内部ツール登録\n registerUITools(deps);\n}\n","import { randomUUID } from \"node:crypto\";\nimport type { FormattedEventsResponse } from \"./formatting.js\";\n\ninterface SearchSessionEntry {\n expiresAt: number;\n result: FormattedEventsResponse;\n}\n\nexport interface SearchSessionStoreOptions {\n ttlMs?: number;\n}\n\nexport class SearchSessionStore {\n private readonly ttlMs: number;\n private readonly sessions = new Map<string, SearchSessionEntry>();\n\n constructor(options: SearchSessionStoreOptions = {}) {\n this.ttlMs = options.ttlMs ?? 10 * 60 * 1000;\n }\n\n save(result: FormattedEventsResponse): string {\n this.cleanup();\n const sessionId = randomUUID();\n this.sessions.set(sessionId, {\n expiresAt: Date.now() + this.ttlMs,\n result,\n });\n return sessionId;\n }\n\n get(sessionId: string): FormattedEventsResponse | undefined {\n this.cleanup();\n const entry = this.sessions.get(sessionId);\n if (!entry) {\n return undefined;\n }\n\n if (entry.expiresAt <= Date.now()) {\n this.sessions.delete(sessionId);\n return undefined;\n }\n\n return entry.result;\n }\n\n private cleanup(): void {\n const now = Date.now();\n for (const [sessionId, entry] of this.sessions.entries()) {\n if (entry.expiresAt <= now) {\n this.sessions.delete(sessionId);\n }\n }\n }\n}\n","#!/usr/bin/env node\n\nimport { connectStdio } from \"@kajidog/mcp-core\";\nimport { getServer } from \"./server.js\";\n\nconnectStdio(getServer()).catch(() => {\n process.exit(1);\n});\n"],"mappings":"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","names":["config","match","server","z","server","config","server","server","z","server","z","z","server","z","z","server","config","z","z","server","z","z","startOfDay","formatDateLabel","server","config","z","z","server","response","formatted","z","z","server","randomUUID","server"]}
package/package.json CHANGED
@@ -1,34 +1,37 @@
1
1
  {
2
2
  "name": "@kajidog/connpass-mcp-server",
3
- "version": "0.3.0",
4
- "description": "MCP server for Connpass API integration",
5
- "main": "dist/index.js",
6
- "types": "dist/index.d.ts",
3
+ "version": "0.4.1",
4
+ "type": "module",
5
+ "description": "MCP server for Connpass API integration with MCP Apps Extension support",
6
+ "main": "./dist/index.js",
7
7
  "bin": {
8
8
  "connpass-mcp-server": "dist/index.js"
9
9
  },
10
10
  "dependencies": {
11
- "@modelcontextprotocol/sdk": "^1.0.0",
12
- "zod": "^3.22.0",
13
- "@kajidog/connpass-api-client": "0.3.0"
11
+ "@modelcontextprotocol/ext-apps": "^1.0.0",
12
+ "@modelcontextprotocol/sdk": "^1.26.0",
13
+ "zod": "^4.3.6",
14
+ "@kajidog/connpass-api-client": "0.4.0"
14
15
  },
15
16
  "devDependencies": {
16
17
  "@types/node": "^20.0.0",
18
+ "tsup": "^8.5.0",
19
+ "tsx": "^4.19.0",
17
20
  "typescript": "^5.0.0",
18
- "vitest": "^1.6.0",
19
- "@kajidog/connpass-widgets": "0.1.0"
21
+ "@kajidog/mcp-core": "0.1.0",
22
+ "@kajidog/connpass-ui": "0.1.0"
20
23
  },
21
24
  "keywords": [
22
25
  "mcp",
23
26
  "mcp-server",
24
- "connpass"
27
+ "connpass",
28
+ "mcp-apps"
25
29
  ],
26
- "type": "commonjs",
27
30
  "license": "MIT",
28
31
  "repository": {
29
32
  "type": "git",
30
33
  "url": "https://github.com/kajidog/connpass-mcp-server.git",
31
- "directory": "packages/mcp-server"
34
+ "directory": "apps/mcp-server"
32
35
  },
33
36
  "bugs": {
34
37
  "url": "https://github.com/kajidog/connpass-mcp-server/issues"
@@ -44,16 +47,13 @@
44
47
  "publishConfig": {
45
48
  "access": "public"
46
49
  },
47
- "sideEffects": false,
48
50
  "exports": {
49
51
  ".": "./dist/index.js"
50
52
  },
51
53
  "scripts": {
52
- "build": "tsc -b",
53
- "dev": "tsc -b --watch",
54
54
  "start": "node dist/index.js",
55
- "lint": "biome lint src",
56
- "typecheck": "tsc --noEmit",
57
- "test": "vitest run"
55
+ "dev": "NODE_ENV=development tsx src/index.ts",
56
+ "dev:http": "NODE_ENV=development MCP_HTTP_MODE=true tsx src/index.ts",
57
+ "build": "tsup"
58
58
  }
59
59
  }
@@ -1,8 +0,0 @@
1
- import type { CallToolResult } from "@modelcontextprotocol/sdk/types.js";
2
- export interface BuildCallToolResultOptions {
3
- toolName: string;
4
- result: unknown;
5
- includeAppsSdkPayload: boolean;
6
- }
7
- export declare function buildCallToolResult(options: BuildCallToolResultOptions): CallToolResult;
8
- //# sourceMappingURL=apps-sdk.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"apps-sdk.d.ts","sourceRoot":"","sources":["../src/apps-sdk.ts"],"names":[],"mappings":"AAAA,OAAO,KAAK,EAAE,cAAc,EAAE,MAAM,oCAAoC,CAAC;AAoBzE,MAAM,WAAW,0BAA0B;IACzC,QAAQ,EAAE,MAAM,CAAC;IACjB,MAAM,EAAE,OAAO,CAAC;IAChB,qBAAqB,EAAE,OAAO,CAAC;CAChC;AAmCD,wBAAgB,mBAAmB,CACjC,OAAO,EAAE,0BAA0B,GAClC,cAAc,CAyDhB"}
package/dist/apps-sdk.js DELETED
@@ -1,93 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.buildCallToolResult = buildCallToolResult;
4
- const index_js_1 = require("./widgets/index.js");
5
- const TOOL_KIND_MAP = new Map([
6
- ["search_events", "events"],
7
- ["get_event_presentations", "event_presentations"],
8
- ["search_schedule", "schedule"],
9
- ["search_groups", "groups"],
10
- ["search_users", "users"],
11
- ["get_user_groups", "groups"],
12
- ["get_user_attended_events", "events"],
13
- ["get_user_presenter_events", "events"],
14
- ]);
15
- const TOOL_WIDGET_CATEGORY_MAP = new Map([
16
- ["search_events", "carousel"],
17
- ["search_schedule", "list"],
18
- ]);
19
- function toDisplayText(result) {
20
- if (typeof result === "string") {
21
- return result;
22
- }
23
- if (result === undefined) {
24
- return "Tool returned no data.";
25
- }
26
- try {
27
- return JSON.stringify(result, null, 2);
28
- }
29
- catch (error) {
30
- return `Failed to serialize tool output: ${error instanceof Error ? error.message : String(error)}`;
31
- }
32
- }
33
- function inferResultKind(toolName) {
34
- return TOOL_KIND_MAP.get(toolName) ?? "generic";
35
- }
36
- function buildStructuredContent(toolName, result) {
37
- return {
38
- app: "connpass",
39
- tool: toolName,
40
- kind: inferResultKind(toolName),
41
- generatedAt: new Date().toISOString(),
42
- data: result ?? null,
43
- };
44
- }
45
- function buildCallToolResult(options) {
46
- const { toolName, result, includeAppsSdkPayload } = options;
47
- const templateUri = includeAppsSdkPayload
48
- ? (0, index_js_1.getWidgetTemplateForTool)(toolName)
49
- : undefined;
50
- const contentItem = {
51
- type: "text",
52
- text: toDisplayText(result),
53
- };
54
- const base = { content: [contentItem] };
55
- if (includeAppsSdkPayload) {
56
- console.log(`[apps-sdk] tool=${toolName} includeAppsSdkPayload=true template=${templateUri ?? "<missing>"}`);
57
- base.structuredContent = buildStructuredContent(toolName, result);
58
- if (templateUri) {
59
- const widgetCategory = TOOL_WIDGET_CATEGORY_MAP.get(toolName) ?? "carousel";
60
- const widgetMeta = {
61
- "openai/outputTemplate": templateUri,
62
- "openai/resultCanProduceWidget": true,
63
- "openai/widgetAccessible": true,
64
- "openai/widgetCategory": widgetCategory,
65
- "openai/widgetDomain": "connpass",
66
- "openai/widgetCSP": {
67
- connect_domains: ["https://connpass.com"],
68
- resource_domains: [
69
- "https://connpass.com",
70
- "https://media.connpass.com",
71
- ],
72
- redirect_domains: ["https://connpass.com"],
73
- },
74
- };
75
- // ★ 重要:content アイテム側にも _meta を付ける
76
- contentItem._meta = {
77
- ...(contentItem._meta ?? {}),
78
- ...widgetMeta,
79
- };
80
- // 互換性のため、従来通り top-level にも残しておく(不要なら削除OK)
81
- base._meta = {
82
- ...(base._meta ?? {}),
83
- ...widgetMeta,
84
- };
85
- console.log("[apps-sdk] widget meta", contentItem._meta);
86
- }
87
- else {
88
- console.warn(`[apps-sdk] tool=${toolName} has no widget template mapping; UI will not render.`);
89
- }
90
- }
91
- return base;
92
- }
93
- //# sourceMappingURL=apps-sdk.js.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"apps-sdk.js","sourceRoot":"","sources":["../src/apps-sdk.ts"],"names":[],"mappings":"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"}
package/dist/config.d.ts DELETED
@@ -1,7 +0,0 @@
1
- export declare function getDefaultUserId(): number | undefined;
2
- export declare function getDefaultIncludePresentations(): boolean | undefined;
3
- export declare function isAppsSdkOutputEnabled(): boolean;
4
- export declare function getRateLimitEnabled(): boolean | undefined;
5
- export declare function getRateLimitDelayMs(): number | undefined;
6
- export declare function getMcpBasePath(): string;
7
- //# sourceMappingURL=config.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"config.d.ts","sourceRoot":"","sources":["../src/config.ts"],"names":[],"mappings":"AAkFA,wBAAgB,gBAAgB,IAAI,MAAM,GAAG,SAAS,CAErD;AAED,wBAAgB,8BAA8B,IAAI,OAAO,GAAG,SAAS,CAEpE;AAED,wBAAgB,sBAAsB,IAAI,OAAO,CAEhD;AAED,wBAAgB,mBAAmB,IAAI,OAAO,GAAG,SAAS,CAEzD;AAED,wBAAgB,mBAAmB,IAAI,MAAM,GAAG,SAAS,CAExD;AAED,wBAAgB,cAAc,IAAI,MAAM,CAYvC"}