@leonailtd/priority-mcp-client 0.1.0 → 0.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/dist/bin.js +191 -20
- package/dist/bin.js.map +1 -1
- package/package.json +1 -1
package/dist/bin.js
CHANGED
|
@@ -14,23 +14,160 @@ import {
|
|
|
14
14
|
ListPromptsRequestSchema,
|
|
15
15
|
GetPromptRequestSchema
|
|
16
16
|
} from "@modelcontextprotocol/sdk/types.js";
|
|
17
|
-
|
|
17
|
+
var sleep = (ms) => new Promise((resolve) => setTimeout(resolve, ms));
|
|
18
|
+
function isRetryableError(error) {
|
|
19
|
+
if (!error) return false;
|
|
20
|
+
const errorStr = String(error).toLowerCase();
|
|
21
|
+
const retryablePatterns = [
|
|
22
|
+
"econnreset",
|
|
23
|
+
"econnrefused",
|
|
24
|
+
"etimedout",
|
|
25
|
+
"timeout",
|
|
26
|
+
"network",
|
|
27
|
+
"socket hang up",
|
|
28
|
+
"502",
|
|
29
|
+
"503",
|
|
30
|
+
"504"
|
|
31
|
+
];
|
|
32
|
+
return retryablePatterns.some((pattern) => errorStr.includes(pattern));
|
|
33
|
+
}
|
|
34
|
+
async function retryWithBackoff(fn, maxRetries = 3, baseDelay = 1e3, debug = false) {
|
|
35
|
+
let lastError;
|
|
36
|
+
for (let attempt = 0; attempt <= maxRetries; attempt++) {
|
|
37
|
+
try {
|
|
38
|
+
return await fn();
|
|
39
|
+
} catch (error) {
|
|
40
|
+
lastError = error;
|
|
41
|
+
if (attempt >= maxRetries) {
|
|
42
|
+
throw error;
|
|
43
|
+
}
|
|
44
|
+
if (!isRetryableError(error)) {
|
|
45
|
+
throw error;
|
|
46
|
+
}
|
|
47
|
+
const delay = Math.min(baseDelay * Math.pow(2, attempt), 3e4);
|
|
48
|
+
const jitter = Math.random() * 0.25 * delay;
|
|
49
|
+
const totalDelay = Math.floor(delay + jitter);
|
|
50
|
+
if (debug) {
|
|
51
|
+
console.error(`[DEBUG] Retry attempt ${attempt + 1}/${maxRetries} after ${totalDelay}ms`);
|
|
52
|
+
}
|
|
53
|
+
await sleep(totalDelay);
|
|
54
|
+
}
|
|
55
|
+
}
|
|
56
|
+
throw lastError;
|
|
57
|
+
}
|
|
58
|
+
var ConnectionMonitor = class {
|
|
59
|
+
lastSuccessfulCall = Date.now();
|
|
60
|
+
failureCount = 0;
|
|
61
|
+
totalCalls = 0;
|
|
62
|
+
debug;
|
|
63
|
+
constructor(debug = false) {
|
|
64
|
+
this.debug = debug;
|
|
65
|
+
}
|
|
66
|
+
async monitorCall(operation, fn) {
|
|
67
|
+
this.totalCalls++;
|
|
68
|
+
const startTime = Date.now();
|
|
69
|
+
try {
|
|
70
|
+
const result = await fn();
|
|
71
|
+
this.lastSuccessfulCall = Date.now();
|
|
72
|
+
this.failureCount = 0;
|
|
73
|
+
const duration = Date.now() - startTime;
|
|
74
|
+
if (this.debug && duration > 5e3) {
|
|
75
|
+
console.error(`[WARNING] Slow operation: ${operation} took ${duration}ms`);
|
|
76
|
+
}
|
|
77
|
+
return result;
|
|
78
|
+
} catch (error) {
|
|
79
|
+
this.failureCount++;
|
|
80
|
+
const timeSinceSuccess = Date.now() - this.lastSuccessfulCall;
|
|
81
|
+
if (timeSinceSuccess > 6e4 && this.failureCount >= 3) {
|
|
82
|
+
console.error("[WARNING] Connection appears degraded");
|
|
83
|
+
console.error(`[WARNING] ${this.failureCount} failures in the last ${Math.floor(timeSinceSuccess / 1e3)}s`);
|
|
84
|
+
}
|
|
85
|
+
throw error;
|
|
86
|
+
}
|
|
87
|
+
}
|
|
88
|
+
};
|
|
89
|
+
function getErrorHint(error, context) {
|
|
90
|
+
const errorStr = String(error).toLowerCase();
|
|
91
|
+
if (errorStr.includes("401") || errorStr.includes("unauthorized")) {
|
|
92
|
+
return "Check your API token (--token sk-cust-...)";
|
|
93
|
+
}
|
|
94
|
+
if (errorStr.includes("403") || errorStr.includes("forbidden")) {
|
|
95
|
+
return "Token is valid but lacks permission. Contact your administrator.";
|
|
96
|
+
}
|
|
97
|
+
if (errorStr.includes("404")) {
|
|
98
|
+
return "Server endpoint not found. Verify --url is correct.";
|
|
99
|
+
}
|
|
100
|
+
if (errorStr.includes("timeout") || errorStr.includes("etimedout")) {
|
|
101
|
+
return "Request timed out. Server may be slow or overloaded.";
|
|
102
|
+
}
|
|
103
|
+
if (errorStr.includes("econnrefused")) {
|
|
104
|
+
return "Connection refused. Verify server is running and URL is correct.";
|
|
105
|
+
}
|
|
106
|
+
if (errorStr.includes("econnreset") || errorStr.includes("socket hang up")) {
|
|
107
|
+
return "Connection lost. Check network connectivity.";
|
|
108
|
+
}
|
|
109
|
+
if (errorStr.includes("dns") || errorStr.includes("getaddrinfo")) {
|
|
110
|
+
return "DNS lookup failed. Check server URL spelling.";
|
|
111
|
+
}
|
|
112
|
+
return "See error details above";
|
|
113
|
+
}
|
|
114
|
+
program.name("priority-mcp-client").description("MCP client that proxies Claude Desktop to Priority MCP server").requiredOption("--url <url>", "Priority MCP server URL (e.g., https://priority-mcp.leonai.io)").requiredOption("--token <token>", "Authentication token (sk-cust-...)").requiredOption("--priority-url <url>", "Priority ERP base URL").requiredOption("--priority-company <company>", "Priority company name").requiredOption("--priority-username <username>", "Priority API username").requiredOption("--priority-password <password>", "Priority API password").option("--debug", "Enable debug logging").option("--verbose", "Enable verbose logging (includes request/response details)").parse();
|
|
18
115
|
var options = program.opts();
|
|
19
116
|
async function main() {
|
|
20
117
|
try {
|
|
21
|
-
const debug = options.debug || false;
|
|
118
|
+
const debug = options.debug || options.verbose || false;
|
|
119
|
+
const verbose = options.verbose || false;
|
|
120
|
+
if (debug) {
|
|
121
|
+
console.error("[DEBUG] Starting Priority MCP Client v1.0.0");
|
|
122
|
+
console.error(`[DEBUG] Server URL: ${options.url}`);
|
|
123
|
+
}
|
|
22
124
|
if (debug) {
|
|
23
|
-
console.error("[DEBUG]
|
|
24
|
-
console.error(`[DEBUG] Connecting to: ${options.url}`);
|
|
125
|
+
console.error("[DEBUG] Performing health check...");
|
|
25
126
|
}
|
|
127
|
+
try {
|
|
128
|
+
const healthUrl = `${options.url}/health`;
|
|
129
|
+
const healthResponse = await retryWithBackoff(
|
|
130
|
+
async () => {
|
|
131
|
+
const response = await fetch(healthUrl);
|
|
132
|
+
if (!response.ok) {
|
|
133
|
+
throw new Error(`Health check failed: ${response.status} ${response.statusText}`);
|
|
134
|
+
}
|
|
135
|
+
return response.json();
|
|
136
|
+
},
|
|
137
|
+
2,
|
|
138
|
+
// 2 retries for health check
|
|
139
|
+
500,
|
|
140
|
+
// 500ms base delay
|
|
141
|
+
debug
|
|
142
|
+
);
|
|
143
|
+
if (debug) {
|
|
144
|
+
console.error("[DEBUG] Health check passed:", JSON.stringify(healthResponse));
|
|
145
|
+
}
|
|
146
|
+
if (healthResponse.status !== "healthy") {
|
|
147
|
+
console.error("[WARNING] Server health check returned non-healthy status");
|
|
148
|
+
console.error("[WARNING] Service may be degraded");
|
|
149
|
+
}
|
|
150
|
+
} catch (error) {
|
|
151
|
+
console.error("[ERROR] Server health check failed");
|
|
152
|
+
console.error("[ERROR]", error);
|
|
153
|
+
console.error("[HINT]", getErrorHint(error, "health check"));
|
|
154
|
+
console.error("[HINT] Verify server is running: curl", `${options.url}/health`);
|
|
155
|
+
process.exit(1);
|
|
156
|
+
}
|
|
157
|
+
const connectionMonitor = new ConnectionMonitor(debug);
|
|
26
158
|
const client = new Client({
|
|
27
159
|
name: "priority-mcp-client",
|
|
28
160
|
version: "1.0.0"
|
|
29
161
|
}, {
|
|
30
162
|
capabilities: {}
|
|
31
163
|
});
|
|
164
|
+
const crypto = await import("crypto");
|
|
165
|
+
const tenantIdSource = `${options.priorityCompany.toLowerCase()}@${options.priorityUrl.toLowerCase()}`;
|
|
166
|
+
const tenantId = crypto.createHash("sha256").update(tenantIdSource).digest("hex").substring(0, 16);
|
|
32
167
|
const headers = {
|
|
33
168
|
"Authorization": `Bearer ${options.token}`,
|
|
169
|
+
"X-Tenant-ID": tenantId,
|
|
170
|
+
// Unique identifier for rate limiting/tracking
|
|
34
171
|
"X-Priority-URL": options.priorityUrl,
|
|
35
172
|
"X-Priority-Company": options.priorityCompany,
|
|
36
173
|
"X-Priority-Username": options.priorityUsername,
|
|
@@ -40,14 +177,33 @@ async function main() {
|
|
|
40
177
|
console.error("[DEBUG] Headers prepared (credentials hidden)");
|
|
41
178
|
}
|
|
42
179
|
const mcpUrl = new URL("/mcp", options.url);
|
|
180
|
+
const controller = new AbortController();
|
|
181
|
+
const timeoutId = setTimeout(() => controller.abort(), 3e4);
|
|
43
182
|
const transport = new StreamableHTTPClientTransport(mcpUrl, {
|
|
44
183
|
requestInit: {
|
|
45
|
-
headers
|
|
184
|
+
headers,
|
|
185
|
+
signal: controller.signal
|
|
46
186
|
}
|
|
47
187
|
});
|
|
48
|
-
|
|
49
|
-
|
|
50
|
-
|
|
188
|
+
try {
|
|
189
|
+
await retryWithBackoff(
|
|
190
|
+
() => client.connect(transport),
|
|
191
|
+
3,
|
|
192
|
+
// 3 retries for connection
|
|
193
|
+
1e3,
|
|
194
|
+
// 1s base delay
|
|
195
|
+
debug
|
|
196
|
+
);
|
|
197
|
+
clearTimeout(timeoutId);
|
|
198
|
+
if (debug) {
|
|
199
|
+
console.error("[DEBUG] Connected to remote Priority MCP server");
|
|
200
|
+
}
|
|
201
|
+
} catch (error) {
|
|
202
|
+
clearTimeout(timeoutId);
|
|
203
|
+
console.error("[ERROR] Failed to connect to Priority MCP server");
|
|
204
|
+
console.error("[ERROR]", error);
|
|
205
|
+
console.error("[HINT]", getErrorHint(error, "connection"));
|
|
206
|
+
throw error;
|
|
51
207
|
}
|
|
52
208
|
const server = new Server({
|
|
53
209
|
name: "priority-mcp-proxy",
|
|
@@ -61,33 +217,48 @@ async function main() {
|
|
|
61
217
|
});
|
|
62
218
|
server.setRequestHandler(ListToolsRequestSchema, async () => {
|
|
63
219
|
if (debug) console.error("[DEBUG] Forwarding tools/list");
|
|
64
|
-
|
|
65
|
-
|
|
220
|
+
return await connectionMonitor.monitorCall("tools/list", async () => {
|
|
221
|
+
return await retryWithBackoff(() => client.listTools(), 2, 500, debug);
|
|
222
|
+
});
|
|
66
223
|
});
|
|
67
224
|
server.setRequestHandler(CallToolRequestSchema, async (request) => {
|
|
68
225
|
if (debug) console.error(`[DEBUG] Forwarding tools/call: ${request.params.name}`);
|
|
69
|
-
|
|
70
|
-
|
|
226
|
+
if (verbose) console.error(`[VERBOSE] Tool params:`, JSON.stringify(request.params));
|
|
227
|
+
try {
|
|
228
|
+
const result = await connectionMonitor.monitorCall(`tools/call/${request.params.name}`, async () => {
|
|
229
|
+
return await retryWithBackoff(() => client.callTool(request.params), 3, 1e3, debug);
|
|
230
|
+
});
|
|
231
|
+
if (verbose) console.error(`[VERBOSE] Tool result:`, JSON.stringify(result).slice(0, 500));
|
|
232
|
+
return result;
|
|
233
|
+
} catch (error) {
|
|
234
|
+
console.error(`[ERROR] Tool call failed: ${request.params.name}`);
|
|
235
|
+
console.error(`[HINT]`, getErrorHint(error, "tool call"));
|
|
236
|
+
throw error;
|
|
237
|
+
}
|
|
71
238
|
});
|
|
72
239
|
server.setRequestHandler(ListResourcesRequestSchema, async () => {
|
|
73
240
|
if (debug) console.error("[DEBUG] Forwarding resources/list");
|
|
74
|
-
|
|
75
|
-
|
|
241
|
+
return await connectionMonitor.monitorCall("resources/list", async () => {
|
|
242
|
+
return await retryWithBackoff(() => client.listResources(), 2, 500, debug);
|
|
243
|
+
});
|
|
76
244
|
});
|
|
77
245
|
server.setRequestHandler(ReadResourceRequestSchema, async (request) => {
|
|
78
246
|
if (debug) console.error(`[DEBUG] Forwarding resources/read: ${request.params.uri}`);
|
|
79
|
-
|
|
80
|
-
|
|
247
|
+
return await connectionMonitor.monitorCall(`resources/read/${request.params.uri}`, async () => {
|
|
248
|
+
return await retryWithBackoff(() => client.readResource(request.params), 3, 1e3, debug);
|
|
249
|
+
});
|
|
81
250
|
});
|
|
82
251
|
server.setRequestHandler(ListPromptsRequestSchema, async () => {
|
|
83
252
|
if (debug) console.error("[DEBUG] Forwarding prompts/list");
|
|
84
|
-
|
|
85
|
-
|
|
253
|
+
return await connectionMonitor.monitorCall("prompts/list", async () => {
|
|
254
|
+
return await retryWithBackoff(() => client.listPrompts(), 2, 500, debug);
|
|
255
|
+
});
|
|
86
256
|
});
|
|
87
257
|
server.setRequestHandler(GetPromptRequestSchema, async (request) => {
|
|
88
258
|
if (debug) console.error(`[DEBUG] Forwarding prompts/get: ${request.params.name}`);
|
|
89
|
-
|
|
90
|
-
|
|
259
|
+
return await connectionMonitor.monitorCall(`prompts/get/${request.params.name}`, async () => {
|
|
260
|
+
return await retryWithBackoff(() => client.getPrompt(request.params), 2, 500, debug);
|
|
261
|
+
});
|
|
91
262
|
});
|
|
92
263
|
const stdioTransport = new StdioServerTransport();
|
|
93
264
|
await server.connect(stdioTransport);
|
package/dist/bin.js.map
CHANGED
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"sources":["../src/bin.ts"],"sourcesContent":["#!/usr/bin/env node\r\n\r\nimport { program } from 'commander';\r\nimport { Client } from '@modelcontextprotocol/sdk/client/index.js';\r\nimport { Server } from '@modelcontextprotocol/sdk/server/index.js';\r\nimport { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';\r\nimport { StreamableHTTPClientTransport } from '@modelcontextprotocol/sdk/client/streamableHttp.js';\r\nimport {\r\n ListToolsRequestSchema,\r\n CallToolRequestSchema,\r\n ListResourcesRequestSchema,\r\n ReadResourceRequestSchema,\r\n ListPromptsRequestSchema,\r\n GetPromptRequestSchema\r\n} from '@modelcontextprotocol/sdk/types.js';\r\n\r\n// Parse command-line arguments\r\nprogram\r\n .name('priority-mcp-client')\r\n .description('MCP client that proxies Claude Desktop to Priority MCP server')\r\n .requiredOption('--url <url>', 'Priority MCP server URL (e.g., https://priority-mcp.leonai.io)')\r\n .requiredOption('--token <token>', 'Authentication token (sk-cust-...)')\r\n .requiredOption('--priority-url <url>', 'Priority ERP base URL')\r\n .requiredOption('--priority-company <company>', 'Priority company name')\r\n .requiredOption('--priority-username <username>', 'Priority API username')\r\n .requiredOption('--priority-password <password>', 'Priority API password')\r\n .option('--debug', 'Enable debug logging')\r\n .parse();\r\n\r\nconst options = program.opts();\r\n\r\nasync function main() {\r\n try {\r\n const debug = options.debug || false;\r\n\r\n if (debug) {\r\n console.error('[DEBUG] Starting Priority MCP Client');\r\n console.error(`[DEBUG] Connecting to: ${options.url}`);\r\n }\r\n\r\n // Create MCP client that will connect to remote Priority server\r\n const client = new Client({\r\n name: 'priority-mcp-client',\r\n version: '1.0.0'\r\n }, {\r\n capabilities: {}\r\n });\r\n\r\n // Prepare headers for authentication\r\n const headers = {\r\n 'Authorization': `Bearer ${options.token}`,\r\n 'X-Priority-URL': options.priorityUrl,\r\n 'X-Priority-Company': options.priorityCompany,\r\n 'X-Priority-Username': options.priorityUsername,\r\n 'X-Priority-Password': options.priorityPassword\r\n };\r\n\r\n if (debug) {\r\n console.error('[DEBUG] Headers prepared (credentials hidden)');\r\n }\r\n\r\n // Connect to remote Priority MCP server via Streamable HTTP\r\n const mcpUrl = new URL('/mcp', options.url);\r\n const transport = new StreamableHTTPClientTransport(mcpUrl, {\r\n requestInit: {\r\n headers: headers\r\n }\r\n });\r\n\r\n await client.connect(transport);\r\n\r\n if (debug) {\r\n console.error('[DEBUG] Connected to remote Priority MCP server');\r\n }\r\n\r\n // Create stdio server for Claude Desktop\r\n const server = new Server({\r\n name: 'priority-mcp-proxy',\r\n version: '1.0.0'\r\n }, {\r\n capabilities: {\r\n tools: {},\r\n resources: {},\r\n prompts: {}\r\n }\r\n });\r\n\r\n // Set up request handlers that forward everything to the remote client\r\n\r\n // List tools\r\n server.setRequestHandler(ListToolsRequestSchema, async () => {\r\n if (debug) console.error('[DEBUG] Forwarding tools/list');\r\n const result = await client.listTools();\r\n return result;\r\n });\r\n\r\n // Call tool\r\n server.setRequestHandler(CallToolRequestSchema, async (request) => {\r\n if (debug) console.error(`[DEBUG] Forwarding tools/call: ${request.params.name}`);\r\n const result = await client.callTool(request.params);\r\n return result;\r\n });\r\n\r\n // List resources\r\n server.setRequestHandler(ListResourcesRequestSchema, async () => {\r\n if (debug) console.error('[DEBUG] Forwarding resources/list');\r\n const result = await client.listResources();\r\n return result;\r\n });\r\n\r\n // Read resource\r\n server.setRequestHandler(ReadResourceRequestSchema, async (request) => {\r\n if (debug) console.error(`[DEBUG] Forwarding resources/read: ${request.params.uri}`);\r\n const result = await client.readResource(request.params);\r\n return result;\r\n });\r\n\r\n // List prompts\r\n server.setRequestHandler(ListPromptsRequestSchema, async () => {\r\n if (debug) console.error('[DEBUG] Forwarding prompts/list');\r\n const result = await client.listPrompts();\r\n return result;\r\n });\r\n\r\n // Get prompt\r\n server.setRequestHandler(GetPromptRequestSchema, async (request) => {\r\n if (debug) console.error(`[DEBUG] Forwarding prompts/get: ${request.params.name}`);\r\n const result = await client.getPrompt(request.params);\r\n return result;\r\n });\r\n\r\n // Connect server to stdio for Claude Desktop\r\n const stdioTransport = new StdioServerTransport();\r\n await server.connect(stdioTransport);\r\n\r\n if (debug) {\r\n console.error('[DEBUG] Stdio server ready, proxy active');\r\n }\r\n\r\n // Keep process running\r\n process.on('SIGINT', async () => {\r\n if (debug) console.error('[DEBUG] Shutting down...');\r\n await server.close();\r\n await client.close();\r\n process.exit(0);\r\n });\r\n\r\n } catch (error) {\r\n console.error('Failed to start Priority MCP client:', error);\r\n process.exit(1);\r\n }\r\n}\r\n\r\nmain().catch((error) => {\r\n console.error('Fatal error:', error);\r\n process.exit(1);\r\n});\r\n"],"mappings":"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","names":[]}
|
|
1
|
+
{"version":3,"sources":["../src/bin.ts"],"sourcesContent":["#!/usr/bin/env node\r\n\r\nimport { program } from 'commander';\r\nimport { Client } from '@modelcontextprotocol/sdk/client/index.js';\r\nimport { Server } from '@modelcontextprotocol/sdk/server/index.js';\r\nimport { StdioServerTransport } from '@modelcontextprotocol/sdk/server/stdio.js';\r\nimport { StreamableHTTPClientTransport } from '@modelcontextprotocol/sdk/client/streamableHttp.js';\r\nimport {\r\n ListToolsRequestSchema,\r\n CallToolRequestSchema,\r\n ListResourcesRequestSchema,\r\n ReadResourceRequestSchema,\r\n ListPromptsRequestSchema,\r\n GetPromptRequestSchema\r\n} from '@modelcontextprotocol/sdk/types.js';\r\n\r\n// Helper: Sleep for specified milliseconds\r\nconst sleep = (ms: number) => new Promise(resolve => setTimeout(resolve, ms));\r\n\r\n// Helper: Check if error is retryable\r\nfunction isRetryableError(error: any): boolean {\r\n if (!error) return false;\r\n\r\n const errorStr = String(error).toLowerCase();\r\n const retryablePatterns = [\r\n 'econnreset',\r\n 'econnrefused',\r\n 'etimedout',\r\n 'timeout',\r\n 'network',\r\n 'socket hang up',\r\n '502',\r\n '503',\r\n '504'\r\n ];\r\n\r\n return retryablePatterns.some(pattern => errorStr.includes(pattern));\r\n}\r\n\r\n// Helper: Retry with exponential backoff\r\nasync function retryWithBackoff<T>(\r\n fn: () => Promise<T>,\r\n maxRetries: number = 3,\r\n baseDelay: number = 1000,\r\n debug: boolean = false\r\n): Promise<T> {\r\n let lastError: any;\r\n\r\n for (let attempt = 0; attempt <= maxRetries; attempt++) {\r\n try {\r\n return await fn();\r\n } catch (error) {\r\n lastError = error;\r\n\r\n if (attempt >= maxRetries) {\r\n throw error;\r\n }\r\n\r\n if (!isRetryableError(error)) {\r\n throw error;\r\n }\r\n\r\n const delay = Math.min(baseDelay * Math.pow(2, attempt), 30000); // Max 30s\r\n const jitter = Math.random() * 0.25 * delay;\r\n const totalDelay = Math.floor(delay + jitter);\r\n\r\n if (debug) {\r\n console.error(`[DEBUG] Retry attempt ${attempt + 1}/${maxRetries} after ${totalDelay}ms`);\r\n }\r\n\r\n await sleep(totalDelay);\r\n }\r\n }\r\n\r\n throw lastError;\r\n}\r\n\r\n// Helper: Connection monitor\r\nclass ConnectionMonitor {\r\n private lastSuccessfulCall: number = Date.now();\r\n private failureCount: number = 0;\r\n private totalCalls: number = 0;\r\n private debug: boolean;\r\n\r\n constructor(debug: boolean = false) {\r\n this.debug = debug;\r\n }\r\n\r\n async monitorCall<T>(operation: string, fn: () => Promise<T>): Promise<T> {\r\n this.totalCalls++;\r\n const startTime = Date.now();\r\n\r\n try {\r\n const result = await fn();\r\n this.lastSuccessfulCall = Date.now();\r\n this.failureCount = 0;\r\n\r\n const duration = Date.now() - startTime;\r\n if (this.debug && duration > 5000) {\r\n console.error(`[WARNING] Slow operation: ${operation} took ${duration}ms`);\r\n }\r\n\r\n return result;\r\n } catch (error) {\r\n this.failureCount++;\r\n\r\n const timeSinceSuccess = Date.now() - this.lastSuccessfulCall;\r\n if (timeSinceSuccess > 60000 && this.failureCount >= 3) {\r\n console.error('[WARNING] Connection appears degraded');\r\n console.error(`[WARNING] ${this.failureCount} failures in the last ${Math.floor(timeSinceSuccess / 1000)}s`);\r\n }\r\n\r\n throw error;\r\n }\r\n }\r\n}\r\n\r\n// Helper: Enhanced error message\r\nfunction getErrorHint(error: any, context: string): string {\r\n const errorStr = String(error).toLowerCase();\r\n\r\n if (errorStr.includes('401') || errorStr.includes('unauthorized')) {\r\n return 'Check your API token (--token sk-cust-...)';\r\n }\r\n if (errorStr.includes('403') || errorStr.includes('forbidden')) {\r\n return 'Token is valid but lacks permission. Contact your administrator.';\r\n }\r\n if (errorStr.includes('404')) {\r\n return 'Server endpoint not found. Verify --url is correct.';\r\n }\r\n if (errorStr.includes('timeout') || errorStr.includes('etimedout')) {\r\n return 'Request timed out. Server may be slow or overloaded.';\r\n }\r\n if (errorStr.includes('econnrefused')) {\r\n return 'Connection refused. Verify server is running and URL is correct.';\r\n }\r\n if (errorStr.includes('econnreset') || errorStr.includes('socket hang up')) {\r\n return 'Connection lost. Check network connectivity.';\r\n }\r\n if (errorStr.includes('dns') || errorStr.includes('getaddrinfo')) {\r\n return 'DNS lookup failed. Check server URL spelling.';\r\n }\r\n\r\n return 'See error details above';\r\n}\r\n\r\n// Parse command-line arguments\r\nprogram\r\n .name('priority-mcp-client')\r\n .description('MCP client that proxies Claude Desktop to Priority MCP server')\r\n .requiredOption('--url <url>', 'Priority MCP server URL (e.g., https://priority-mcp.leonai.io)')\r\n .requiredOption('--token <token>', 'Authentication token (sk-cust-...)')\r\n .requiredOption('--priority-url <url>', 'Priority ERP base URL')\r\n .requiredOption('--priority-company <company>', 'Priority company name')\r\n .requiredOption('--priority-username <username>', 'Priority API username')\r\n .requiredOption('--priority-password <password>', 'Priority API password')\r\n .option('--debug', 'Enable debug logging')\r\n .option('--verbose', 'Enable verbose logging (includes request/response details)')\r\n .parse();\r\n\r\nconst options = program.opts();\r\n\r\nasync function main() {\r\n try {\r\n const debug = options.debug || options.verbose || false;\r\n const verbose = options.verbose || false;\r\n\r\n if (debug) {\r\n console.error('[DEBUG] Starting Priority MCP Client v1.0.0');\r\n console.error(`[DEBUG] Server URL: ${options.url}`);\r\n }\r\n\r\n // Health check on startup\r\n if (debug) {\r\n console.error('[DEBUG] Performing health check...');\r\n }\r\n\r\n try {\r\n const healthUrl = `${options.url}/health`;\r\n const healthResponse = await retryWithBackoff(\r\n async () => {\r\n const response = await fetch(healthUrl);\r\n if (!response.ok) {\r\n throw new Error(`Health check failed: ${response.status} ${response.statusText}`);\r\n }\r\n return response.json();\r\n },\r\n 2, // 2 retries for health check\r\n 500, // 500ms base delay\r\n debug\r\n );\r\n\r\n if (debug) {\r\n console.error('[DEBUG] Health check passed:', JSON.stringify(healthResponse));\r\n }\r\n\r\n if (healthResponse.status !== 'healthy') {\r\n console.error('[WARNING] Server health check returned non-healthy status');\r\n console.error('[WARNING] Service may be degraded');\r\n }\r\n } catch (error) {\r\n console.error('[ERROR] Server health check failed');\r\n console.error('[ERROR]', error);\r\n console.error('[HINT]', getErrorHint(error, 'health check'));\r\n console.error('[HINT] Verify server is running: curl', `${options.url}/health`);\r\n process.exit(1);\r\n }\r\n\r\n // Initialize connection monitor\r\n const connectionMonitor = new ConnectionMonitor(debug);\r\n\r\n // Create MCP client that will connect to remote Priority server\r\n const client = new Client({\r\n name: 'priority-mcp-client',\r\n version: '1.0.0'\r\n }, {\r\n capabilities: {}\r\n });\r\n\r\n // Generate tenant ID from company+url for uniqueness\r\n // This allows server to track/rate-limit per customer without storing credentials\r\n const crypto = await import('crypto');\r\n const tenantIdSource = `${options.priorityCompany.toLowerCase()}@${options.priorityUrl.toLowerCase()}`;\r\n const tenantId = crypto.createHash('sha256').update(tenantIdSource).digest('hex').substring(0, 16);\r\n\r\n // Prepare headers for authentication\r\n const headers = {\r\n 'Authorization': `Bearer ${options.token}`,\r\n 'X-Tenant-ID': tenantId, // Unique identifier for rate limiting/tracking\r\n 'X-Priority-URL': options.priorityUrl,\r\n 'X-Priority-Company': options.priorityCompany,\r\n 'X-Priority-Username': options.priorityUsername,\r\n 'X-Priority-Password': options.priorityPassword\r\n };\r\n\r\n if (debug) {\r\n console.error('[DEBUG] Headers prepared (credentials hidden)');\r\n }\r\n\r\n // Connect to remote Priority MCP server via Streamable HTTP with timeout\r\n const mcpUrl = new URL('/mcp', options.url);\r\n const controller = new AbortController();\r\n const timeoutId = setTimeout(() => controller.abort(), 30000); // 30 second timeout\r\n\r\n const transport = new StreamableHTTPClientTransport(mcpUrl, {\r\n requestInit: {\r\n headers: headers,\r\n signal: controller.signal\r\n }\r\n });\r\n\r\n try {\r\n await retryWithBackoff(\r\n () => client.connect(transport),\r\n 3, // 3 retries for connection\r\n 1000, // 1s base delay\r\n debug\r\n );\r\n clearTimeout(timeoutId);\r\n\r\n if (debug) {\r\n console.error('[DEBUG] Connected to remote Priority MCP server');\r\n }\r\n } catch (error) {\r\n clearTimeout(timeoutId);\r\n console.error('[ERROR] Failed to connect to Priority MCP server');\r\n console.error('[ERROR]', error);\r\n console.error('[HINT]', getErrorHint(error, 'connection'));\r\n throw error;\r\n }\r\n\r\n // Create stdio server for Claude Desktop\r\n const server = new Server({\r\n name: 'priority-mcp-proxy',\r\n version: '1.0.0'\r\n }, {\r\n capabilities: {\r\n tools: {},\r\n resources: {},\r\n prompts: {}\r\n }\r\n });\r\n\r\n // Set up request handlers with retry logic and monitoring\r\n\r\n // List tools\r\n server.setRequestHandler(ListToolsRequestSchema, async () => {\r\n if (debug) console.error('[DEBUG] Forwarding tools/list');\r\n return await connectionMonitor.monitorCall('tools/list', async () => {\r\n return await retryWithBackoff(() => client.listTools(), 2, 500, debug);\r\n });\r\n });\r\n\r\n // Call tool (with retries for transient failures)\r\n server.setRequestHandler(CallToolRequestSchema, async (request) => {\r\n if (debug) console.error(`[DEBUG] Forwarding tools/call: ${request.params.name}`);\r\n if (verbose) console.error(`[VERBOSE] Tool params:`, JSON.stringify(request.params));\r\n\r\n try {\r\n const result = await connectionMonitor.monitorCall(`tools/call/${request.params.name}`, async () => {\r\n return await retryWithBackoff(() => client.callTool(request.params), 3, 1000, debug);\r\n });\r\n\r\n if (verbose) console.error(`[VERBOSE] Tool result:`, JSON.stringify(result).slice(0, 500));\r\n return result;\r\n } catch (error) {\r\n console.error(`[ERROR] Tool call failed: ${request.params.name}`);\r\n console.error(`[HINT]`, getErrorHint(error, 'tool call'));\r\n throw error;\r\n }\r\n });\r\n\r\n // List resources\r\n server.setRequestHandler(ListResourcesRequestSchema, async () => {\r\n if (debug) console.error('[DEBUG] Forwarding resources/list');\r\n return await connectionMonitor.monitorCall('resources/list', async () => {\r\n return await retryWithBackoff(() => client.listResources(), 2, 500, debug);\r\n });\r\n });\r\n\r\n // Read resource\r\n server.setRequestHandler(ReadResourceRequestSchema, async (request) => {\r\n if (debug) console.error(`[DEBUG] Forwarding resources/read: ${request.params.uri}`);\r\n return await connectionMonitor.monitorCall(`resources/read/${request.params.uri}`, async () => {\r\n return await retryWithBackoff(() => client.readResource(request.params), 3, 1000, debug);\r\n });\r\n });\r\n\r\n // List prompts\r\n server.setRequestHandler(ListPromptsRequestSchema, async () => {\r\n if (debug) console.error('[DEBUG] Forwarding prompts/list');\r\n return await connectionMonitor.monitorCall('prompts/list', async () => {\r\n return await retryWithBackoff(() => client.listPrompts(), 2, 500, debug);\r\n });\r\n });\r\n\r\n // Get prompt\r\n server.setRequestHandler(GetPromptRequestSchema, async (request) => {\r\n if (debug) console.error(`[DEBUG] Forwarding prompts/get: ${request.params.name}`);\r\n return await connectionMonitor.monitorCall(`prompts/get/${request.params.name}`, async () => {\r\n return await retryWithBackoff(() => client.getPrompt(request.params), 2, 500, debug);\r\n });\r\n });\r\n\r\n // Connect server to stdio for Claude Desktop\r\n const stdioTransport = new StdioServerTransport();\r\n await server.connect(stdioTransport);\r\n\r\n if (debug) {\r\n console.error('[DEBUG] Stdio server ready, proxy active');\r\n }\r\n\r\n // Keep process running\r\n process.on('SIGINT', async () => {\r\n if (debug) console.error('[DEBUG] Shutting down...');\r\n await server.close();\r\n await client.close();\r\n process.exit(0);\r\n });\r\n\r\n } catch (error) {\r\n console.error('Failed to start Priority MCP client:', error);\r\n process.exit(1);\r\n }\r\n}\r\n\r\nmain().catch((error) => {\r\n console.error('Fatal error:', error);\r\n process.exit(1);\r\n});\r\n"],"mappings":"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","names":[]}
|
package/package.json
CHANGED