@loop_ouroboros/mcp-hub-lite 1.2.9 → 1.3.0
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/CHANGELOG.md +33 -0
- package/dist/client/assets/{HomeView-CGezWc0j.js → HomeView-Bi2bkUKf.js} +1 -1
- package/dist/client/assets/{ResourceDetailView-CDmWGdAK.css → ResourceDetailView-BkTSg91z.css} +1 -1
- package/dist/client/assets/ResourceDetailView-DyuSovH9.js +1 -0
- package/dist/client/assets/ResourcesView-CU0VbNy5.js +1 -0
- package/dist/client/assets/ResourcesView-zgV8Nq7w.css +1 -0
- package/dist/client/assets/{ServerDashboard-g5p4VC_-.js → ServerDashboard-BGyyZAti.js} +1 -1
- package/dist/client/assets/{ServerDetail-DCQH8HIb.css → ServerDetail-CPNAFBPM.css} +1 -1
- package/dist/client/assets/ServerDetail-bcQ8BVXR.js +2 -0
- package/dist/client/assets/{ServerListView-DZsy2gaQ.js → ServerListView-yQPVJFHG.js} +1 -1
- package/dist/client/assets/{ServerStatusTags.vue_vue_type_script_setup_true_lang-DmGg4uuV.js → ServerStatusTags.vue_vue_type_script_setup_true_lang-C8gQlxGE.js} +1 -1
- package/dist/client/assets/{SettingsView-DQSWb9xM.js → SettingsView-B1DxbFP3.js} +1 -1
- package/dist/client/assets/ToolCallDialog-BQ9UJZ_-.css +1 -0
- package/dist/client/assets/ToolCallDialog-DEapCO06.js +1 -0
- package/dist/client/assets/ToolsView-DA0u_bCw.js +1 -0
- package/dist/client/assets/ToolsView-cO61nMNr.css +1 -0
- package/dist/client/assets/{_baseClone-DsVtZfPm.js → _baseClone-B991Lvrt.js} +1 -1
- package/dist/client/assets/{el-form-item-CTsVV8sm.js → el-form-item-DfWq_kSy.js} +1 -1
- package/dist/client/assets/{el-input-Bh1VGJTU.js → el-input-5YzZrwir.js} +1 -1
- package/dist/client/assets/{el-loading-huOeK9cW.js → el-loading-DE3FcxNH.js} +1 -1
- package/dist/client/assets/{el-overlay-CR_KVhLU.js → el-overlay-BTeTueuN.js} +1 -1
- package/dist/client/assets/{el-radio-group-BSbtAW4k.js → el-radio-group-Y1E2bxIW.js} +1 -1
- package/dist/client/assets/{el-skeleton-item-BSxOLPFM.js → el-skeleton-item-DhgR50Jx.js} +1 -1
- package/dist/client/assets/{el-switch-BaQUQWTL.js → el-switch-fF--nMSD.js} +1 -1
- package/dist/client/assets/{el-tab-pane-9JxLgdS7.js → el-tab-pane-rvS_KTwP.js} +1 -1
- package/dist/client/assets/{el-table-column-Du1l9Ww3.js → el-table-column-B1O8mY47.js} +1 -1
- package/dist/client/assets/{index-BNmwPGMT.css → index-Bzz3tYbS.css} +1 -1
- package/dist/client/assets/{index-CsZoFRv1.js → index-DkqV9kH4.js} +2 -2
- package/dist/client/assets/{omit-Btci9mp3.js → omit-BIIebEYo.js} +1 -1
- package/dist/client/assets/{raf-tUu4BwZS.js → raf-Cj-gATZv.js} +1 -1
- package/dist/client/index.html +2 -2
- package/dist/server/shared/models/constants.d.ts +3 -0
- package/dist/server/shared/models/constants.d.ts.map +1 -0
- package/dist/server/shared/models/constants.js +2 -0
- package/dist/server/shared/models/index.d.ts +1 -0
- package/dist/server/shared/models/index.d.ts.map +1 -1
- package/dist/server/shared/models/index.js +1 -0
- package/dist/server/shared/types/index.d.ts +0 -1
- package/dist/server/shared/types/index.d.ts.map +1 -1
- package/dist/server/shared/types/index.js +0 -1
- package/dist/server/src/api/mcp/debug-response-wrapper.js +1 -1
- package/dist/server/src/api/mcp/gateway.d.ts.map +1 -1
- package/dist/server/src/api/mcp/gateway.js +23 -41
- package/dist/server/src/api/web/hub-tools.d.ts.map +1 -1
- package/dist/server/src/api/web/hub-tools.js +11 -0
- package/dist/server/src/api/web/mcp-status.js +2 -2
- package/dist/server/src/api/web/search.d.ts +2 -16
- package/dist/server/src/api/web/search.d.ts.map +1 -1
- package/dist/server/src/api/web/search.js +22 -30
- package/dist/server/src/api/web/servers.js +1 -1
- package/dist/server/src/api/ws/events.js +1 -1
- package/dist/server/src/api/ws/ws-handler.js +1 -1
- package/dist/server/src/app.js +1 -1
- package/dist/server/src/cli/commands/tool-use.d.ts +10 -3
- package/dist/server/src/cli/commands/tool-use.d.ts.map +1 -1
- package/dist/server/src/cli/commands/tool-use.js +69 -30
- package/dist/server/src/config/config-change-logger.js +1 -1
- package/dist/server/src/config/config-loader.js +1 -1
- package/dist/server/src/config/config-manager.js +1 -1
- package/dist/server/src/config/config-migrator.d.ts +4 -48
- package/dist/server/src/config/config-migrator.d.ts.map +1 -1
- package/dist/server/src/config/config-migrator.js +2 -103
- package/dist/server/src/config/config-saver.js +1 -1
- package/dist/server/src/config/server-config-manager.js +1 -1
- package/dist/server/src/models/system-tools.constants.d.ts +2 -1
- package/dist/server/src/models/system-tools.constants.d.ts.map +1 -1
- package/dist/server/src/models/system-tools.constants.js +2 -1
- package/dist/server/src/pid/manager.js +1 -1
- package/dist/server/src/server/dev-server.js +2 -2
- package/dist/server/src/server/runner.js +2 -2
- package/dist/server/src/server/startup.js +2 -2
- package/dist/server/src/services/connection/connection-manager.d.ts.map +1 -1
- package/dist/server/src/services/connection/connection-manager.js +16 -21
- package/dist/server/src/services/connection/tool-cache.d.ts.map +1 -1
- package/dist/server/src/services/connection/tool-cache.js +10 -8
- package/dist/server/src/services/event-bus.service.d.ts +3 -1
- package/dist/server/src/services/event-bus.service.d.ts.map +1 -1
- package/dist/server/src/services/event-bus.service.js +1 -0
- package/dist/server/src/services/gateway/gateway.service.d.ts +1 -0
- package/dist/server/src/services/gateway/gateway.service.d.ts.map +1 -1
- package/dist/server/src/services/gateway/gateway.service.js +29 -7
- package/dist/server/src/services/gateway/request-handlers/call-tool-handler.d.ts +1 -2
- package/dist/server/src/services/gateway/request-handlers/call-tool-handler.d.ts.map +1 -1
- package/dist/server/src/services/gateway/request-handlers/call-tool-handler.js +24 -13
- package/dist/server/src/services/gateway/request-handlers/resources-handler.d.ts.map +1 -1
- package/dist/server/src/services/gateway/request-handlers/resources-handler.js +7 -3
- package/dist/server/src/services/gateway/tool-list-generator.d.ts +14 -19
- package/dist/server/src/services/gateway/tool-list-generator.d.ts.map +1 -1
- package/dist/server/src/services/gateway/tool-list-generator.js +221 -80
- package/dist/server/src/services/hub-manager.service.d.ts.map +1 -1
- package/dist/server/src/services/hub-manager.service.js +15 -2
- package/dist/server/src/services/hub-tools/instance-selector.js +1 -1
- package/dist/server/src/services/hub-tools/resource-generator.d.ts +0 -21
- package/dist/server/src/services/hub-tools/resource-generator.d.ts.map +1 -1
- package/dist/server/src/services/hub-tools/resource-generator.js +15 -15
- package/dist/server/src/services/hub-tools/server-selector.js +1 -1
- package/dist/server/src/services/hub-tools.service.d.ts.map +1 -1
- package/dist/server/src/services/hub-tools.service.js +15 -12
- package/dist/server/src/services/log-storage.service.js +1 -1
- package/dist/server/src/services/system-tool-handler.js +1 -1
- package/dist/server/src/utils/error-handler.js +1 -1
- package/dist/server/src/utils/index.d.ts +1 -1
- package/dist/server/src/utils/index.d.ts.map +1 -1
- package/dist/server/src/utils/index.js +1 -1
- package/dist/server/src/utils/json-utils.js +4 -4
- package/dist/server/src/utils/log-rotator.d.ts +0 -15
- package/dist/server/src/utils/log-rotator.d.ts.map +1 -1
- package/dist/server/src/utils/log-rotator.js +0 -18
- package/dist/server/src/utils/port-checker.js +1 -1
- package/dist/server/src/utils/transports/stdio-transport.js +1 -1
- package/dist/server/src/utils/transports/streamable-http-transport.js +1 -1
- package/dist/server/src/utils/transports/transport-factory.d.ts.map +1 -1
- package/dist/server/src/utils/transports/transport-factory.js +26 -3
- package/dist/server/tests/contract/mcp-protocol/initialize.test.js +1 -1
- package/dist/server/tests/contract/mcp-protocol/tools-call.test.js +1 -1
- package/dist/server/tests/contract/mcp-protocol/tools-list.test.js +1 -1
- package/dist/server/tests/integration/gateway/fault-tolerance.test.js +1 -1
- package/dist/server/tests/integration/gateway/mcp-connection.test.js +1 -1
- package/dist/server/tests/types/logger-test-helpers.d.ts +1 -1
- package/dist/server/tests/types/logger-test-helpers.d.ts.map +1 -1
- package/dist/server/tests/unit/config/config-migrator.test.js +45 -105
- package/dist/server/tests/unit/config/config-saver.test.js +1 -1
- package/dist/server/tests/unit/server/runner.test.js +5 -6
- package/dist/server/tests/unit/services/gateway-logging.test.js +1 -1
- package/dist/server/tests/unit/services/hub-manager-service.test.js +4 -5
- package/dist/server/tests/unit/services/hub-tools.service.test.js +78 -2
- package/dist/server/tests/unit/utils/log-rotator.test.js +1 -15
- package/dist/server/tests/unit/utils/logger.test.js +1 -1
- package/package.json +1 -3
- package/dist/client/assets/ResourceDetailView-Bi5UsbFq.js +0 -1
- package/dist/client/assets/ResourcesView-B9anSm85.js +0 -1
- package/dist/client/assets/ResourcesView-Cc8RHtia.css +0 -1
- package/dist/client/assets/ServerDetail-DMoFqWCp.js +0 -2
- package/dist/client/assets/ToolCallDialog-BEyRp_J3.js +0 -1
- package/dist/client/assets/ToolCallDialog-BhdPX-Kf.css +0 -1
- package/dist/client/assets/ToolsView-BU7PKJwt.js +0 -1
- package/dist/client/assets/ToolsView-BkrQLjH9.css +0 -1
|
@@ -3,14 +3,13 @@
|
|
|
3
3
|
*/
|
|
4
4
|
import { CallToolRequestSchema, ListToolsRequestSchema, McpError } from '@modelcontextprotocol/sdk/types.js';
|
|
5
5
|
import { logger, LOG_MODULES } from '../../../utils/index.js';
|
|
6
|
-
import {
|
|
7
|
-
import { mcpConnectionManager } from '../../mcp-connection-manager.js';
|
|
6
|
+
import { hubToolsService } from '../../hub-tools.service.js';
|
|
8
7
|
import { SystemToolHandler } from '../../system-tool-handler.js';
|
|
9
8
|
import { ErrorHandler } from '../../../utils/error-handler.js';
|
|
10
9
|
import { ToolArgsParser } from '../../../utils/tool-args-parser.js';
|
|
11
10
|
import { SYSTEM_TOOL_NAMES, MCP_HUB_LITE_SERVER } from '../../../models/system-tools.constants.js';
|
|
12
11
|
import { formatToolArgs, formatToolResponse } from '../log-formatter.js';
|
|
13
|
-
import {
|
|
12
|
+
import { getOrBuildGatewayToolMap, getOrBuildGatewayToolsList } from '../tool-list-generator.js';
|
|
14
13
|
/**
|
|
15
14
|
* Type guard to check if a result is a valid CallToolResult.
|
|
16
15
|
*
|
|
@@ -37,7 +36,7 @@ async function executeSystemToolCall(toolName, toolArgs) {
|
|
|
37
36
|
content: [
|
|
38
37
|
{
|
|
39
38
|
type: 'text',
|
|
40
|
-
text:
|
|
39
|
+
text: JSON.stringify(result, null, 2)
|
|
41
40
|
}
|
|
42
41
|
]
|
|
43
42
|
};
|
|
@@ -75,10 +74,10 @@ function getSystemToolName(toolName) {
|
|
|
75
74
|
* @param server - MCP server instance to register handlers on
|
|
76
75
|
* @param toolMap - Tool map for routing tool calls
|
|
77
76
|
*/
|
|
78
|
-
export function registerCallToolHandler(server
|
|
77
|
+
export function registerCallToolHandler(server) {
|
|
79
78
|
// Original list tools handler (for compatibility)
|
|
80
79
|
server.server.setRequestHandler(ListToolsRequestSchema, async () => {
|
|
81
|
-
const gatewayTools =
|
|
80
|
+
const gatewayTools = getOrBuildGatewayToolsList();
|
|
82
81
|
return {
|
|
83
82
|
tools: gatewayTools
|
|
84
83
|
};
|
|
@@ -88,26 +87,38 @@ export function registerCallToolHandler(server, toolMap) {
|
|
|
88
87
|
const toolName = request.params.name;
|
|
89
88
|
const toolArgs = request.params.arguments || {};
|
|
90
89
|
const systemToolName = getSystemToolName(toolName);
|
|
90
|
+
const gwToolMap = getOrBuildGatewayToolMap();
|
|
91
91
|
// Log incoming tool request with full context
|
|
92
92
|
logger.info(`Tool call REQUEST received: toolName=${toolName}, args=${formatToolArgs(toolArgs)}`, LOG_MODULES.GATEWAY);
|
|
93
|
-
logger.debug(`Tool context: toolMap size=${
|
|
93
|
+
logger.debug(`Tool context: toolMap size=${gwToolMap.size}, available tools=${Array.from(gwToolMap.keys()).slice(0, 10).join(', ')}${gwToolMap.size > 10 ? '...' : ''}`, LOG_MODULES.GATEWAY);
|
|
94
94
|
// Handle system tools
|
|
95
95
|
if (systemToolName) {
|
|
96
96
|
logger.debug(`System tool called: ${systemToolName}, args=${formatToolArgs(toolArgs)}`, LOG_MODULES.GATEWAY);
|
|
97
97
|
return await executeSystemToolCall(systemToolName, toolArgs);
|
|
98
98
|
}
|
|
99
|
-
const target =
|
|
99
|
+
const target = gwToolMap.get(toolName);
|
|
100
100
|
logger.debug(`Tool lookup SUCCESS: toolName=${toolName} -> serverName=${target?.serverName}, serverIndex=${target?.serverIndex}, realToolName=${target?.realToolName}`, LOG_MODULES.GATEWAY);
|
|
101
101
|
if (!target) {
|
|
102
|
-
logger.error(`Tool NOT FOUND: toolName=${toolName}, available tools=${Array.from(
|
|
102
|
+
logger.error(`Tool NOT FOUND: toolName=${toolName}, available tools=${Array.from(gwToolMap.keys()).join(', ')}`, LOG_MODULES.GATEWAY);
|
|
103
103
|
throw new McpError(-32801, `Tool ${toolName} not found`);
|
|
104
104
|
}
|
|
105
|
+
// Extract wrapped arguments (LLM provides serverName/toolName per schema guidance)
|
|
106
|
+
const wrappedArgs = toolArgs;
|
|
107
|
+
const callServerName = wrappedArgs.serverName || target.serverName;
|
|
108
|
+
const callToolName = wrappedArgs.toolName || target.realToolName;
|
|
109
|
+
const callArgs = wrappedArgs.toolArgs || {};
|
|
110
|
+
const callOptions = wrappedArgs.requestOptions || {};
|
|
105
111
|
const startTime = Date.now();
|
|
106
112
|
try {
|
|
107
|
-
logger.debug(`Tool call EXECUTING: serverName=${
|
|
108
|
-
const result = await
|
|
113
|
+
logger.debug(`Tool call EXECUTING: serverName=${callServerName}, toolName=${callToolName}, args=${formatToolArgs(callArgs)}`, LOG_MODULES.GATEWAY);
|
|
114
|
+
const result = await hubToolsService.callTool({
|
|
115
|
+
serverName: callServerName,
|
|
116
|
+
toolName: callToolName,
|
|
117
|
+
toolArgs: callArgs,
|
|
118
|
+
requestOptions: callOptions
|
|
119
|
+
});
|
|
109
120
|
const duration = Date.now() - startTime;
|
|
110
|
-
logger.info(`Tool call SUCCESS: serverName=${
|
|
121
|
+
logger.info(`Tool call SUCCESS: serverName=${callServerName}, toolName=${callToolName}, duration=${duration}ms, response=${formatToolResponse(result)}`, LOG_MODULES.GATEWAY);
|
|
111
122
|
// Wrap the result in a valid CallToolResult structure
|
|
112
123
|
if (typeof result === 'object' && result !== null) {
|
|
113
124
|
return {
|
|
@@ -127,7 +138,7 @@ export function registerCallToolHandler(server, toolMap) {
|
|
|
127
138
|
}
|
|
128
139
|
}
|
|
129
140
|
catch (error) {
|
|
130
|
-
ErrorHandler.handleToolCallError(
|
|
141
|
+
ErrorHandler.handleToolCallError(callServerName, callToolName, error);
|
|
131
142
|
}
|
|
132
143
|
});
|
|
133
144
|
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"resources-handler.d.ts","sourceRoot":"","sources":["../../../../../../src/services/gateway/request-handlers/resources-handler.ts"],"names":[],"mappings":"AAAA;;GAEG;AAOH,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;
|
|
1
|
+
{"version":3,"file":"resources-handler.d.ts","sourceRoot":"","sources":["../../../../../../src/services/gateway/request-handlers/resources-handler.ts"],"names":[],"mappings":"AAAA;;GAEG;AAOH,OAAO,KAAK,EAAE,SAAS,EAAE,MAAM,yCAAyC,CAAC;AAKzE;;;;GAIG;AACH,wBAAgB,yBAAyB,CAAC,MAAM,EAAE,SAAS,GAAG,IAAI,CAiDjE"}
|
|
@@ -4,7 +4,6 @@
|
|
|
4
4
|
import { ListResourcesRequestSchema, ReadResourceRequestSchema, McpError } from '@modelcontextprotocol/sdk/types.js';
|
|
5
5
|
import { logger } from '../../../utils/index.js';
|
|
6
6
|
import { LOG_MODULES } from '../../../utils/logger/log-modules.js';
|
|
7
|
-
import { stringifyForLogging } from '../../../utils/json-utils.js';
|
|
8
7
|
import { hubToolsService } from '../../hub-tools.service.js';
|
|
9
8
|
/**
|
|
10
9
|
* Register resources-related handlers on the MCP server.
|
|
@@ -32,13 +31,18 @@ export function registerResourcesHandlers(server) {
|
|
|
32
31
|
try {
|
|
33
32
|
const { uri } = request.params;
|
|
34
33
|
const content = await hubToolsService.readResource(uri);
|
|
34
|
+
// Preserve original mimeType if content is already in MCP ReadResourceResult format
|
|
35
|
+
const mcpResult = content && typeof content === 'object' && 'contents' in content
|
|
36
|
+
? content
|
|
37
|
+
: null;
|
|
38
|
+
const mimeType = mcpResult?.contents?.[0]?.mimeType || 'application/json';
|
|
35
39
|
// Convert to official MCP format: contents array with required uri field
|
|
36
40
|
return {
|
|
37
41
|
contents: [
|
|
38
42
|
{
|
|
39
43
|
uri,
|
|
40
|
-
mimeType
|
|
41
|
-
text: typeof content === 'string' ? content :
|
|
44
|
+
mimeType,
|
|
45
|
+
text: typeof content === 'string' ? content : JSON.stringify(content, null, 2)
|
|
42
46
|
}
|
|
43
47
|
]
|
|
44
48
|
};
|
|
@@ -1,23 +1,18 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Gateway tool list generator with collision resolution.
|
|
2
|
+
* Gateway tool list generator with collision resolution and global caching.
|
|
3
3
|
*/
|
|
4
4
|
import type { ToolMapEntry, GatewayTool } from './types.js';
|
|
5
|
-
|
|
6
|
-
|
|
7
|
-
|
|
8
|
-
|
|
9
|
-
|
|
10
|
-
|
|
11
|
-
|
|
12
|
-
|
|
13
|
-
|
|
14
|
-
|
|
15
|
-
|
|
16
|
-
|
|
17
|
-
|
|
18
|
-
*
|
|
19
|
-
* @param {Map<string, ToolMapEntry>} toolMap - Map to populate with gateway tool name to actual tool mappings
|
|
20
|
-
* @returns {Array<GatewayTool>} Array of gateway tools with resolved names and descriptions
|
|
21
|
-
*/
|
|
22
|
-
export declare function generateGatewayToolsList(toolMap: Map<string, ToolMapEntry>): Array<GatewayTool>;
|
|
5
|
+
export declare function generateGatewayToolsList(toolMap: Map<string, ToolMapEntry>): GatewayTool[];
|
|
6
|
+
/** Lazy get — first-request fallback when cache hasn't been built yet */
|
|
7
|
+
export declare function getOrBuildGatewayToolMap(): Map<string, ToolMapEntry>;
|
|
8
|
+
/** Lazy get — first-request fallback when cache hasn't been built yet */
|
|
9
|
+
export declare function getOrBuildGatewayToolsList(): GatewayTool[];
|
|
10
|
+
/** Lazy get — returns only aggregated external tools (excludes system tools) */
|
|
11
|
+
export declare function getExternalGatewayTools(): GatewayTool[];
|
|
12
|
+
/** Full rebuild from global state — used by TOOLS_UPDATED / SERVER_DISCONNECTED */
|
|
13
|
+
export declare function rebuildFromScratch(): void;
|
|
14
|
+
/** Incremental add — update raw data layer, re-resolve names only */
|
|
15
|
+
export declare function addToCache(serverName: string, toolNames: string[]): void;
|
|
16
|
+
/** Incremental remove — update raw data layer, re-resolve names only */
|
|
17
|
+
export declare function removeFromCache(serverName: string, toolNames: string[]): void;
|
|
23
18
|
//# sourceMappingURL=tool-list-generator.d.ts.map
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"tool-list-generator.d.ts","sourceRoot":"","sources":["../../../../../src/services/gateway/tool-list-generator.ts"],"names":[],"mappings":"AAAA;;GAEG;AAQH,OAAO,KAAK,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;
|
|
1
|
+
{"version":3,"file":"tool-list-generator.d.ts","sourceRoot":"","sources":["../../../../../src/services/gateway/tool-list-generator.ts"],"names":[],"mappings":"AAAA;;GAEG;AAQH,OAAO,KAAK,EAAE,YAAY,EAAE,WAAW,EAAE,MAAM,YAAY,CAAC;AAwM5D,wBAAgB,wBAAwB,CAAC,OAAO,EAAE,GAAG,CAAC,MAAM,EAAE,YAAY,CAAC,GAAG,WAAW,EAAE,CAG1F;AAID,yEAAyE;AACzE,wBAAgB,wBAAwB,IAAI,GAAG,CAAC,MAAM,EAAE,YAAY,CAAC,CAKpE;AAED,yEAAyE;AACzE,wBAAgB,0BAA0B,IAAI,WAAW,EAAE,CAK1D;AAED,gFAAgF;AAChF,wBAAgB,uBAAuB,IAAI,WAAW,EAAE,CAEvD;AAED,mFAAmF;AACnF,wBAAgB,kBAAkB,IAAI,IAAI,CASzC;AAED,qEAAqE;AACrE,wBAAgB,UAAU,CAAC,UAAU,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,GAAG,IAAI,CAoDxE;AAED,wEAAwE;AACxE,wBAAgB,eAAe,CAAC,UAAU,EAAE,MAAM,EAAE,SAAS,EAAE,MAAM,EAAE,GAAG,IAAI,CAqC7E"}
|
|
@@ -1,35 +1,86 @@
|
|
|
1
1
|
/**
|
|
2
|
-
* Gateway tool list generator with collision resolution.
|
|
2
|
+
* Gateway tool list generator with collision resolution and global caching.
|
|
3
3
|
*/
|
|
4
|
-
import { logger } from '../../utils/index.js';
|
|
5
|
-
import {
|
|
6
|
-
import { mcpConnectionManager } from '../mcp-connection-manager.js';
|
|
4
|
+
import { logger, LOG_MODULES } from '../../utils/index.js';
|
|
5
|
+
import { mcpConnectionManager } from '../connection/index.js';
|
|
7
6
|
import { hubManager } from '../hub-manager.service.js';
|
|
8
|
-
import {
|
|
7
|
+
import { getSystemTools } from '../hub-tools/index.js';
|
|
9
8
|
import { parseCompositeKey } from '../../utils/composite-key.js';
|
|
9
|
+
// ── Double-layer cache ──────────────────────────────────────────
|
|
10
|
+
/** Raw data layer: per-server aggregated tool snapshots */
|
|
11
|
+
let rawToolData = null;
|
|
12
|
+
/** Resolved name layer */
|
|
13
|
+
let cachedToolMap = null;
|
|
14
|
+
let cachedGatewayTools = null;
|
|
15
|
+
// ── Phase 1: Data gathering ─────────────────────────────────────
|
|
10
16
|
/**
|
|
11
|
-
*
|
|
12
|
-
*
|
|
13
|
-
* This method creates a comprehensive tool list that includes both system tools and
|
|
14
|
-
* aggregated tools from all connected MCP servers. It implements sophisticated
|
|
15
|
-
* naming collision resolution by:
|
|
16
|
-
* - Adding server hash suffixes to non-unique tool names
|
|
17
|
-
* - Handling conflicts with system tool names
|
|
18
|
-
* - Ensuring names don't exceed 60 characters
|
|
19
|
-
* - Providing final uniqueness guarantees through counter suffixes
|
|
20
|
-
*
|
|
21
|
-
* The method also populates the provided toolMap with mappings from gateway tool names
|
|
22
|
-
* to actual server IDs and real tool names for efficient routing during tool execution.
|
|
23
|
-
*
|
|
24
|
-
* @param {Map<string, ToolMapEntry>} toolMap - Map to populate with gateway tool name to actual tool mappings
|
|
25
|
-
* @returns {Array<GatewayTool>} Array of gateway tools with resolved names and descriptions
|
|
17
|
+
* Collects aggregated tool data from all connected servers.
|
|
18
|
+
* Iterates tool cache entries and applies the aggregatedTools filter.
|
|
26
19
|
*/
|
|
27
|
-
|
|
28
|
-
const
|
|
20
|
+
function gatherRawToolData() {
|
|
21
|
+
const rawData = new Map();
|
|
22
|
+
for (const [compositeKey, tools] of mcpConnectionManager.getToolCacheEntries()) {
|
|
23
|
+
const parsed = parseCompositeKey(compositeKey);
|
|
24
|
+
const serverName = parsed ? parsed.serverName : compositeKey;
|
|
25
|
+
const serverIndex = parsed ? parsed.serverIndex : 0;
|
|
26
|
+
const serverConfig = hubManager.getServerByName(serverName);
|
|
27
|
+
if (!serverConfig)
|
|
28
|
+
continue;
|
|
29
|
+
const aggregatedTools = serverConfig.template?.aggregatedTools;
|
|
30
|
+
if (!aggregatedTools?.length)
|
|
31
|
+
continue;
|
|
32
|
+
const filteredTools = tools.filter((t) => aggregatedTools.includes(t.name));
|
|
33
|
+
if (filteredTools.length === 0)
|
|
34
|
+
continue;
|
|
35
|
+
const existing = rawData.get(serverName);
|
|
36
|
+
if (existing) {
|
|
37
|
+
// Merge tools from multiple instances — later tools overwrite earlier by name
|
|
38
|
+
const mergedTools = new Map();
|
|
39
|
+
for (const t of existing.tools)
|
|
40
|
+
mergedTools.set(t.name, t);
|
|
41
|
+
for (const t of filteredTools)
|
|
42
|
+
mergedTools.set(t.name, t);
|
|
43
|
+
// Update toolIndexes for tools from this instance
|
|
44
|
+
const mergedIndexes = new Map(existing.toolIndexes);
|
|
45
|
+
for (const t of filteredTools) {
|
|
46
|
+
mergedIndexes.set(t.name, serverIndex);
|
|
47
|
+
}
|
|
48
|
+
rawData.set(serverName, {
|
|
49
|
+
tools: Array.from(mergedTools.values()),
|
|
50
|
+
aggregated: aggregatedTools,
|
|
51
|
+
toolIndexes: mergedIndexes
|
|
52
|
+
});
|
|
53
|
+
}
|
|
54
|
+
else {
|
|
55
|
+
const toolIndexes = new Map();
|
|
56
|
+
for (const t of filteredTools) {
|
|
57
|
+
toolIndexes.set(t.name, serverIndex);
|
|
58
|
+
}
|
|
59
|
+
rawData.set(serverName, {
|
|
60
|
+
tools: filteredTools,
|
|
61
|
+
aggregated: aggregatedTools,
|
|
62
|
+
toolIndexes
|
|
63
|
+
});
|
|
64
|
+
}
|
|
65
|
+
}
|
|
66
|
+
let totalTools = 0;
|
|
67
|
+
for (const [, data] of rawData) {
|
|
68
|
+
totalTools += data.tools.length;
|
|
69
|
+
}
|
|
70
|
+
logger.debug(`Gateway tool cache: gathered ${totalTools} aggregated tools across ${rawData.size} servers`, LOG_MODULES.TOOL_LIST_GENERATOR);
|
|
71
|
+
return rawData;
|
|
72
|
+
}
|
|
73
|
+
// ── Phase 2: Name resolution ────────────────────────────────────
|
|
74
|
+
/**
|
|
75
|
+
* Resolves final gateway tool names from raw data.
|
|
76
|
+
* Pure computation — no external calls.
|
|
77
|
+
*/
|
|
78
|
+
function resolveToolNames(rawData, toolMap) {
|
|
29
79
|
toolMap.clear();
|
|
30
|
-
|
|
31
|
-
const systemTools = hubToolsService.getSystemTools();
|
|
80
|
+
const gatewayTools = [];
|
|
32
81
|
const usedNames = new Set();
|
|
82
|
+
// Add system tools
|
|
83
|
+
const systemTools = getSystemTools();
|
|
33
84
|
for (const tool of systemTools) {
|
|
34
85
|
gatewayTools.push({
|
|
35
86
|
name: tool.name,
|
|
@@ -39,72 +90,31 @@ export function generateGatewayToolsList(toolMap) {
|
|
|
39
90
|
});
|
|
40
91
|
usedNames.add(tool.name);
|
|
41
92
|
}
|
|
42
|
-
// First pass:
|
|
93
|
+
// First pass: count tool name frequencies
|
|
43
94
|
const toolNameCounts = new Map();
|
|
44
|
-
|
|
45
|
-
|
|
46
|
-
for (const tool of tools) {
|
|
47
|
-
// Parse composite key to get serverName
|
|
48
|
-
const parsed = parseCompositeKey(compositeKey);
|
|
49
|
-
const serverName = parsed ? parsed.serverName : compositeKey;
|
|
50
|
-
const serverConfig = hubManager.getServerByName(serverName);
|
|
51
|
-
if (serverConfig) {
|
|
52
|
-
// Only include tools if server has aggregatedTools configured AND tool is in aggregatedTools
|
|
53
|
-
const aggregatedTools = serverConfig.template?.aggregatedTools;
|
|
54
|
-
const hasAggregatedTools = aggregatedTools && aggregatedTools.length > 0;
|
|
55
|
-
if (!hasAggregatedTools) {
|
|
56
|
-
continue;
|
|
57
|
-
}
|
|
58
|
-
if (!Array.isArray(aggregatedTools) || !aggregatedTools.includes(tool.name)) {
|
|
59
|
-
continue;
|
|
60
|
-
}
|
|
61
|
-
}
|
|
95
|
+
for (const [, data] of rawData) {
|
|
96
|
+
for (const tool of data.tools) {
|
|
62
97
|
toolNameCounts.set(tool.name, (toolNameCounts.get(tool.name) || 0) + 1);
|
|
63
98
|
}
|
|
64
99
|
}
|
|
65
|
-
// Second pass:
|
|
66
|
-
for (const [
|
|
67
|
-
|
|
68
|
-
const parsed = parseCompositeKey(compositeKey);
|
|
69
|
-
const serverName = parsed ? parsed.serverName : compositeKey;
|
|
70
|
-
const serverIndex = parsed ? parsed.serverIndex : 0;
|
|
71
|
-
const serverConfig = hubManager.getServerByName(serverName);
|
|
72
|
-
// Skip if server configuration not found
|
|
73
|
-
if (!serverConfig) {
|
|
74
|
-
logger.warn(`Server configuration not found for serverName: ${serverName}, skipping tools`, LOG_MODULES.TOOL_LIST_GENERATOR);
|
|
75
|
-
continue;
|
|
76
|
-
}
|
|
77
|
-
// Only include tools if server has aggregatedTools configured
|
|
78
|
-
const aggregatedTools = serverConfig.template?.aggregatedTools;
|
|
79
|
-
const hasAggregatedTools = aggregatedTools && aggregatedTools.length > 0;
|
|
80
|
-
if (!hasAggregatedTools) {
|
|
81
|
-
continue;
|
|
82
|
-
}
|
|
83
|
-
// Get the instance ID for hash suffix (use compositeKey which contains serverName-serverIndex format)
|
|
84
|
-
const hashSuffix = compositeKey;
|
|
85
|
-
for (const tool of tools) {
|
|
86
|
-
if (!aggregatedTools.includes(tool.name)) {
|
|
87
|
-
continue;
|
|
88
|
-
}
|
|
100
|
+
// Second pass: generate gateway tools with collision resolution
|
|
101
|
+
for (const [serverName, data] of rawData) {
|
|
102
|
+
for (const tool of data.tools) {
|
|
89
103
|
let gatewayToolName = tool.name;
|
|
90
104
|
const isUnique = toolNameCounts.get(tool.name) === 1;
|
|
91
105
|
const isSystemConflict = usedNames.has(tool.name);
|
|
92
|
-
// If tool name is not unique or conflicts with system tool, append server hash
|
|
93
106
|
if (!isUnique || isSystemConflict) {
|
|
94
|
-
|
|
95
|
-
const hash = hashSuffix.substring(0, 4);
|
|
107
|
+
const hash = serverName.substring(0, 4);
|
|
96
108
|
gatewayToolName = `${tool.name}_${hash}`;
|
|
97
109
|
}
|
|
98
110
|
// Ensure name doesn't exceed 60 chars
|
|
99
111
|
if (gatewayToolName.length > 60) {
|
|
100
|
-
|
|
101
|
-
const hash = hashSuffix.substring(0, 4);
|
|
102
|
-
// Reserve space for hash and separator
|
|
112
|
+
const hash = serverName.substring(0, 4);
|
|
103
113
|
const maxToolNameLen = 60 - hash.length - 1;
|
|
104
114
|
const truncatedToolName = tool.name.substring(0, maxToolNameLen);
|
|
105
115
|
gatewayToolName = `${truncatedToolName}_${hash}`;
|
|
106
116
|
}
|
|
107
|
-
// Final uniqueness check
|
|
117
|
+
// Final uniqueness check
|
|
108
118
|
let finalName = gatewayToolName;
|
|
109
119
|
let counter = 1;
|
|
110
120
|
while (usedNames.has(finalName)) {
|
|
@@ -118,19 +128,150 @@ export function generateGatewayToolsList(toolMap) {
|
|
|
118
128
|
}
|
|
119
129
|
counter++;
|
|
120
130
|
}
|
|
121
|
-
|
|
122
|
-
|
|
123
|
-
toolMap.set(gatewayToolName, {
|
|
131
|
+
usedNames.add(finalName);
|
|
132
|
+
toolMap.set(finalName, {
|
|
124
133
|
serverName,
|
|
125
|
-
serverIndex,
|
|
134
|
+
serverIndex: data.toolIndexes.get(tool.name) ?? 0,
|
|
126
135
|
realToolName: tool.name
|
|
127
136
|
});
|
|
128
137
|
gatewayTools.push({
|
|
129
|
-
name:
|
|
138
|
+
name: finalName,
|
|
130
139
|
description: `[From ${serverName}] ${tool.description || ''}`,
|
|
131
|
-
inputSchema:
|
|
140
|
+
inputSchema: {
|
|
141
|
+
type: 'object',
|
|
142
|
+
properties: {
|
|
143
|
+
serverName: {
|
|
144
|
+
type: 'string',
|
|
145
|
+
description: `Must be "${serverName}"`
|
|
146
|
+
},
|
|
147
|
+
toolName: {
|
|
148
|
+
type: 'string',
|
|
149
|
+
description: `Must be "${tool.name}"`
|
|
150
|
+
},
|
|
151
|
+
toolArgs: tool.inputSchema || { type: 'object', properties: {} },
|
|
152
|
+
requestOptions: {
|
|
153
|
+
type: 'object',
|
|
154
|
+
description: 'Optional: instance selection options',
|
|
155
|
+
properties: {
|
|
156
|
+
sessionId: {
|
|
157
|
+
type: 'string',
|
|
158
|
+
description: 'Session ID for instance stickiness'
|
|
159
|
+
},
|
|
160
|
+
tags: {
|
|
161
|
+
type: 'object',
|
|
162
|
+
description: 'Tags for instance matching (e.g. {"env":"prod"})'
|
|
163
|
+
}
|
|
164
|
+
}
|
|
165
|
+
}
|
|
166
|
+
},
|
|
167
|
+
required: ['serverName', 'toolName', 'toolArgs']
|
|
168
|
+
}
|
|
132
169
|
});
|
|
133
170
|
}
|
|
134
171
|
}
|
|
135
172
|
return gatewayTools;
|
|
136
173
|
}
|
|
174
|
+
// ── Composite: original function (backward compatible) ──────────
|
|
175
|
+
export function generateGatewayToolsList(toolMap) {
|
|
176
|
+
const rawData = gatherRawToolData();
|
|
177
|
+
return resolveToolNames(rawData, toolMap);
|
|
178
|
+
}
|
|
179
|
+
// ── Cache API ────────────────────────────────────────────────────
|
|
180
|
+
/** Lazy get — first-request fallback when cache hasn't been built yet */
|
|
181
|
+
export function getOrBuildGatewayToolMap() {
|
|
182
|
+
if (!cachedToolMap) {
|
|
183
|
+
rebuildFromScratch();
|
|
184
|
+
}
|
|
185
|
+
return cachedToolMap;
|
|
186
|
+
}
|
|
187
|
+
/** Lazy get — first-request fallback when cache hasn't been built yet */
|
|
188
|
+
export function getOrBuildGatewayToolsList() {
|
|
189
|
+
if (!cachedGatewayTools) {
|
|
190
|
+
rebuildFromScratch();
|
|
191
|
+
}
|
|
192
|
+
return cachedGatewayTools;
|
|
193
|
+
}
|
|
194
|
+
/** Lazy get — returns only aggregated external tools (excludes system tools) */
|
|
195
|
+
export function getExternalGatewayTools() {
|
|
196
|
+
return getOrBuildGatewayToolsList().filter((t) => !t.description?.startsWith('[System]'));
|
|
197
|
+
}
|
|
198
|
+
/** Full rebuild from global state — used by TOOLS_UPDATED / SERVER_DISCONNECTED */
|
|
199
|
+
export function rebuildFromScratch() {
|
|
200
|
+
const prevSize = cachedToolMap?.size ?? 0;
|
|
201
|
+
rawToolData = gatherRawToolData();
|
|
202
|
+
cachedToolMap = new Map();
|
|
203
|
+
cachedGatewayTools = resolveToolNames(rawToolData, cachedToolMap);
|
|
204
|
+
logger.info(`Gateway tool cache rebuilt: ${cachedToolMap.size} tools (was ${prevSize})`, LOG_MODULES.TOOL_LIST_GENERATOR);
|
|
205
|
+
}
|
|
206
|
+
/** Incremental add — update raw data layer, re-resolve names only */
|
|
207
|
+
export function addToCache(serverName, toolNames) {
|
|
208
|
+
if (!rawToolData) {
|
|
209
|
+
logger.debug(`Gateway tool cache: raw data not initialized, falling back to full rebuild`, LOG_MODULES.TOOL_LIST_GENERATOR);
|
|
210
|
+
rebuildFromScratch();
|
|
211
|
+
return;
|
|
212
|
+
}
|
|
213
|
+
// Find tool definitions with instance info from tool cache entries
|
|
214
|
+
const toolDefs = new Map();
|
|
215
|
+
const toolIndexes = new Map();
|
|
216
|
+
for (const [compositeKey, tools] of mcpConnectionManager.getToolCacheEntries()) {
|
|
217
|
+
const parsed = parseCompositeKey(compositeKey);
|
|
218
|
+
if (!parsed || parsed.serverName !== serverName)
|
|
219
|
+
continue;
|
|
220
|
+
for (const t of tools) {
|
|
221
|
+
if (toolNames.includes(t.name)) {
|
|
222
|
+
toolDefs.set(t.name, t);
|
|
223
|
+
toolIndexes.set(t.name, parsed.serverIndex);
|
|
224
|
+
}
|
|
225
|
+
}
|
|
226
|
+
}
|
|
227
|
+
const newTools = Array.from(toolDefs.values());
|
|
228
|
+
const existing = rawToolData.get(serverName);
|
|
229
|
+
if (existing) {
|
|
230
|
+
const mergedTools = new Map();
|
|
231
|
+
for (const t of existing.tools)
|
|
232
|
+
mergedTools.set(t.name, t);
|
|
233
|
+
for (const t of newTools)
|
|
234
|
+
mergedTools.set(t.name, t);
|
|
235
|
+
const mergedIndexes = new Map(existing.toolIndexes);
|
|
236
|
+
for (const [name, idx] of toolIndexes) {
|
|
237
|
+
mergedIndexes.set(name, idx);
|
|
238
|
+
}
|
|
239
|
+
existing.tools = Array.from(mergedTools.values());
|
|
240
|
+
existing.aggregated = [...new Set([...existing.aggregated, ...toolNames])];
|
|
241
|
+
existing.toolIndexes = mergedIndexes;
|
|
242
|
+
}
|
|
243
|
+
else {
|
|
244
|
+
const serverConfig = hubManager.getServerByName(serverName);
|
|
245
|
+
rawToolData.set(serverName, {
|
|
246
|
+
tools: newTools,
|
|
247
|
+
aggregated: serverConfig?.template?.aggregatedTools ?? toolNames,
|
|
248
|
+
toolIndexes
|
|
249
|
+
});
|
|
250
|
+
}
|
|
251
|
+
cachedToolMap = new Map();
|
|
252
|
+
cachedGatewayTools = resolveToolNames(rawToolData, cachedToolMap);
|
|
253
|
+
logger.info(`Gateway tool cache: added ${toolNames.length} tool(s) from [${serverName}], total=${cachedToolMap.size}`, LOG_MODULES.TOOL_LIST_GENERATOR);
|
|
254
|
+
}
|
|
255
|
+
/** Incremental remove — update raw data layer, re-resolve names only */
|
|
256
|
+
export function removeFromCache(serverName, toolNames) {
|
|
257
|
+
if (!rawToolData) {
|
|
258
|
+
logger.debug(`Gateway tool cache: raw data not initialized, falling back to full rebuild`, LOG_MODULES.TOOL_LIST_GENERATOR);
|
|
259
|
+
rebuildFromScratch();
|
|
260
|
+
return;
|
|
261
|
+
}
|
|
262
|
+
const existing = rawToolData.get(serverName);
|
|
263
|
+
if (!existing) {
|
|
264
|
+
logger.debug(`Gateway tool cache: no cached data for [${serverName}], skip remove`, LOG_MODULES.TOOL_LIST_GENERATOR);
|
|
265
|
+
return;
|
|
266
|
+
}
|
|
267
|
+
const removeSet = new Set(toolNames);
|
|
268
|
+
existing.tools = existing.tools.filter((t) => !removeSet.has(t.name));
|
|
269
|
+
existing.aggregated = existing.aggregated.filter((t) => !removeSet.has(t));
|
|
270
|
+
if (existing.aggregated.length === 0) {
|
|
271
|
+
rawToolData.delete(serverName);
|
|
272
|
+
logger.info(`Gateway tool cache: all aggregated tools removed from [${serverName}], server removed from cache`, LOG_MODULES.TOOL_LIST_GENERATOR);
|
|
273
|
+
}
|
|
274
|
+
cachedToolMap = new Map();
|
|
275
|
+
cachedGatewayTools = resolveToolNames(rawToolData, cachedToolMap);
|
|
276
|
+
logger.info(`Gateway tool cache: removed ${toolNames.length} tool(s) from [${serverName}], total=${cachedToolMap.size}`, LOG_MODULES.TOOL_LIST_GENERATOR);
|
|
277
|
+
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"hub-manager.service.d.ts","sourceRoot":"","sources":["../../../../src/services/hub-manager.service.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,aAAa,EAEb,YAAY,EACZ,cAAc,EACd,cAAc,EACf,MAAM,2BAA2B,CAAC;AAEnC,OAAO,KAAK,EACV,mBAAmB,EACnB,yBAAyB,EAC1B,MAAM,gCAAgC,CAAC;AAKxC;;;;;;GAMG;AACH,qBAAa,iBAAiB;IAC5B,OAAO,CAAC,aAAa,CAAgB;gBAEzB,OAAO,GAAE,aAA6B;IAIlD;;OAEG;IACG,0BAA0B,CAC9B,OAAO,EAAE,KAAK,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,OAAO,CAAC,cAAc,CAAC,CAAA;KAAE,CAAC,GAChE,OAAO,CAAC,IAAI,CAAC;IAUhB;;OAEG;IACG,gCAAgC,CAAC,WAAW,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;IAM5E;;OAEG;IACG,sBAAsB,CAAC,WAAW,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;IA8BlE;;OAEG;IACH,aAAa,IAAI,KAAK,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,YAAY,CAAA;KAAE,CAAC;IAI9D;;OAEG;IACH,aAAa,CACX,EAAE,EAAE,MAAM,GACT;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,YAAY,CAAC;QAAC,QAAQ,EAAE,cAAc,CAAA;KAAE,GAAG,SAAS;IAe/E;;OAEG;IACH,eAAe,CAAC,IAAI,EAAE,MAAM,GAAG,YAAY,GAAG,SAAS;IAIvD;;OAEG;IACH,wBAAwB,CAAC,IAAI,EAAE,MAAM,GAAG,cAAc,EAAE;IAIxD;;OAEG;IACH,kBAAkB,IAAI,MAAM,CAAC,MAAM,EAAE,cAAc,EAAE,CAAC;IAStD;;OAEG;IACH,uBAAuB,CAAC,IAAI,EAAE,MAAM,EAAE,UAAU,CAAC,EAAE,MAAM,GAAG,mBAAmB,GAAG,IAAI;IAQtF;;OAEG;IACG,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,EAAE,OAAO,CAAC,cAAc,CAAC,GAAG,OAAO,CAAC,YAAY,CAAC;IAOrF;;;;OAIG;IACG,iBAAiB,CACrB,IAAI,EAAE,MAAM,EACZ,QAAQ,EAAE,OAAO,CAAC,cAAc,CAAC,GAChC,OAAO,CAAC,cAAc,CAAC;IAQ1B;;OAEG;IACG,YAAY,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,CAAC,cAAc,CAAC,GAAG,OAAO,CAAC,YAAY,GAAG,IAAI,CAAC;
|
|
1
|
+
{"version":3,"file":"hub-manager.service.d.ts","sourceRoot":"","sources":["../../../../src/services/hub-manager.service.ts"],"names":[],"mappings":"AAAA,OAAO,EACL,aAAa,EAEb,YAAY,EACZ,cAAc,EACd,cAAc,EACf,MAAM,2BAA2B,CAAC;AAEnC,OAAO,KAAK,EACV,mBAAmB,EACnB,yBAAyB,EAC1B,MAAM,gCAAgC,CAAC;AAKxC;;;;;;GAMG;AACH,qBAAa,iBAAiB;IAC5B,OAAO,CAAC,aAAa,CAAgB;gBAEzB,OAAO,GAAE,aAA6B;IAIlD;;OAEG;IACG,0BAA0B,CAC9B,OAAO,EAAE,KAAK,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,OAAO,CAAC,cAAc,CAAC,CAAA;KAAE,CAAC,GAChE,OAAO,CAAC,IAAI,CAAC;IAUhB;;OAEG;IACG,gCAAgC,CAAC,WAAW,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;IAM5E;;OAEG;IACG,sBAAsB,CAAC,WAAW,EAAE,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC;IA8BlE;;OAEG;IACH,aAAa,IAAI,KAAK,CAAC;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,YAAY,CAAA;KAAE,CAAC;IAI9D;;OAEG;IACH,aAAa,CACX,EAAE,EAAE,MAAM,GACT;QAAE,IAAI,EAAE,MAAM,CAAC;QAAC,MAAM,EAAE,YAAY,CAAC;QAAC,QAAQ,EAAE,cAAc,CAAA;KAAE,GAAG,SAAS;IAe/E;;OAEG;IACH,eAAe,CAAC,IAAI,EAAE,MAAM,GAAG,YAAY,GAAG,SAAS;IAIvD;;OAEG;IACH,wBAAwB,CAAC,IAAI,EAAE,MAAM,GAAG,cAAc,EAAE;IAIxD;;OAEG;IACH,kBAAkB,IAAI,MAAM,CAAC,MAAM,EAAE,cAAc,EAAE,CAAC;IAStD;;OAEG;IACH,uBAAuB,CAAC,IAAI,EAAE,MAAM,EAAE,UAAU,CAAC,EAAE,MAAM,GAAG,mBAAmB,GAAG,IAAI;IAQtF;;OAEG;IACG,SAAS,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,EAAE,OAAO,CAAC,cAAc,CAAC,GAAG,OAAO,CAAC,YAAY,CAAC;IAOrF;;;;OAIG;IACG,iBAAiB,CACrB,IAAI,EAAE,MAAM,EACZ,QAAQ,EAAE,OAAO,CAAC,cAAc,CAAC,GAChC,OAAO,CAAC,cAAc,CAAC;IAQ1B;;OAEG;IACG,YAAY,CAAC,IAAI,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,CAAC,cAAc,CAAC,GAAG,OAAO,CAAC,YAAY,GAAG,IAAI,CAAC;IAkChG;;OAEG;IACG,oBAAoB,CACxB,IAAI,EAAE,MAAM,EACZ,KAAK,EAAE,MAAM,EACb,OAAO,EAAE,OAAO,CAAC,cAAc,CAAC,GAC/B,OAAO,CAAC,IAAI,CAAC;IAShB;;OAEG;IACG,YAAY,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAqBlD;;OAEG;IACG,oBAAoB,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAmBtE;;OAEG;IACG,uBAAuB,CAAC,IAAI,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAI7D;;OAEG;IACG,qCAAqC,CACzC,IAAI,EAAE,MAAM,EACZ,QAAQ,EAAE,yBAAyB,GAClC,OAAO,CAAC,YAAY,GAAG,IAAI,CAAC;CAwBhC;AAED,eAAO,MAAM,UAAU,mBAA0B,CAAC"}
|
|
@@ -1,7 +1,7 @@
|
|
|
1
1
|
import { configManager } from '../config/config-manager.js';
|
|
2
2
|
import { resolveInstanceConfig } from '../config/config-migrator.js';
|
|
3
|
-
import { logger, LOG_MODULES } from '../utils/logger.js';
|
|
4
|
-
import { mcpConnectionManager } from './
|
|
3
|
+
import { logger, LOG_MODULES } from '../utils/logger/index.js';
|
|
4
|
+
import { mcpConnectionManager } from './connection/index.js';
|
|
5
5
|
import { eventBus, EventTypes } from './event-bus.service.js';
|
|
6
6
|
/**
|
|
7
7
|
* Manages MCP server configurations and lifecycle operations through the configuration manager.
|
|
@@ -148,11 +148,24 @@ export class HubManagerService {
|
|
|
148
148
|
logger.warn(`Attempted to update non-existent server: ${name}`, LOG_MODULES.HUB_MANAGER);
|
|
149
149
|
return null;
|
|
150
150
|
}
|
|
151
|
+
const oldAggregated = existing.template?.aggregatedTools ?? [];
|
|
151
152
|
await this.configManager.updateServer(name, updates);
|
|
152
153
|
logger.info(`Server updated: ${name}`, LOG_MODULES.HUB_MANAGER);
|
|
153
154
|
const updatedServer = this.getServerByName(name) || null;
|
|
154
155
|
if (updatedServer) {
|
|
155
156
|
eventBus.publish(EventTypes.SERVER_UPDATED, { name, config: updatedServer });
|
|
157
|
+
if ('aggregatedTools' in updates) {
|
|
158
|
+
const newAggregated = (updates.aggregatedTools ?? []);
|
|
159
|
+
const oldSet = new Set(oldAggregated);
|
|
160
|
+
const newSet = new Set(newAggregated);
|
|
161
|
+
const added = newAggregated.filter((t) => !oldSet.has(t));
|
|
162
|
+
const removed = oldAggregated.filter((t) => !newSet.has(t));
|
|
163
|
+
eventBus.publish(EventTypes.AGGREGATED_TOOLS_CHANGED, {
|
|
164
|
+
name,
|
|
165
|
+
added,
|
|
166
|
+
removed
|
|
167
|
+
});
|
|
168
|
+
}
|
|
156
169
|
}
|
|
157
170
|
return updatedServer;
|
|
158
171
|
}
|
|
@@ -1,5 +1,5 @@
|
|
|
1
1
|
import { InstanceSelectionStrategy } from '../../models/server.model.js';
|
|
2
|
-
import { mcpConnectionManager } from '../
|
|
2
|
+
import { mcpConnectionManager } from '../connection/index.js';
|
|
3
3
|
/**
|
|
4
4
|
* Error thrown when tag-match-unique instance selection fails.
|
|
5
5
|
* Passes raw data so the error class itself can format the message.
|
|
@@ -1,26 +1,5 @@
|
|
|
1
1
|
import type { Resource } from '../../../shared/models/resource.model.js';
|
|
2
2
|
import type { ServerStatus } from '../../../shared/types/common.types.js';
|
|
3
|
-
/**
|
|
4
|
-
* Clears the Hub to MCP URI mapping.
|
|
5
|
-
* Should be called before regenerating resources.
|
|
6
|
-
*/
|
|
7
|
-
export declare function clearHubToMcpUriMap(): void;
|
|
8
|
-
/**
|
|
9
|
-
* URI for the use guide resource.
|
|
10
|
-
*/
|
|
11
|
-
export declare const USE_GUIDE_URI = "hub://use-guide";
|
|
12
|
-
/**
|
|
13
|
-
* Name of the use guide resource.
|
|
14
|
-
*/
|
|
15
|
-
export declare const USE_GUIDE_NAME = "MCP Hub Lite Use Guide";
|
|
16
|
-
/**
|
|
17
|
-
* Description of the use guide resource.
|
|
18
|
-
*/
|
|
19
|
-
export declare const USE_GUIDE_DESCRIPTION = "Comprehensive guide to using MCP Hub Lite gateway and its features";
|
|
20
|
-
/**
|
|
21
|
-
* MIME type for the use guide resource.
|
|
22
|
-
*/
|
|
23
|
-
export declare const USE_GUIDE_MIME_TYPE = "text/markdown";
|
|
24
3
|
/**
|
|
25
4
|
* Server metadata resource content.
|
|
26
5
|
*/
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"resource-generator.d.ts","sourceRoot":"","sources":["../../../../../src/services/hub-tools/resource-generator.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,kCAAkC,CAAC;AACjE,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,+BAA+B,CAAC;
|
|
1
|
+
{"version":3,"file":"resource-generator.d.ts","sourceRoot":"","sources":["../../../../../src/services/hub-tools/resource-generator.ts"],"names":[],"mappings":"AAKA,OAAO,KAAK,EAAE,QAAQ,EAAE,MAAM,kCAAkC,CAAC;AACjE,OAAO,KAAK,EAAE,YAAY,EAAE,MAAM,+BAA+B,CAAC;AAiLlE;;GAEG;AACH,MAAM,WAAW,cAAc;IAC7B,IAAI,EAAE,MAAM,CAAC;IACb,MAAM,EAAE,YAAY,CAAC;IACrB,UAAU,EAAE,MAAM,CAAC;IACnB,KAAK,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IAC9B,cAAc,EAAE,MAAM,CAAC;IACvB,IAAI,EAAE,MAAM,CAAC,MAAM,EAAE,MAAM,CAAC,CAAC;IAC7B,aAAa,EAAE,MAAM,CAAC;IACtB,MAAM,EAAE,MAAM,CAAC;IACf,WAAW,EAAE,MAAM,CAAC;CACrB;AAED;;;;;;;;;;;;;;;;GAgBG;AACH,wBAAgB,wBAAwB,IAAI,QAAQ,EAAE,CAqErD;AAED;;;;;;;;;;;;;;;;;;;;;;;;;GAyBG;AACH,wBAAsB,YAAY,CAChC,GAAG,EAAE,MAAM,GACV,OAAO,CAAC,cAAc,GAAG,QAAQ,EAAE,GAAG,MAAM,GAAG,OAAO,CAAC,CAyGzD"}
|