@local-labs-jpollock/local-cli 0.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (86) hide show
  1. package/addon-dist/bin/mcp-stdio.js +2808 -0
  2. package/addon-dist/lib/common/constants.d.ts +22 -0
  3. package/addon-dist/lib/common/constants.js +26 -0
  4. package/addon-dist/lib/common/theme.d.ts +68 -0
  5. package/addon-dist/lib/common/theme.js +126 -0
  6. package/addon-dist/lib/common/types.d.ts +298 -0
  7. package/addon-dist/lib/common/types.js +6 -0
  8. package/addon-dist/lib/main/config/ConnectionInfo.d.ts +25 -0
  9. package/addon-dist/lib/main/config/ConnectionInfo.js +82 -0
  10. package/addon-dist/lib/main/index.d.ts +12 -0
  11. package/addon-dist/lib/main/index.js +3322 -0
  12. package/addon-dist/lib/main/mcp/McpAuth.d.ts +37 -0
  13. package/addon-dist/lib/main/mcp/McpAuth.js +87 -0
  14. package/addon-dist/lib/main/mcp/McpServer.d.ts +67 -0
  15. package/addon-dist/lib/main/mcp/McpServer.js +343 -0
  16. package/addon-dist/lib/main/mcp/tools/changePhpVersion.d.ts +7 -0
  17. package/addon-dist/lib/main/mcp/tools/changePhpVersion.js +81 -0
  18. package/addon-dist/lib/main/mcp/tools/cloneSite.d.ts +7 -0
  19. package/addon-dist/lib/main/mcp/tools/cloneSite.js +66 -0
  20. package/addon-dist/lib/main/mcp/tools/createSite.d.ts +7 -0
  21. package/addon-dist/lib/main/mcp/tools/createSite.js +137 -0
  22. package/addon-dist/lib/main/mcp/tools/deleteSite.d.ts +7 -0
  23. package/addon-dist/lib/main/mcp/tools/deleteSite.js +72 -0
  24. package/addon-dist/lib/main/mcp/tools/exportDatabase.d.ts +7 -0
  25. package/addon-dist/lib/main/mcp/tools/exportDatabase.js +72 -0
  26. package/addon-dist/lib/main/mcp/tools/exportSite.d.ts +7 -0
  27. package/addon-dist/lib/main/mcp/tools/exportSite.js +103 -0
  28. package/addon-dist/lib/main/mcp/tools/getLocalInfo.d.ts +7 -0
  29. package/addon-dist/lib/main/mcp/tools/getLocalInfo.js +72 -0
  30. package/addon-dist/lib/main/mcp/tools/getSite.d.ts +7 -0
  31. package/addon-dist/lib/main/mcp/tools/getSite.js +68 -0
  32. package/addon-dist/lib/main/mcp/tools/getSiteLogs.d.ts +7 -0
  33. package/addon-dist/lib/main/mcp/tools/getSiteLogs.js +149 -0
  34. package/addon-dist/lib/main/mcp/tools/helpers.d.ts +59 -0
  35. package/addon-dist/lib/main/mcp/tools/helpers.js +179 -0
  36. package/addon-dist/lib/main/mcp/tools/importDatabase.d.ts +7 -0
  37. package/addon-dist/lib/main/mcp/tools/importDatabase.js +109 -0
  38. package/addon-dist/lib/main/mcp/tools/importSite.d.ts +7 -0
  39. package/addon-dist/lib/main/mcp/tools/importSite.js +149 -0
  40. package/addon-dist/lib/main/mcp/tools/index.d.ts +26 -0
  41. package/addon-dist/lib/main/mcp/tools/index.js +117 -0
  42. package/addon-dist/lib/main/mcp/tools/listBlueprints.d.ts +7 -0
  43. package/addon-dist/lib/main/mcp/tools/listBlueprints.js +54 -0
  44. package/addon-dist/lib/main/mcp/tools/listServices.d.ts +7 -0
  45. package/addon-dist/lib/main/mcp/tools/listServices.js +112 -0
  46. package/addon-dist/lib/main/mcp/tools/listSites.d.ts +7 -0
  47. package/addon-dist/lib/main/mcp/tools/listSites.js +62 -0
  48. package/addon-dist/lib/main/mcp/tools/openAdminer.d.ts +7 -0
  49. package/addon-dist/lib/main/mcp/tools/openAdminer.js +59 -0
  50. package/addon-dist/lib/main/mcp/tools/openSite.d.ts +7 -0
  51. package/addon-dist/lib/main/mcp/tools/openSite.js +62 -0
  52. package/addon-dist/lib/main/mcp/tools/renameSite.d.ts +7 -0
  53. package/addon-dist/lib/main/mcp/tools/renameSite.js +70 -0
  54. package/addon-dist/lib/main/mcp/tools/restartSite.d.ts +7 -0
  55. package/addon-dist/lib/main/mcp/tools/restartSite.js +56 -0
  56. package/addon-dist/lib/main/mcp/tools/saveBlueprint.d.ts +7 -0
  57. package/addon-dist/lib/main/mcp/tools/saveBlueprint.js +89 -0
  58. package/addon-dist/lib/main/mcp/tools/startSite.d.ts +7 -0
  59. package/addon-dist/lib/main/mcp/tools/startSite.js +54 -0
  60. package/addon-dist/lib/main/mcp/tools/stopSite.d.ts +7 -0
  61. package/addon-dist/lib/main/mcp/tools/stopSite.js +54 -0
  62. package/addon-dist/lib/main/mcp/tools/toggleXdebug.d.ts +7 -0
  63. package/addon-dist/lib/main/mcp/tools/toggleXdebug.js +69 -0
  64. package/addon-dist/lib/main/mcp/tools/trustSsl.d.ts +7 -0
  65. package/addon-dist/lib/main/mcp/tools/trustSsl.js +59 -0
  66. package/addon-dist/lib/main/mcp/tools/wpCli.d.ts +7 -0
  67. package/addon-dist/lib/main/mcp/tools/wpCli.js +110 -0
  68. package/addon-dist/lib/main.d.ts +1 -0
  69. package/addon-dist/lib/main.js +10 -0
  70. package/addon-dist/lib/renderer/index.d.ts +7 -0
  71. package/addon-dist/lib/renderer/index.js +479 -0
  72. package/addon-dist/package.json +73 -0
  73. package/bin/lwp.js +10 -0
  74. package/lib/bootstrap/index.d.ts +98 -0
  75. package/lib/bootstrap/index.js +493 -0
  76. package/lib/bootstrap/paths.d.ts +28 -0
  77. package/lib/bootstrap/paths.js +96 -0
  78. package/lib/client/GraphQLClient.d.ts +38 -0
  79. package/lib/client/GraphQLClient.js +71 -0
  80. package/lib/client/index.d.ts +4 -0
  81. package/lib/client/index.js +10 -0
  82. package/lib/formatters/index.d.ts +75 -0
  83. package/lib/formatters/index.js +139 -0
  84. package/lib/index.d.ts +8 -0
  85. package/lib/index.js +1173 -0
  86. package/package.json +72 -0
@@ -0,0 +1,37 @@
1
+ /**
2
+ * MCP Authentication Manager
3
+ * Handles token generation and validation
4
+ */
5
+ export declare class McpAuth {
6
+ private token;
7
+ private logger;
8
+ constructor(logger: any);
9
+ /**
10
+ * Generate a new authentication token
11
+ */
12
+ generateToken(): string;
13
+ /**
14
+ * Get current token or generate if not exists
15
+ */
16
+ getToken(): string;
17
+ /**
18
+ * Set token (for loading from saved config)
19
+ */
20
+ setToken(token: string): void;
21
+ /**
22
+ * Regenerate the authentication token
23
+ */
24
+ regenerateToken(): string;
25
+ /**
26
+ * Validate provided token against stored token
27
+ */
28
+ validateToken(providedToken: string | undefined): boolean;
29
+ /**
30
+ * Validate that request comes from trusted IP
31
+ */
32
+ validateIP(ip: string | undefined): boolean;
33
+ /**
34
+ * Extract Bearer token from Authorization header
35
+ */
36
+ extractBearerToken(authHeader: string | undefined): string | undefined;
37
+ }
@@ -0,0 +1,87 @@
1
+ "use strict";
2
+ /**
3
+ * MCP Authentication Manager
4
+ * Handles token generation and validation
5
+ */
6
+ var __importDefault = (this && this.__importDefault) || function (mod) {
7
+ return (mod && mod.__esModule) ? mod : { "default": mod };
8
+ };
9
+ Object.defineProperty(exports, "__esModule", { value: true });
10
+ exports.McpAuth = void 0;
11
+ const crypto_1 = __importDefault(require("crypto"));
12
+ const constants_1 = require("../../common/constants");
13
+ class McpAuth {
14
+ constructor(logger) {
15
+ this.token = null;
16
+ this.logger = logger;
17
+ }
18
+ /**
19
+ * Generate a new authentication token
20
+ */
21
+ generateToken() {
22
+ this.token = crypto_1.default.randomBytes(constants_1.AUTH_TOKEN_LENGTH / 2).toString('base64');
23
+ this.logger.info(`[MCP] Generated new authentication token: ${this.token.substring(0, 20)}...`);
24
+ return this.token;
25
+ }
26
+ /**
27
+ * Get current token or generate if not exists
28
+ */
29
+ getToken() {
30
+ if (!this.token) {
31
+ return this.generateToken();
32
+ }
33
+ return this.token;
34
+ }
35
+ /**
36
+ * Set token (for loading from saved config)
37
+ */
38
+ setToken(token) {
39
+ this.token = token;
40
+ }
41
+ /**
42
+ * Regenerate the authentication token
43
+ */
44
+ regenerateToken() {
45
+ this.logger.info('[MCP] Regenerating authentication token');
46
+ return this.generateToken();
47
+ }
48
+ /**
49
+ * Validate provided token against stored token
50
+ */
51
+ validateToken(providedToken) {
52
+ if (!this.token || !providedToken) {
53
+ this.logger.warn(`[MCP Auth] Validation failed: token=${!!this.token}, provided=${!!providedToken}`);
54
+ return false;
55
+ }
56
+ // Support both raw token and Bearer prefix
57
+ const rawToken = providedToken.startsWith('Bearer ') ? providedToken.slice(7) : providedToken;
58
+ const isValid = rawToken === this.token;
59
+ if (!isValid) {
60
+ this.logger.warn(`[MCP Auth] Token mismatch: provided=${rawToken.substring(0, 20)}... stored=${this.token.substring(0, 20)}...`);
61
+ }
62
+ return isValid;
63
+ }
64
+ /**
65
+ * Validate that request comes from trusted IP
66
+ */
67
+ validateIP(ip) {
68
+ if (!ip) {
69
+ return false;
70
+ }
71
+ return constants_1.TRUSTED_IPS.includes(ip);
72
+ }
73
+ /**
74
+ * Extract Bearer token from Authorization header
75
+ */
76
+ extractBearerToken(authHeader) {
77
+ if (!authHeader) {
78
+ return undefined;
79
+ }
80
+ if (authHeader.startsWith('Bearer ')) {
81
+ return authHeader.slice(7);
82
+ }
83
+ return authHeader;
84
+ }
85
+ }
86
+ exports.McpAuth = McpAuth;
87
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiTWNwQXV0aC5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uL3NyYy9tYWluL21jcC9NY3BBdXRoLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7O0dBR0c7Ozs7OztBQUVILG9EQUE0QjtBQUM1QixzREFBd0U7QUFFeEUsTUFBYSxPQUFPO0lBSWxCLFlBQVksTUFBVztRQUhmLFVBQUssR0FBa0IsSUFBSSxDQUFDO1FBSWxDLElBQUksQ0FBQyxNQUFNLEdBQUcsTUFBTSxDQUFDO0lBQ3ZCLENBQUM7SUFFRDs7T0FFRztJQUNILGFBQWE7UUFDWCxJQUFJLENBQUMsS0FBSyxHQUFHLGdCQUFNLENBQUMsV0FBVyxDQUFDLDZCQUFpQixHQUFHLENBQUMsQ0FBQyxDQUFDLFFBQVEsQ0FBQyxRQUFRLENBQUMsQ0FBQztRQUMxRSxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FBQyw2Q0FBNkMsSUFBSSxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxLQUFLLENBQUMsQ0FBQztRQUNoRyxPQUFPLElBQUksQ0FBQyxLQUFLLENBQUM7SUFDcEIsQ0FBQztJQUVEOztPQUVHO0lBQ0gsUUFBUTtRQUNOLElBQUksQ0FBQyxJQUFJLENBQUMsS0FBSyxFQUFFLENBQUM7WUFDaEIsT0FBTyxJQUFJLENBQUMsYUFBYSxFQUFFLENBQUM7UUFDOUIsQ0FBQztRQUNELE9BQU8sSUFBSSxDQUFDLEtBQUssQ0FBQztJQUNwQixDQUFDO0lBRUQ7O09BRUc7SUFDSCxRQUFRLENBQUMsS0FBYTtRQUNwQixJQUFJLENBQUMsS0FBSyxHQUFHLEtBQUssQ0FBQztJQUNyQixDQUFDO0lBRUQ7O09BRUc7SUFDSCxlQUFlO1FBQ2IsSUFBSSxDQUFDLE1BQU0sQ0FBQyxJQUFJLENBQUMseUNBQXlDLENBQUMsQ0FBQztRQUM1RCxPQUFPLElBQUksQ0FBQyxhQUFhLEVBQUUsQ0FBQztJQUM5QixDQUFDO0lBRUQ7O09BRUc7SUFDSCxhQUFhLENBQUMsYUFBaUM7UUFDN0MsSUFBSSxDQUFDLElBQUksQ0FBQyxLQUFLLElBQUksQ0FBQyxhQUFhLEVBQUUsQ0FBQztZQUNsQyxJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FDZCx1Q0FBdUMsQ0FBQyxDQUFDLElBQUksQ0FBQyxLQUFLLGNBQWMsQ0FBQyxDQUFDLGFBQWEsRUFBRSxDQUNuRixDQUFDO1lBQ0YsT0FBTyxLQUFLLENBQUM7UUFDZixDQUFDO1FBRUQsMkNBQTJDO1FBQzNDLE1BQU0sUUFBUSxHQUFHLGFBQWEsQ0FBQyxVQUFVLENBQUMsU0FBUyxDQUFDLENBQUMsQ0FBQyxDQUFDLGFBQWEsQ0FBQyxLQUFLLENBQUMsQ0FBQyxDQUFDLENBQUMsQ0FBQyxDQUFDLGFBQWEsQ0FBQztRQUU5RixNQUFNLE9BQU8sR0FBRyxRQUFRLEtBQUssSUFBSSxDQUFDLEtBQUssQ0FBQztRQUN4QyxJQUFJLENBQUMsT0FBTyxFQUFFLENBQUM7WUFDYixJQUFJLENBQUMsTUFBTSxDQUFDLElBQUksQ0FDZCx1Q0FBdUMsUUFBUSxDQUFDLFNBQVMsQ0FBQyxDQUFDLEVBQUUsRUFBRSxDQUFDLGNBQWMsSUFBSSxDQUFDLEtBQUssQ0FBQyxTQUFTLENBQUMsQ0FBQyxFQUFFLEVBQUUsQ0FBQyxLQUFLLENBQy9HLENBQUM7UUFDSixDQUFDO1FBQ0QsT0FBTyxPQUFPLENBQUM7SUFDakIsQ0FBQztJQUVEOztPQUVHO0lBQ0gsVUFBVSxDQUFDLEVBQXNCO1FBQy9CLElBQUksQ0FBQyxFQUFFLEVBQUUsQ0FBQztZQUNSLE9BQU8sS0FBSyxDQUFDO1FBQ2YsQ0FBQztRQUNELE9BQVEsdUJBQWlDLENBQUMsUUFBUSxDQUFDLEVBQUUsQ0FBQyxDQUFDO0lBQ3pELENBQUM7SUFFRDs7T0FFRztJQUNILGtCQUFrQixDQUFDLFVBQThCO1FBQy9DLElBQUksQ0FBQyxVQUFVLEVBQUUsQ0FBQztZQUNoQixPQUFPLFNBQVMsQ0FBQztRQUNuQixDQUFDO1FBRUQsSUFBSSxVQUFVLENBQUMsVUFBVSxDQUFDLFNBQVMsQ0FBQyxFQUFFLENBQUM7WUFDckMsT0FBTyxVQUFVLENBQUMsS0FBSyxDQUFDLENBQUMsQ0FBQyxDQUFDO1FBQzdCLENBQUM7UUFFRCxPQUFPLFVBQVUsQ0FBQztJQUNwQixDQUFDO0NBQ0Y7QUF6RkQsMEJBeUZDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBNQ1AgQXV0aGVudGljYXRpb24gTWFuYWdlclxuICogSGFuZGxlcyB0b2tlbiBnZW5lcmF0aW9uIGFuZCB2YWxpZGF0aW9uXG4gKi9cblxuaW1wb3J0IGNyeXB0byBmcm9tICdjcnlwdG8nO1xuaW1wb3J0IHsgQVVUSF9UT0tFTl9MRU5HVEgsIFRSVVNURURfSVBTIH0gZnJvbSAnLi4vLi4vY29tbW9uL2NvbnN0YW50cyc7XG5cbmV4cG9ydCBjbGFzcyBNY3BBdXRoIHtcbiAgcHJpdmF0ZSB0b2tlbjogc3RyaW5nIHwgbnVsbCA9IG51bGw7XG4gIHByaXZhdGUgbG9nZ2VyOiBhbnk7XG5cbiAgY29uc3RydWN0b3IobG9nZ2VyOiBhbnkpIHtcbiAgICB0aGlzLmxvZ2dlciA9IGxvZ2dlcjtcbiAgfVxuXG4gIC8qKlxuICAgKiBHZW5lcmF0ZSBhIG5ldyBhdXRoZW50aWNhdGlvbiB0b2tlblxuICAgKi9cbiAgZ2VuZXJhdGVUb2tlbigpOiBzdHJpbmcge1xuICAgIHRoaXMudG9rZW4gPSBjcnlwdG8ucmFuZG9tQnl0ZXMoQVVUSF9UT0tFTl9MRU5HVEggLyAyKS50b1N0cmluZygnYmFzZTY0Jyk7XG4gICAgdGhpcy5sb2dnZXIuaW5mbyhgW01DUF0gR2VuZXJhdGVkIG5ldyBhdXRoZW50aWNhdGlvbiB0b2tlbjogJHt0aGlzLnRva2VuLnN1YnN0cmluZygwLCAyMCl9Li4uYCk7XG4gICAgcmV0dXJuIHRoaXMudG9rZW47XG4gIH1cblxuICAvKipcbiAgICogR2V0IGN1cnJlbnQgdG9rZW4gb3IgZ2VuZXJhdGUgaWYgbm90IGV4aXN0c1xuICAgKi9cbiAgZ2V0VG9rZW4oKTogc3RyaW5nIHtcbiAgICBpZiAoIXRoaXMudG9rZW4pIHtcbiAgICAgIHJldHVybiB0aGlzLmdlbmVyYXRlVG9rZW4oKTtcbiAgICB9XG4gICAgcmV0dXJuIHRoaXMudG9rZW47XG4gIH1cblxuICAvKipcbiAgICogU2V0IHRva2VuIChmb3IgbG9hZGluZyBmcm9tIHNhdmVkIGNvbmZpZylcbiAgICovXG4gIHNldFRva2VuKHRva2VuOiBzdHJpbmcpOiB2b2lkIHtcbiAgICB0aGlzLnRva2VuID0gdG9rZW47XG4gIH1cblxuICAvKipcbiAgICogUmVnZW5lcmF0ZSB0aGUgYXV0aGVudGljYXRpb24gdG9rZW5cbiAgICovXG4gIHJlZ2VuZXJhdGVUb2tlbigpOiBzdHJpbmcge1xuICAgIHRoaXMubG9nZ2VyLmluZm8oJ1tNQ1BdIFJlZ2VuZXJhdGluZyBhdXRoZW50aWNhdGlvbiB0b2tlbicpO1xuICAgIHJldHVybiB0aGlzLmdlbmVyYXRlVG9rZW4oKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBWYWxpZGF0ZSBwcm92aWRlZCB0b2tlbiBhZ2FpbnN0IHN0b3JlZCB0b2tlblxuICAgKi9cbiAgdmFsaWRhdGVUb2tlbihwcm92aWRlZFRva2VuOiBzdHJpbmcgfCB1bmRlZmluZWQpOiBib29sZWFuIHtcbiAgICBpZiAoIXRoaXMudG9rZW4gfHwgIXByb3ZpZGVkVG9rZW4pIHtcbiAgICAgIHRoaXMubG9nZ2VyLndhcm4oXG4gICAgICAgIGBbTUNQIEF1dGhdIFZhbGlkYXRpb24gZmFpbGVkOiB0b2tlbj0keyEhdGhpcy50b2tlbn0sIHByb3ZpZGVkPSR7ISFwcm92aWRlZFRva2VufWBcbiAgICAgICk7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuXG4gICAgLy8gU3VwcG9ydCBib3RoIHJhdyB0b2tlbiBhbmQgQmVhcmVyIHByZWZpeFxuICAgIGNvbnN0IHJhd1Rva2VuID0gcHJvdmlkZWRUb2tlbi5zdGFydHNXaXRoKCdCZWFyZXIgJykgPyBwcm92aWRlZFRva2VuLnNsaWNlKDcpIDogcHJvdmlkZWRUb2tlbjtcblxuICAgIGNvbnN0IGlzVmFsaWQgPSByYXdUb2tlbiA9PT0gdGhpcy50b2tlbjtcbiAgICBpZiAoIWlzVmFsaWQpIHtcbiAgICAgIHRoaXMubG9nZ2VyLndhcm4oXG4gICAgICAgIGBbTUNQIEF1dGhdIFRva2VuIG1pc21hdGNoOiBwcm92aWRlZD0ke3Jhd1Rva2VuLnN1YnN0cmluZygwLCAyMCl9Li4uIHN0b3JlZD0ke3RoaXMudG9rZW4uc3Vic3RyaW5nKDAsIDIwKX0uLi5gXG4gICAgICApO1xuICAgIH1cbiAgICByZXR1cm4gaXNWYWxpZDtcbiAgfVxuXG4gIC8qKlxuICAgKiBWYWxpZGF0ZSB0aGF0IHJlcXVlc3QgY29tZXMgZnJvbSB0cnVzdGVkIElQXG4gICAqL1xuICB2YWxpZGF0ZUlQKGlwOiBzdHJpbmcgfCB1bmRlZmluZWQpOiBib29sZWFuIHtcbiAgICBpZiAoIWlwKSB7XG4gICAgICByZXR1cm4gZmFsc2U7XG4gICAgfVxuICAgIHJldHVybiAoVFJVU1RFRF9JUFMgYXMgcmVhZG9ubHkgc3RyaW5nW10pLmluY2x1ZGVzKGlwKTtcbiAgfVxuXG4gIC8qKlxuICAgKiBFeHRyYWN0IEJlYXJlciB0b2tlbiBmcm9tIEF1dGhvcml6YXRpb24gaGVhZGVyXG4gICAqL1xuICBleHRyYWN0QmVhcmVyVG9rZW4oYXV0aEhlYWRlcjogc3RyaW5nIHwgdW5kZWZpbmVkKTogc3RyaW5nIHwgdW5kZWZpbmVkIHtcbiAgICBpZiAoIWF1dGhIZWFkZXIpIHtcbiAgICAgIHJldHVybiB1bmRlZmluZWQ7XG4gICAgfVxuXG4gICAgaWYgKGF1dGhIZWFkZXIuc3RhcnRzV2l0aCgnQmVhcmVyICcpKSB7XG4gICAgICByZXR1cm4gYXV0aEhlYWRlci5zbGljZSg3KTtcbiAgICB9XG5cbiAgICByZXR1cm4gYXV0aEhlYWRlcjtcbiAgfVxufVxuIl19
@@ -0,0 +1,67 @@
1
+ /**
2
+ * MCP Server
3
+ * Main server class that handles HTTP/SSE connections for MCP protocol
4
+ */
5
+ import { McpConnectionInfo, McpServerConfig, McpServerStatus, LocalServices } from '../../common/types';
6
+ export declare class McpServer {
7
+ private server;
8
+ private port;
9
+ private auth;
10
+ private connectionInfo;
11
+ private services;
12
+ private logger;
13
+ private startTime;
14
+ constructor(config: McpServerConfig, services: LocalServices, logger: any);
15
+ /**
16
+ * Start the MCP server
17
+ */
18
+ start(): Promise<void>;
19
+ /**
20
+ * Stop the MCP server
21
+ */
22
+ stop(): Promise<void>;
23
+ /**
24
+ * Check if server is running
25
+ */
26
+ isRunning(): boolean;
27
+ /**
28
+ * Get server status
29
+ */
30
+ getStatus(): McpServerStatus;
31
+ /**
32
+ * Get connection info for external tools
33
+ */
34
+ getConnectionInfo(): McpConnectionInfo;
35
+ /**
36
+ * Regenerate the authentication token
37
+ */
38
+ regenerateToken(): Promise<string>;
39
+ /**
40
+ * Find an available port starting from the preferred port
41
+ */
42
+ private findAvailablePort;
43
+ /**
44
+ * Handle incoming HTTP requests
45
+ */
46
+ private handleRequest;
47
+ /**
48
+ * Handle health check requests
49
+ */
50
+ private handleHealth;
51
+ /**
52
+ * Handle SSE connection for MCP protocol
53
+ */
54
+ private handleSSE;
55
+ /**
56
+ * Handle MCP message requests
57
+ */
58
+ private handleMessage;
59
+ /**
60
+ * Process an MCP message and return response
61
+ */
62
+ private processMessage;
63
+ /**
64
+ * Send an error response
65
+ */
66
+ private sendError;
67
+ }
@@ -0,0 +1,343 @@
1
+ "use strict";
2
+ /**
3
+ * MCP Server
4
+ * Main server class that handles HTTP/SSE connections for MCP protocol
5
+ */
6
+ var __importDefault = (this && this.__importDefault) || function (mod) {
7
+ return (mod && mod.__esModule) ? mod : { "default": mod };
8
+ };
9
+ Object.defineProperty(exports, "__esModule", { value: true });
10
+ exports.McpServer = void 0;
11
+ const http_1 = __importDefault(require("http"));
12
+ const url_1 = require("url");
13
+ const constants_1 = require("../../common/constants");
14
+ const McpAuth_1 = require("./McpAuth");
15
+ const ConnectionInfo_1 = require("../config/ConnectionInfo");
16
+ const tools_1 = require("./tools");
17
+ class McpServer {
18
+ constructor(config, services, logger) {
19
+ this.server = null;
20
+ this.startTime = 0;
21
+ this.port = config.port || constants_1.MCP_SERVER.DEFAULT_PORT;
22
+ this.services = services;
23
+ this.logger = logger;
24
+ this.auth = new McpAuth_1.McpAuth(logger);
25
+ this.connectionInfo = new ConnectionInfo_1.ConnectionInfoManager(logger);
26
+ // Set token if provided
27
+ if (config.authToken) {
28
+ this.auth.setToken(config.authToken);
29
+ }
30
+ // Register all tools
31
+ (0, tools_1.registerTools)();
32
+ }
33
+ /**
34
+ * Start the MCP server
35
+ */
36
+ async start() {
37
+ if (this.server) {
38
+ this.logger.warn('[MCP] Server already running');
39
+ return;
40
+ }
41
+ // Try to load existing token from connection info (for persistence across restarts)
42
+ const existingInfo = await this.connectionInfo.load();
43
+ if (existingInfo?.authToken) {
44
+ this.auth.setToken(existingInfo.authToken);
45
+ this.logger.info(`[MCP] Loaded existing auth token: ${existingInfo.authToken.substring(0, 20)}...`);
46
+ }
47
+ // Find available port
48
+ this.port = await this.findAvailablePort(this.port);
49
+ return new Promise((resolve, reject) => {
50
+ this.server = http_1.default.createServer((req, res) => this.handleRequest(req, res));
51
+ this.server.on('error', (err) => {
52
+ if (err.code === 'EADDRINUSE') {
53
+ this.logger.error(`[MCP] Port ${this.port} is in use`);
54
+ reject(new Error(`Port ${this.port} is already in use`));
55
+ }
56
+ else {
57
+ this.logger.error('[MCP] Server error:', err);
58
+ reject(err);
59
+ }
60
+ });
61
+ this.server.listen(this.port, '127.0.0.1', async () => {
62
+ this.startTime = Date.now();
63
+ this.logger.info(`[MCP] Server started on http://127.0.0.1:${this.port}`);
64
+ // Save connection info
65
+ const info = this.getConnectionInfo();
66
+ await this.connectionInfo.save(info);
67
+ resolve();
68
+ });
69
+ });
70
+ }
71
+ /**
72
+ * Stop the MCP server
73
+ */
74
+ async stop() {
75
+ if (!this.server) {
76
+ return;
77
+ }
78
+ return new Promise((resolve) => {
79
+ this.server.close(() => {
80
+ this.logger.info('[MCP] Server stopped');
81
+ this.server = null;
82
+ this.connectionInfo.delete();
83
+ resolve();
84
+ });
85
+ });
86
+ }
87
+ /**
88
+ * Check if server is running
89
+ */
90
+ isRunning() {
91
+ return this.server !== null;
92
+ }
93
+ /**
94
+ * Get server status
95
+ */
96
+ getStatus() {
97
+ return {
98
+ running: this.isRunning(),
99
+ port: this.port,
100
+ uptime: this.isRunning() ? (Date.now() - this.startTime) / 1000 : 0,
101
+ };
102
+ }
103
+ /**
104
+ * Get connection info for external tools
105
+ */
106
+ getConnectionInfo() {
107
+ return {
108
+ url: `http://127.0.0.1:${this.port}`,
109
+ authToken: this.auth.getToken(),
110
+ port: this.port,
111
+ version: constants_1.MCP_SERVER.VERSION,
112
+ tools: (0, tools_1.getToolNames)(),
113
+ };
114
+ }
115
+ /**
116
+ * Regenerate the authentication token
117
+ */
118
+ async regenerateToken() {
119
+ const newToken = this.auth.regenerateToken();
120
+ // Update connection info file with new token
121
+ const info = this.getConnectionInfo();
122
+ await this.connectionInfo.save(info);
123
+ this.logger.info(`[MCP] Auth token regenerated`);
124
+ return newToken;
125
+ }
126
+ /**
127
+ * Find an available port starting from the preferred port
128
+ */
129
+ async findAvailablePort(preferredPort) {
130
+ const isPortAvailable = (port) => {
131
+ return new Promise((resolve) => {
132
+ const testServer = http_1.default.createServer();
133
+ testServer.once('error', () => resolve(false));
134
+ testServer.once('listening', () => {
135
+ testServer.close(() => resolve(true));
136
+ });
137
+ testServer.listen(port, '127.0.0.1');
138
+ });
139
+ };
140
+ for (let port = preferredPort; port <= constants_1.MCP_SERVER.PORT_RANGE.MAX; port++) {
141
+ if (await isPortAvailable(port)) {
142
+ if (port !== preferredPort) {
143
+ this.logger.info(`[MCP] Port ${preferredPort} unavailable, using ${port}`);
144
+ }
145
+ return port;
146
+ }
147
+ }
148
+ throw new Error(`No available ports in range ${constants_1.MCP_SERVER.PORT_RANGE.MIN}-${constants_1.MCP_SERVER.PORT_RANGE.MAX}`);
149
+ }
150
+ /**
151
+ * Handle incoming HTTP requests
152
+ */
153
+ async handleRequest(req, res) {
154
+ const url = new url_1.URL(req.url || '/', `http://127.0.0.1:${this.port}`);
155
+ const pathname = url.pathname;
156
+ // IP check
157
+ if (!this.auth.validateIP(req.socket.remoteAddress)) {
158
+ this.sendError(res, 403, 'Forbidden: Only localhost connections allowed');
159
+ return;
160
+ }
161
+ // Health endpoint (no auth required)
162
+ if (pathname === constants_1.MCP_ENDPOINTS.HEALTH) {
163
+ this.handleHealth(req, res);
164
+ return;
165
+ }
166
+ // Auth check for all other endpoints
167
+ const authHeader = req.headers.authorization;
168
+ if (!this.auth.validateToken(authHeader)) {
169
+ this.sendError(res, 401, 'Unauthorized: Invalid or missing authentication token');
170
+ return;
171
+ }
172
+ // Handle CORS preflight
173
+ if (req.method === 'OPTIONS') {
174
+ res.writeHead(200, {
175
+ 'Access-Control-Allow-Origin': '*',
176
+ 'Access-Control-Allow-Methods': 'GET, POST, OPTIONS',
177
+ 'Access-Control-Allow-Headers': 'Content-Type, Authorization',
178
+ 'Access-Control-Max-Age': '86400',
179
+ });
180
+ res.end();
181
+ return;
182
+ }
183
+ // Route to appropriate handler
184
+ if (pathname === constants_1.MCP_ENDPOINTS.SSE && req.method === 'GET') {
185
+ this.handleSSE(req, res);
186
+ }
187
+ else if (pathname === constants_1.MCP_ENDPOINTS.MESSAGES && req.method === 'POST') {
188
+ await this.handleMessage(req, res);
189
+ }
190
+ else {
191
+ this.sendError(res, 404, 'Not Found');
192
+ }
193
+ }
194
+ /**
195
+ * Handle health check requests
196
+ */
197
+ handleHealth(req, res) {
198
+ const status = this.getStatus();
199
+ const token = this.auth.getToken();
200
+ res.writeHead(200, { 'Content-Type': 'application/json' });
201
+ res.end(JSON.stringify({
202
+ status: 'ok',
203
+ version: constants_1.MCP_SERVER.VERSION,
204
+ uptime: status.uptime,
205
+ tools: (0, tools_1.getToolNames)(),
206
+ tokenPrefix: token.substring(0, 20),
207
+ }));
208
+ }
209
+ /**
210
+ * Handle SSE connection for MCP protocol
211
+ */
212
+ handleSSE(req, res) {
213
+ res.writeHead(200, {
214
+ 'Content-Type': 'text/event-stream',
215
+ 'Cache-Control': 'no-cache',
216
+ Connection: 'keep-alive',
217
+ 'Access-Control-Allow-Origin': '*',
218
+ });
219
+ // Send endpoint information
220
+ const messagesUrl = `http://127.0.0.1:${this.port}${constants_1.MCP_ENDPOINTS.MESSAGES}`;
221
+ res.write(`event: endpoint\ndata: ${messagesUrl}\n\n`);
222
+ // Keep connection alive
223
+ const keepAlive = setInterval(() => {
224
+ res.write(': keepalive\n\n');
225
+ }, 30000);
226
+ req.on('close', () => {
227
+ clearInterval(keepAlive);
228
+ });
229
+ }
230
+ /**
231
+ * Handle MCP message requests
232
+ */
233
+ async handleMessage(req, res) {
234
+ let body = '';
235
+ req.on('data', (chunk) => {
236
+ body += chunk;
237
+ });
238
+ req.on('end', async () => {
239
+ try {
240
+ const request = JSON.parse(body);
241
+ // Notifications don't have an id and don't expect a response
242
+ if (request.method?.startsWith('notifications/')) {
243
+ this.logger.info(`[MCP] Received notification: ${request.method}`);
244
+ res.writeHead(200, { 'Content-Type': 'application/json' });
245
+ res.end('{}');
246
+ return;
247
+ }
248
+ const response = await this.processMessage(request);
249
+ res.writeHead(200, {
250
+ 'Content-Type': 'application/json',
251
+ 'Access-Control-Allow-Origin': '*',
252
+ });
253
+ res.end(JSON.stringify(response));
254
+ }
255
+ catch (error) {
256
+ this.logger.error('[MCP] Message handling error:', error);
257
+ this.sendError(res, 400, `Invalid request: ${error.message}`);
258
+ }
259
+ });
260
+ }
261
+ /**
262
+ * Process an MCP message and return response
263
+ */
264
+ async processMessage(request) {
265
+ const { id, method, params } = request;
266
+ switch (method) {
267
+ case 'initialize':
268
+ return {
269
+ jsonrpc: '2.0',
270
+ id,
271
+ result: {
272
+ protocolVersion: '2024-11-05',
273
+ capabilities: {
274
+ tools: {},
275
+ },
276
+ serverInfo: {
277
+ name: constants_1.MCP_SERVER.NAME,
278
+ version: constants_1.MCP_SERVER.VERSION,
279
+ },
280
+ },
281
+ };
282
+ case 'tools/list':
283
+ return {
284
+ jsonrpc: '2.0',
285
+ id,
286
+ result: {
287
+ tools: (0, tools_1.getToolDefinitions)(),
288
+ },
289
+ };
290
+ case 'tools/call':
291
+ if (!params?.name) {
292
+ return {
293
+ jsonrpc: '2.0',
294
+ id,
295
+ error: {
296
+ code: -32602,
297
+ message: 'Missing tool name',
298
+ },
299
+ };
300
+ }
301
+ if (!(0, tools_1.hasTool)(params.name)) {
302
+ return {
303
+ jsonrpc: '2.0',
304
+ id,
305
+ error: {
306
+ code: -32602,
307
+ message: `Unknown tool: ${params.name}`,
308
+ },
309
+ };
310
+ }
311
+ const result = await (0, tools_1.executeTool)(params.name, params.arguments || {}, this.services);
312
+ return {
313
+ jsonrpc: '2.0',
314
+ id,
315
+ result,
316
+ };
317
+ case 'ping':
318
+ return {
319
+ jsonrpc: '2.0',
320
+ id,
321
+ result: {},
322
+ };
323
+ default:
324
+ return {
325
+ jsonrpc: '2.0',
326
+ id,
327
+ error: {
328
+ code: -32601,
329
+ message: `Unknown method: ${method}`,
330
+ },
331
+ };
332
+ }
333
+ }
334
+ /**
335
+ * Send an error response
336
+ */
337
+ sendError(res, statusCode, message) {
338
+ res.writeHead(statusCode, { 'Content-Type': 'application/json' });
339
+ res.end(JSON.stringify({ error: message }));
340
+ }
341
+ }
342
+ exports.McpServer = McpServer;
343
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"McpServer.js","sourceRoot":"","sources":["../../../src/main/mcp/McpServer.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * MCP Server\n * Main server class that handles HTTP/SSE connections for MCP protocol\n */\n\nimport http from 'http';\nimport { URL } from 'url';\nimport { MCP_SERVER, MCP_ENDPOINTS } from '../../common/constants';\nimport {\n  McpConnectionInfo,\n  McpServerConfig,\n  McpServerStatus,\n  McpRequest,\n  McpResponse,\n  LocalServices,\n} from '../../common/types';\nimport { McpAuth } from './McpAuth';\nimport { ConnectionInfoManager } from '../config/ConnectionInfo';\nimport { registerTools, getToolDefinitions, getToolNames, executeTool, hasTool } from './tools';\n\nexport class McpServer {\n  private server: http.Server | null = null;\n  private port: number;\n  private auth: McpAuth;\n  private connectionInfo: ConnectionInfoManager;\n  private services: LocalServices;\n  private logger: any;\n  private startTime: number = 0;\n\n  constructor(config: McpServerConfig, services: LocalServices, logger: any) {\n    this.port = config.port || MCP_SERVER.DEFAULT_PORT;\n    this.services = services;\n    this.logger = logger;\n    this.auth = new McpAuth(logger);\n    this.connectionInfo = new ConnectionInfoManager(logger);\n\n    // Set token if provided\n    if (config.authToken) {\n      this.auth.setToken(config.authToken);\n    }\n\n    // Register all tools\n    registerTools();\n  }\n\n  /**\n   * Start the MCP server\n   */\n  async start(): Promise<void> {\n    if (this.server) {\n      this.logger.warn('[MCP] Server already running');\n      return;\n    }\n\n    // Try to load existing token from connection info (for persistence across restarts)\n    const existingInfo = await this.connectionInfo.load();\n    if (existingInfo?.authToken) {\n      this.auth.setToken(existingInfo.authToken);\n      this.logger.info(\n        `[MCP] Loaded existing auth token: ${existingInfo.authToken.substring(0, 20)}...`\n      );\n    }\n\n    // Find available port\n    this.port = await this.findAvailablePort(this.port);\n\n    return new Promise((resolve, reject) => {\n      this.server = http.createServer((req, res) => this.handleRequest(req, res));\n\n      this.server.on('error', (err: any) => {\n        if (err.code === 'EADDRINUSE') {\n          this.logger.error(`[MCP] Port ${this.port} is in use`);\n          reject(new Error(`Port ${this.port} is already in use`));\n        } else {\n          this.logger.error('[MCP] Server error:', err);\n          reject(err);\n        }\n      });\n\n      this.server.listen(this.port, '127.0.0.1', async () => {\n        this.startTime = Date.now();\n        this.logger.info(`[MCP] Server started on http://127.0.0.1:${this.port}`);\n\n        // Save connection info\n        const info = this.getConnectionInfo();\n        await this.connectionInfo.save(info);\n\n        resolve();\n      });\n    });\n  }\n\n  /**\n   * Stop the MCP server\n   */\n  async stop(): Promise<void> {\n    if (!this.server) {\n      return;\n    }\n\n    return new Promise((resolve) => {\n      this.server!.close(() => {\n        this.logger.info('[MCP] Server stopped');\n        this.server = null;\n        this.connectionInfo.delete();\n        resolve();\n      });\n    });\n  }\n\n  /**\n   * Check if server is running\n   */\n  isRunning(): boolean {\n    return this.server !== null;\n  }\n\n  /**\n   * Get server status\n   */\n  getStatus(): McpServerStatus {\n    return {\n      running: this.isRunning(),\n      port: this.port,\n      uptime: this.isRunning() ? (Date.now() - this.startTime) / 1000 : 0,\n    };\n  }\n\n  /**\n   * Get connection info for external tools\n   */\n  getConnectionInfo(): McpConnectionInfo {\n    return {\n      url: `http://127.0.0.1:${this.port}`,\n      authToken: this.auth.getToken(),\n      port: this.port,\n      version: MCP_SERVER.VERSION,\n      tools: getToolNames(),\n    };\n  }\n\n  /**\n   * Regenerate the authentication token\n   */\n  async regenerateToken(): Promise<string> {\n    const newToken = this.auth.regenerateToken();\n    // Update connection info file with new token\n    const info = this.getConnectionInfo();\n    await this.connectionInfo.save(info);\n    this.logger.info(`[MCP] Auth token regenerated`);\n    return newToken;\n  }\n\n  /**\n   * Find an available port starting from the preferred port\n   */\n  private async findAvailablePort(preferredPort: number): Promise<number> {\n    const isPortAvailable = (port: number): Promise<boolean> => {\n      return new Promise((resolve) => {\n        const testServer = http.createServer();\n        testServer.once('error', () => resolve(false));\n        testServer.once('listening', () => {\n          testServer.close(() => resolve(true));\n        });\n        testServer.listen(port, '127.0.0.1');\n      });\n    };\n\n    for (let port = preferredPort; port <= MCP_SERVER.PORT_RANGE.MAX; port++) {\n      if (await isPortAvailable(port)) {\n        if (port !== preferredPort) {\n          this.logger.info(`[MCP] Port ${preferredPort} unavailable, using ${port}`);\n        }\n        return port;\n      }\n    }\n\n    throw new Error(\n      `No available ports in range ${MCP_SERVER.PORT_RANGE.MIN}-${MCP_SERVER.PORT_RANGE.MAX}`\n    );\n  }\n\n  /**\n   * Handle incoming HTTP requests\n   */\n  private async handleRequest(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\n    const url = new URL(req.url || '/', `http://127.0.0.1:${this.port}`);\n    const pathname = url.pathname;\n\n    // IP check\n    if (!this.auth.validateIP(req.socket.remoteAddress)) {\n      this.sendError(res, 403, 'Forbidden: Only localhost connections allowed');\n      return;\n    }\n\n    // Health endpoint (no auth required)\n    if (pathname === MCP_ENDPOINTS.HEALTH) {\n      this.handleHealth(req, res);\n      return;\n    }\n\n    // Auth check for all other endpoints\n    const authHeader = req.headers.authorization;\n    if (!this.auth.validateToken(authHeader)) {\n      this.sendError(res, 401, 'Unauthorized: Invalid or missing authentication token');\n      return;\n    }\n\n    // Handle CORS preflight\n    if (req.method === 'OPTIONS') {\n      res.writeHead(200, {\n        'Access-Control-Allow-Origin': '*',\n        'Access-Control-Allow-Methods': 'GET, POST, OPTIONS',\n        'Access-Control-Allow-Headers': 'Content-Type, Authorization',\n        'Access-Control-Max-Age': '86400',\n      });\n      res.end();\n      return;\n    }\n\n    // Route to appropriate handler\n    if (pathname === MCP_ENDPOINTS.SSE && req.method === 'GET') {\n      this.handleSSE(req, res);\n    } else if (pathname === MCP_ENDPOINTS.MESSAGES && req.method === 'POST') {\n      await this.handleMessage(req, res);\n    } else {\n      this.sendError(res, 404, 'Not Found');\n    }\n  }\n\n  /**\n   * Handle health check requests\n   */\n  private handleHealth(req: http.IncomingMessage, res: http.ServerResponse): void {\n    const status = this.getStatus();\n    const token = this.auth.getToken();\n    res.writeHead(200, { 'Content-Type': 'application/json' });\n    res.end(\n      JSON.stringify({\n        status: 'ok',\n        version: MCP_SERVER.VERSION,\n        uptime: status.uptime,\n        tools: getToolNames(),\n        tokenPrefix: token.substring(0, 20),\n      })\n    );\n  }\n\n  /**\n   * Handle SSE connection for MCP protocol\n   */\n  private handleSSE(req: http.IncomingMessage, res: http.ServerResponse): void {\n    res.writeHead(200, {\n      'Content-Type': 'text/event-stream',\n      'Cache-Control': 'no-cache',\n      Connection: 'keep-alive',\n      'Access-Control-Allow-Origin': '*',\n    });\n\n    // Send endpoint information\n    const messagesUrl = `http://127.0.0.1:${this.port}${MCP_ENDPOINTS.MESSAGES}`;\n    res.write(`event: endpoint\\ndata: ${messagesUrl}\\n\\n`);\n\n    // Keep connection alive\n    const keepAlive = setInterval(() => {\n      res.write(': keepalive\\n\\n');\n    }, 30000);\n\n    req.on('close', () => {\n      clearInterval(keepAlive);\n    });\n  }\n\n  /**\n   * Handle MCP message requests\n   */\n  private async handleMessage(req: http.IncomingMessage, res: http.ServerResponse): Promise<void> {\n    let body = '';\n\n    req.on('data', (chunk) => {\n      body += chunk;\n    });\n\n    req.on('end', async () => {\n      try {\n        const request: McpRequest = JSON.parse(body);\n\n        // Notifications don't have an id and don't expect a response\n        if (request.method?.startsWith('notifications/')) {\n          this.logger.info(`[MCP] Received notification: ${request.method}`);\n          res.writeHead(200, { 'Content-Type': 'application/json' });\n          res.end('{}');\n          return;\n        }\n\n        const response = await this.processMessage(request);\n\n        res.writeHead(200, {\n          'Content-Type': 'application/json',\n          'Access-Control-Allow-Origin': '*',\n        });\n        res.end(JSON.stringify(response));\n      } catch (error: any) {\n        this.logger.error('[MCP] Message handling error:', error);\n        this.sendError(res, 400, `Invalid request: ${error.message}`);\n      }\n    });\n  }\n\n  /**\n   * Process an MCP message and return response\n   */\n  private async processMessage(request: McpRequest): Promise<McpResponse> {\n    const { id, method, params } = request;\n\n    switch (method) {\n      case 'initialize':\n        return {\n          jsonrpc: '2.0',\n          id,\n          result: {\n            protocolVersion: '2024-11-05',\n            capabilities: {\n              tools: {},\n            },\n            serverInfo: {\n              name: MCP_SERVER.NAME,\n              version: MCP_SERVER.VERSION,\n            },\n          },\n        };\n\n      case 'tools/list':\n        return {\n          jsonrpc: '2.0',\n          id,\n          result: {\n            tools: getToolDefinitions(),\n          },\n        };\n\n      case 'tools/call':\n        if (!params?.name) {\n          return {\n            jsonrpc: '2.0',\n            id,\n            error: {\n              code: -32602,\n              message: 'Missing tool name',\n            },\n          };\n        }\n\n        if (!hasTool(params.name)) {\n          return {\n            jsonrpc: '2.0',\n            id,\n            error: {\n              code: -32602,\n              message: `Unknown tool: ${params.name}`,\n            },\n          };\n        }\n\n        const result = await executeTool(params.name, params.arguments || {}, this.services);\n\n        return {\n          jsonrpc: '2.0',\n          id,\n          result,\n        };\n\n      case 'ping':\n        return {\n          jsonrpc: '2.0',\n          id,\n          result: {},\n        };\n\n      default:\n        return {\n          jsonrpc: '2.0',\n          id,\n          error: {\n            code: -32601,\n            message: `Unknown method: ${method}`,\n          },\n        };\n    }\n  }\n\n  /**\n   * Send an error response\n   */\n  private sendError(res: http.ServerResponse, statusCode: number, message: string): void {\n    res.writeHead(statusCode, { 'Content-Type': 'application/json' });\n    res.end(JSON.stringify({ error: message }));\n  }\n}\n"]}
@@ -0,0 +1,7 @@
1
+ /**
2
+ * change_php_version Tool
3
+ * Change PHP version for a site
4
+ */
5
+ import { McpToolDefinition, McpToolResult, LocalServices } from '../../../common/types';
6
+ export declare const changePhpVersionDefinition: McpToolDefinition;
7
+ export declare function changePhpVersion(args: Record<string, unknown>, services: LocalServices): Promise<McpToolResult>;