@leonailtd/priority-mcp-client 0.1.0 → 0.2.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 CHANGED
@@ -14,23 +14,160 @@ import {
14
14
  ListPromptsRequestSchema,
15
15
  GetPromptRequestSchema
16
16
  } from "@modelcontextprotocol/sdk/types.js";
17
- 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").parse();
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] Starting Priority MCP Client");
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
- await client.connect(transport);
49
- if (debug) {
50
- console.error("[DEBUG] Connected to remote Priority MCP server");
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
- const result = await client.listTools();
65
- return result;
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
- const result = await client.callTool(request.params);
70
- return result;
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
- const result = await client.listResources();
75
- return result;
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
- const result = await client.readResource(request.params);
80
- return result;
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
- const result = await client.listPrompts();
85
- return result;
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
- const result = await client.getPrompt(request.params);
90
- return result;
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
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@leonailtd/priority-mcp-client",
3
- "version": "0.1.0",
3
+ "version": "0.2.0",
4
4
  "description": "MCP client for Priority ERP - connects Claude Desktop to remote Priority MCP server",
5
5
  "type": "module",
6
6
  "main": "./dist/bin.js",