@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,7 @@
1
+ /**
2
+ * get_local_info Tool
3
+ * Get information about the Local application
4
+ */
5
+ import { McpToolDefinition, McpToolResult, LocalServices } from '../../../common/types';
6
+ export declare const getLocalInfoDefinition: McpToolDefinition;
7
+ export declare function getLocalInfo(_args: Record<string, unknown>, services: LocalServices): Promise<McpToolResult>;
@@ -0,0 +1,72 @@
1
+ "use strict";
2
+ /**
3
+ * get_local_info Tool
4
+ * Get information about the Local application
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.getLocalInfoDefinition = void 0;
11
+ exports.getLocalInfo = getLocalInfo;
12
+ const os_1 = __importDefault(require("os"));
13
+ const constants_1 = require("../../../common/constants");
14
+ const index_1 = require("./index");
15
+ exports.getLocalInfoDefinition = {
16
+ name: 'get_local_info',
17
+ description: 'Get information about the Local application including version, platform, MCP server status, and available tools',
18
+ inputSchema: {
19
+ type: 'object',
20
+ properties: {},
21
+ },
22
+ };
23
+ async function getLocalInfo(_args, services) {
24
+ try {
25
+ const sitesMap = services.siteData.getSites();
26
+ const allSites = Object.values(sitesMap);
27
+ // Count sites by status
28
+ let runningSites = 0;
29
+ let stoppedSites = 0;
30
+ for (const site of allSites) {
31
+ const status = await services.siteProcessManager.getSiteStatus(site);
32
+ if (status === 'running') {
33
+ runningSites++;
34
+ }
35
+ else {
36
+ stoppedSites++;
37
+ }
38
+ }
39
+ const info = {
40
+ local: {
41
+ version: process.env.LOCAL_VERSION || 'unknown',
42
+ platform: os_1.default.platform(),
43
+ arch: os_1.default.arch(),
44
+ },
45
+ mcp: {
46
+ version: constants_1.MCP_SERVER.VERSION,
47
+ name: constants_1.MCP_SERVER.NAME,
48
+ tools: (0, index_1.getToolNames)(),
49
+ },
50
+ sites: {
51
+ total: allSites.length,
52
+ running: runningSites,
53
+ stopped: stoppedSites,
54
+ },
55
+ system: {
56
+ nodeVersion: process.version,
57
+ hostname: os_1.default.hostname(),
58
+ homeDir: os_1.default.homedir(),
59
+ },
60
+ };
61
+ return {
62
+ content: [{ type: 'text', text: JSON.stringify(info, null, 2) }],
63
+ };
64
+ }
65
+ catch (error) {
66
+ return {
67
+ content: [{ type: 'text', text: `Failed to get Local info: ${error.message}` }],
68
+ isError: true,
69
+ };
70
+ }
71
+ }
72
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2V0TG9jYWxJbmZvLmpzIiwic291cmNlUm9vdCI6IiIsInNvdXJjZXMiOlsiLi4vLi4vLi4vLi4vc3JjL21haW4vbWNwL3Rvb2xzL2dldExvY2FsSW5mby50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiO0FBQUE7OztHQUdHOzs7Ozs7QUFpQkgsb0NBb0RDO0FBbkVELDRDQUFvQjtBQUVwQix5REFBdUQ7QUFDdkQsbUNBQXVDO0FBRTFCLFFBQUEsc0JBQXNCLEdBQXNCO0lBQ3ZELElBQUksRUFBRSxnQkFBZ0I7SUFDdEIsV0FBVyxFQUNULGlIQUFpSDtJQUNuSCxXQUFXLEVBQUU7UUFDWCxJQUFJLEVBQUUsUUFBUTtRQUNkLFVBQVUsRUFBRSxFQUFFO0tBQ2Y7Q0FDRixDQUFDO0FBRUssS0FBSyxVQUFVLFlBQVksQ0FDaEMsS0FBOEIsRUFDOUIsUUFBdUI7SUFFdkIsSUFBSSxDQUFDO1FBQ0gsTUFBTSxRQUFRLEdBQUcsUUFBUSxDQUFDLFFBQVEsQ0FBQyxRQUFRLEVBQUUsQ0FBQztRQUM5QyxNQUFNLFFBQVEsR0FBRyxNQUFNLENBQUMsTUFBTSxDQUFDLFFBQVEsQ0FBVSxDQUFDO1FBRWxELHdCQUF3QjtRQUN4QixJQUFJLFlBQVksR0FBRyxDQUFDLENBQUM7UUFDckIsSUFBSSxZQUFZLEdBQUcsQ0FBQyxDQUFDO1FBQ3JCLEtBQUssTUFBTSxJQUFJLElBQUksUUFBUSxFQUFFLENBQUM7WUFDNUIsTUFBTSxNQUFNLEdBQUcsTUFBTSxRQUFRLENBQUMsa0JBQWtCLENBQUMsYUFBYSxDQUFDLElBQUksQ0FBQyxDQUFDO1lBQ3JFLElBQUksTUFBTSxLQUFLLFNBQVMsRUFBRSxDQUFDO2dCQUN6QixZQUFZLEVBQUUsQ0FBQztZQUNqQixDQUFDO2lCQUFNLENBQUM7Z0JBQ04sWUFBWSxFQUFFLENBQUM7WUFDakIsQ0FBQztRQUNILENBQUM7UUFFRCxNQUFNLElBQUksR0FBRztZQUNYLEtBQUssRUFBRTtnQkFDTCxPQUFPLEVBQUUsT0FBTyxDQUFDLEdBQUcsQ0FBQyxhQUFhLElBQUksU0FBUztnQkFDL0MsUUFBUSxFQUFFLFlBQUUsQ0FBQyxRQUFRLEVBQUU7Z0JBQ3ZCLElBQUksRUFBRSxZQUFFLENBQUMsSUFBSSxFQUFFO2FBQ2hCO1lBQ0QsR0FBRyxFQUFFO2dCQUNILE9BQU8sRUFBRSxzQkFBVSxDQUFDLE9BQU87Z0JBQzNCLElBQUksRUFBRSxzQkFBVSxDQUFDLElBQUk7Z0JBQ3JCLEtBQUssRUFBRSxJQUFBLG9CQUFZLEdBQUU7YUFDdEI7WUFDRCxLQUFLLEVBQUU7Z0JBQ0wsS0FBSyxFQUFFLFFBQVEsQ0FBQyxNQUFNO2dCQUN0QixPQUFPLEVBQUUsWUFBWTtnQkFDckIsT0FBTyxFQUFFLFlBQVk7YUFDdEI7WUFDRCxNQUFNLEVBQUU7Z0JBQ04sV0FBVyxFQUFFLE9BQU8sQ0FBQyxPQUFPO2dCQUM1QixRQUFRLEVBQUUsWUFBRSxDQUFDLFFBQVEsRUFBRTtnQkFDdkIsT0FBTyxFQUFFLFlBQUUsQ0FBQyxPQUFPLEVBQUU7YUFDdEI7U0FDRixDQUFDO1FBRUYsT0FBTztZQUNMLE9BQU8sRUFBRSxDQUFDLEVBQUUsSUFBSSxFQUFFLE1BQU0sRUFBRSxJQUFJLEVBQUUsSUFBSSxDQUFDLFNBQVMsQ0FBQyxJQUFJLEVBQUUsSUFBSSxFQUFFLENBQUMsQ0FBQyxFQUFFLENBQUM7U0FDakUsQ0FBQztJQUNKLENBQUM7SUFBQyxPQUFPLEtBQVUsRUFBRSxDQUFDO1FBQ3BCLE9BQU87WUFDTCxPQUFPLEVBQUUsQ0FBQyxFQUFFLElBQUksRUFBRSxNQUFNLEVBQUUsSUFBSSxFQUFFLDZCQUE2QixLQUFLLENBQUMsT0FBTyxFQUFFLEVBQUUsQ0FBQztZQUMvRSxPQUFPLEVBQUUsSUFBSTtTQUNkLENBQUM7SUFDSixDQUFDO0FBQ0gsQ0FBQyIsInNvdXJjZXNDb250ZW50IjpbIi8qKlxuICogZ2V0X2xvY2FsX2luZm8gVG9vbFxuICogR2V0IGluZm9ybWF0aW9uIGFib3V0IHRoZSBMb2NhbCBhcHBsaWNhdGlvblxuICovXG5cbmltcG9ydCBvcyBmcm9tICdvcyc7XG5pbXBvcnQgeyBNY3BUb29sRGVmaW5pdGlvbiwgTWNwVG9vbFJlc3VsdCwgTG9jYWxTZXJ2aWNlcyB9IGZyb20gJy4uLy4uLy4uL2NvbW1vbi90eXBlcyc7XG5pbXBvcnQgeyBNQ1BfU0VSVkVSIH0gZnJvbSAnLi4vLi4vLi4vY29tbW9uL2NvbnN0YW50cyc7XG5pbXBvcnQgeyBnZXRUb29sTmFtZXMgfSBmcm9tICcuL2luZGV4JztcblxuZXhwb3J0IGNvbnN0IGdldExvY2FsSW5mb0RlZmluaXRpb246IE1jcFRvb2xEZWZpbml0aW9uID0ge1xuICBuYW1lOiAnZ2V0X2xvY2FsX2luZm8nLFxuICBkZXNjcmlwdGlvbjpcbiAgICAnR2V0IGluZm9ybWF0aW9uIGFib3V0IHRoZSBMb2NhbCBhcHBsaWNhdGlvbiBpbmNsdWRpbmcgdmVyc2lvbiwgcGxhdGZvcm0sIE1DUCBzZXJ2ZXIgc3RhdHVzLCBhbmQgYXZhaWxhYmxlIHRvb2xzJyxcbiAgaW5wdXRTY2hlbWE6IHtcbiAgICB0eXBlOiAnb2JqZWN0JyxcbiAgICBwcm9wZXJ0aWVzOiB7fSxcbiAgfSxcbn07XG5cbmV4cG9ydCBhc3luYyBmdW5jdGlvbiBnZXRMb2NhbEluZm8oXG4gIF9hcmdzOiBSZWNvcmQ8c3RyaW5nLCB1bmtub3duPixcbiAgc2VydmljZXM6IExvY2FsU2VydmljZXNcbik6IFByb21pc2U8TWNwVG9vbFJlc3VsdD4ge1xuICB0cnkge1xuICAgIGNvbnN0IHNpdGVzTWFwID0gc2VydmljZXMuc2l0ZURhdGEuZ2V0U2l0ZXMoKTtcbiAgICBjb25zdCBhbGxTaXRlcyA9IE9iamVjdC52YWx1ZXMoc2l0ZXNNYXApIGFzIGFueVtdO1xuXG4gICAgLy8gQ291bnQgc2l0ZXMgYnkgc3RhdHVzXG4gICAgbGV0IHJ1bm5pbmdTaXRlcyA9IDA7XG4gICAgbGV0IHN0b3BwZWRTaXRlcyA9IDA7XG4gICAgZm9yIChjb25zdCBzaXRlIG9mIGFsbFNpdGVzKSB7XG4gICAgICBjb25zdCBzdGF0dXMgPSBhd2FpdCBzZXJ2aWNlcy5zaXRlUHJvY2Vzc01hbmFnZXIuZ2V0U2l0ZVN0YXR1cyhzaXRlKTtcbiAgICAgIGlmIChzdGF0dXMgPT09ICdydW5uaW5nJykge1xuICAgICAgICBydW5uaW5nU2l0ZXMrKztcbiAgICAgIH0gZWxzZSB7XG4gICAgICAgIHN0b3BwZWRTaXRlcysrO1xuICAgICAgfVxuICAgIH1cblxuICAgIGNvbnN0IGluZm8gPSB7XG4gICAgICBsb2NhbDoge1xuICAgICAgICB2ZXJzaW9uOiBwcm9jZXNzLmVudi5MT0NBTF9WRVJTSU9OIHx8ICd1bmtub3duJyxcbiAgICAgICAgcGxhdGZvcm06IG9zLnBsYXRmb3JtKCksXG4gICAgICAgIGFyY2g6IG9zLmFyY2goKSxcbiAgICAgIH0sXG4gICAgICBtY3A6IHtcbiAgICAgICAgdmVyc2lvbjogTUNQX1NFUlZFUi5WRVJTSU9OLFxuICAgICAgICBuYW1lOiBNQ1BfU0VSVkVSLk5BTUUsXG4gICAgICAgIHRvb2xzOiBnZXRUb29sTmFtZXMoKSxcbiAgICAgIH0sXG4gICAgICBzaXRlczoge1xuICAgICAgICB0b3RhbDogYWxsU2l0ZXMubGVuZ3RoLFxuICAgICAgICBydW5uaW5nOiBydW5uaW5nU2l0ZXMsXG4gICAgICAgIHN0b3BwZWQ6IHN0b3BwZWRTaXRlcyxcbiAgICAgIH0sXG4gICAgICBzeXN0ZW06IHtcbiAgICAgICAgbm9kZVZlcnNpb246IHByb2Nlc3MudmVyc2lvbixcbiAgICAgICAgaG9zdG5hbWU6IG9zLmhvc3RuYW1lKCksXG4gICAgICAgIGhvbWVEaXI6IG9zLmhvbWVkaXIoKSxcbiAgICAgIH0sXG4gICAgfTtcblxuICAgIHJldHVybiB7XG4gICAgICBjb250ZW50OiBbeyB0eXBlOiAndGV4dCcsIHRleHQ6IEpTT04uc3RyaW5naWZ5KGluZm8sIG51bGwsIDIpIH1dLFxuICAgIH07XG4gIH0gY2F0Y2ggKGVycm9yOiBhbnkpIHtcbiAgICByZXR1cm4ge1xuICAgICAgY29udGVudDogW3sgdHlwZTogJ3RleHQnLCB0ZXh0OiBgRmFpbGVkIHRvIGdldCBMb2NhbCBpbmZvOiAke2Vycm9yLm1lc3NhZ2V9YCB9XSxcbiAgICAgIGlzRXJyb3I6IHRydWUsXG4gICAgfTtcbiAgfVxufVxuIl19
@@ -0,0 +1,7 @@
1
+ /**
2
+ * get_site Tool
3
+ * Get detailed information about a specific site
4
+ */
5
+ import { McpToolDefinition, McpToolResult, LocalServices } from '../../../common/types';
6
+ export declare const getSiteDefinition: McpToolDefinition;
7
+ export declare function getSite(args: Record<string, unknown>, services: LocalServices): Promise<McpToolResult>;
@@ -0,0 +1,68 @@
1
+ "use strict";
2
+ /**
3
+ * get_site Tool
4
+ * Get detailed information about a specific site
5
+ */
6
+ Object.defineProperty(exports, "__esModule", { value: true });
7
+ exports.getSiteDefinition = void 0;
8
+ exports.getSite = getSite;
9
+ const helpers_1 = require("./helpers");
10
+ exports.getSiteDefinition = {
11
+ name: 'get_site',
12
+ description: 'Get detailed information about a WordPress site including PHP version, web server, database, and WordPress version',
13
+ inputSchema: {
14
+ type: 'object',
15
+ properties: {
16
+ site: {
17
+ type: 'string',
18
+ description: 'Site name or ID (partial names work)',
19
+ },
20
+ },
21
+ required: ['site'],
22
+ },
23
+ };
24
+ async function getSite(args, services) {
25
+ const { site: siteQuery } = args;
26
+ // Validate required parameter
27
+ const paramError = (0, helpers_1.validateRequiredParam)(siteQuery, 'site');
28
+ if (paramError)
29
+ return paramError;
30
+ // Find site or return error
31
+ const siteResult = (0, helpers_1.findSiteOrError)(siteQuery, services.siteData);
32
+ if ('error' in siteResult)
33
+ return siteResult.error;
34
+ const { site } = siteResult;
35
+ try {
36
+ const currentStatus = await services.siteProcessManager.getSiteStatus(site);
37
+ const siteInfo = {
38
+ id: site.id,
39
+ name: site.name,
40
+ status: currentStatus,
41
+ domain: site.domain,
42
+ path: site.path,
43
+ url: `https://${site.domain}`,
44
+ adminUrl: `https://${site.domain}/wp-admin`,
45
+ environment: site.environment,
46
+ services: {
47
+ php: site.phpVersion || site.services?.php?.version,
48
+ webServer: site.webServer || site.services?.nginx?.version || site.services?.apache?.version,
49
+ database: site.mysql || site.services?.mysql?.version,
50
+ },
51
+ wordpress: {
52
+ version: site.wordPressVersion,
53
+ multisite: site.multisite || false,
54
+ },
55
+ hostConnections: site.hostConnections || [],
56
+ };
57
+ return {
58
+ content: [{ type: 'text', text: JSON.stringify(siteInfo, null, 2) }],
59
+ };
60
+ }
61
+ catch (error) {
62
+ return {
63
+ content: [{ type: 'text', text: `Failed to get site info: ${error.message}` }],
64
+ isError: true,
65
+ };
66
+ }
67
+ }
68
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiZ2V0U2l0ZS5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uLy4uLy4uL3NyYy9tYWluL21jcC90b29scy9nZXRTaXRlLnRzIl0sIm5hbWVzIjpbXSwibWFwcGluZ3MiOiI7QUFBQTs7O0dBR0c7OztBQXlCSCwwQkFpREM7QUF2RUQsdUNBQW1FO0FBRXRELFFBQUEsaUJBQWlCLEdBQXNCO0lBQ2xELElBQUksRUFBRSxVQUFVO0lBQ2hCLFdBQVcsRUFDVCxvSEFBb0g7SUFDdEgsV0FBVyxFQUFFO1FBQ1gsSUFBSSxFQUFFLFFBQVE7UUFDZCxVQUFVLEVBQUU7WUFDVixJQUFJLEVBQUU7Z0JBQ0osSUFBSSxFQUFFLFFBQVE7Z0JBQ2QsV0FBVyxFQUFFLHNDQUFzQzthQUNwRDtTQUNGO1FBQ0QsUUFBUSxFQUFFLENBQUMsTUFBTSxDQUFDO0tBQ25CO0NBQ0YsQ0FBQztBQU1LLEtBQUssVUFBVSxPQUFPLENBQzNCLElBQTZCLEVBQzdCLFFBQXVCO0lBRXZCLE1BQU0sRUFBRSxJQUFJLEVBQUUsU0FBUyxFQUFFLEdBQUcsSUFBOEIsQ0FBQztJQUUzRCw4QkFBOEI7SUFDOUIsTUFBTSxVQUFVLEdBQUcsSUFBQSwrQkFBcUIsRUFBQyxTQUFTLEVBQUUsTUFBTSxDQUFDLENBQUM7SUFDNUQsSUFBSSxVQUFVO1FBQUUsT0FBTyxVQUFVLENBQUM7SUFFbEMsNEJBQTRCO0lBQzVCLE1BQU0sVUFBVSxHQUFHLElBQUEseUJBQWUsRUFBQyxTQUFTLEVBQUUsUUFBUSxDQUFDLFFBQVEsQ0FBQyxDQUFDO0lBQ2pFLElBQUksT0FBTyxJQUFJLFVBQVU7UUFBRSxPQUFPLFVBQVUsQ0FBQyxLQUFLLENBQUM7SUFDbkQsTUFBTSxFQUFFLElBQUksRUFBRSxHQUFHLFVBQVUsQ0FBQztJQUU1QixJQUFJLENBQUM7UUFDSCxNQUFNLGFBQWEsR0FBRyxNQUFNLFFBQVEsQ0FBQyxrQkFBa0IsQ0FBQyxhQUFhLENBQUMsSUFBSSxDQUFDLENBQUM7UUFFNUUsTUFBTSxRQUFRLEdBQUc7WUFDZixFQUFFLEVBQUUsSUFBSSxDQUFDLEVBQUU7WUFDWCxJQUFJLEVBQUUsSUFBSSxDQUFDLElBQUk7WUFDZixNQUFNLEVBQUUsYUFBYTtZQUNyQixNQUFNLEVBQUUsSUFBSSxDQUFDLE1BQU07WUFDbkIsSUFBSSxFQUFFLElBQUksQ0FBQyxJQUFJO1lBQ2YsR0FBRyxFQUFFLFdBQVcsSUFBSSxDQUFDLE1BQU0sRUFBRTtZQUM3QixRQUFRLEVBQUUsV0FBVyxJQUFJLENBQUMsTUFBTSxXQUFXO1lBQzNDLFdBQVcsRUFBRSxJQUFJLENBQUMsV0FBVztZQUM3QixRQUFRLEVBQUU7Z0JBQ1IsR0FBRyxFQUFFLElBQUksQ0FBQyxVQUFVLElBQUksSUFBSSxDQUFDLFFBQVEsRUFBRSxHQUFHLEVBQUUsT0FBTztnQkFDbkQsU0FBUyxFQUNQLElBQUksQ0FBQyxTQUFTLElBQUksSUFBSSxDQUFDLFFBQVEsRUFBRSxLQUFLLEVBQUUsT0FBTyxJQUFJLElBQUksQ0FBQyxRQUFRLEVBQUUsTUFBTSxFQUFFLE9BQU87Z0JBQ25GLFFBQVEsRUFBRSxJQUFJLENBQUMsS0FBSyxJQUFJLElBQUksQ0FBQyxRQUFRLEVBQUUsS0FBSyxFQUFFLE9BQU87YUFDdEQ7WUFDRCxTQUFTLEVBQUU7Z0JBQ1QsT0FBTyxFQUFFLElBQUksQ0FBQyxnQkFBZ0I7Z0JBQzlCLFNBQVMsRUFBRSxJQUFJLENBQUMsU0FBUyxJQUFJLEtBQUs7YUFDbkM7WUFDRCxlQUFlLEVBQUUsSUFBSSxDQUFDLGVBQWUsSUFBSSxFQUFFO1NBQzVDLENBQUM7UUFFRixPQUFPO1lBQ0wsT0FBTyxFQUFFLENBQUMsRUFBRSxJQUFJLEVBQUUsTUFBTSxFQUFFLElBQUksRUFBRSxJQUFJLENBQUMsU0FBUyxDQUFDLFFBQVEsRUFBRSxJQUFJLEVBQUUsQ0FBQyxDQUFDLEVBQUUsQ0FBQztTQUNyRSxDQUFDO0lBQ0osQ0FBQztJQUFDLE9BQU8sS0FBVSxFQUFFLENBQUM7UUFDcEIsT0FBTztZQUNMLE9BQU8sRUFBRSxDQUFDLEVBQUUsSUFBSSxFQUFFLE1BQU0sRUFBRSxJQUFJLEVBQUUsNEJBQTRCLEtBQUssQ0FBQyxPQUFPLEVBQUUsRUFBRSxDQUFDO1lBQzlFLE9BQU8sRUFBRSxJQUFJO1NBQ2QsQ0FBQztJQUNKLENBQUM7QUFDSCxDQUFDIiwic291cmNlc0NvbnRlbnQiOlsiLyoqXG4gKiBnZXRfc2l0ZSBUb29sXG4gKiBHZXQgZGV0YWlsZWQgaW5mb3JtYXRpb24gYWJvdXQgYSBzcGVjaWZpYyBzaXRlXG4gKi9cblxuaW1wb3J0IHsgTWNwVG9vbERlZmluaXRpb24sIE1jcFRvb2xSZXN1bHQsIExvY2FsU2VydmljZXMgfSBmcm9tICcuLi8uLi8uLi9jb21tb24vdHlwZXMnO1xuaW1wb3J0IHsgdmFsaWRhdGVSZXF1aXJlZFBhcmFtLCBmaW5kU2l0ZU9yRXJyb3IgfSBmcm9tICcuL2hlbHBlcnMnO1xuXG5leHBvcnQgY29uc3QgZ2V0U2l0ZURlZmluaXRpb246IE1jcFRvb2xEZWZpbml0aW9uID0ge1xuICBuYW1lOiAnZ2V0X3NpdGUnLFxuICBkZXNjcmlwdGlvbjpcbiAgICAnR2V0IGRldGFpbGVkIGluZm9ybWF0aW9uIGFib3V0IGEgV29yZFByZXNzIHNpdGUgaW5jbHVkaW5nIFBIUCB2ZXJzaW9uLCB3ZWIgc2VydmVyLCBkYXRhYmFzZSwgYW5kIFdvcmRQcmVzcyB2ZXJzaW9uJyxcbiAgaW5wdXRTY2hlbWE6IHtcbiAgICB0eXBlOiAnb2JqZWN0JyxcbiAgICBwcm9wZXJ0aWVzOiB7XG4gICAgICBzaXRlOiB7XG4gICAgICAgIHR5cGU6ICdzdHJpbmcnLFxuICAgICAgICBkZXNjcmlwdGlvbjogJ1NpdGUgbmFtZSBvciBJRCAocGFydGlhbCBuYW1lcyB3b3JrKScsXG4gICAgICB9LFxuICAgIH0sXG4gICAgcmVxdWlyZWQ6IFsnc2l0ZSddLFxuICB9LFxufTtcblxuaW50ZXJmYWNlIEdldFNpdGVBcmdzIHtcbiAgc2l0ZTogc3RyaW5nO1xufVxuXG5leHBvcnQgYXN5bmMgZnVuY3Rpb24gZ2V0U2l0ZShcbiAgYXJnczogUmVjb3JkPHN0cmluZywgdW5rbm93bj4sXG4gIHNlcnZpY2VzOiBMb2NhbFNlcnZpY2VzXG4pOiBQcm9taXNlPE1jcFRvb2xSZXN1bHQ+IHtcbiAgY29uc3QgeyBzaXRlOiBzaXRlUXVlcnkgfSA9IGFyZ3MgYXMgdW5rbm93biBhcyBHZXRTaXRlQXJncztcblxuICAvLyBWYWxpZGF0ZSByZXF1aXJlZCBwYXJhbWV0ZXJcbiAgY29uc3QgcGFyYW1FcnJvciA9IHZhbGlkYXRlUmVxdWlyZWRQYXJhbShzaXRlUXVlcnksICdzaXRlJyk7XG4gIGlmIChwYXJhbUVycm9yKSByZXR1cm4gcGFyYW1FcnJvcjtcblxuICAvLyBGaW5kIHNpdGUgb3IgcmV0dXJuIGVycm9yXG4gIGNvbnN0IHNpdGVSZXN1bHQgPSBmaW5kU2l0ZU9yRXJyb3Ioc2l0ZVF1ZXJ5LCBzZXJ2aWNlcy5zaXRlRGF0YSk7XG4gIGlmICgnZXJyb3InIGluIHNpdGVSZXN1bHQpIHJldHVybiBzaXRlUmVzdWx0LmVycm9yO1xuICBjb25zdCB7IHNpdGUgfSA9IHNpdGVSZXN1bHQ7XG5cbiAgdHJ5IHtcbiAgICBjb25zdCBjdXJyZW50U3RhdHVzID0gYXdhaXQgc2VydmljZXMuc2l0ZVByb2Nlc3NNYW5hZ2VyLmdldFNpdGVTdGF0dXMoc2l0ZSk7XG5cbiAgICBjb25zdCBzaXRlSW5mbyA9IHtcbiAgICAgIGlkOiBzaXRlLmlkLFxuICAgICAgbmFtZTogc2l0ZS5uYW1lLFxuICAgICAgc3RhdHVzOiBjdXJyZW50U3RhdHVzLFxuICAgICAgZG9tYWluOiBzaXRlLmRvbWFpbixcbiAgICAgIHBhdGg6IHNpdGUucGF0aCxcbiAgICAgIHVybDogYGh0dHBzOi8vJHtzaXRlLmRvbWFpbn1gLFxuICAgICAgYWRtaW5Vcmw6IGBodHRwczovLyR7c2l0ZS5kb21haW59L3dwLWFkbWluYCxcbiAgICAgIGVudmlyb25tZW50OiBzaXRlLmVudmlyb25tZW50LFxuICAgICAgc2VydmljZXM6IHtcbiAgICAgICAgcGhwOiBzaXRlLnBocFZlcnNpb24gfHwgc2l0ZS5zZXJ2aWNlcz8ucGhwPy52ZXJzaW9uLFxuICAgICAgICB3ZWJTZXJ2ZXI6XG4gICAgICAgICAgc2l0ZS53ZWJTZXJ2ZXIgfHwgc2l0ZS5zZXJ2aWNlcz8ubmdpbng/LnZlcnNpb24gfHwgc2l0ZS5zZXJ2aWNlcz8uYXBhY2hlPy52ZXJzaW9uLFxuICAgICAgICBkYXRhYmFzZTogc2l0ZS5teXNxbCB8fCBzaXRlLnNlcnZpY2VzPy5teXNxbD8udmVyc2lvbixcbiAgICAgIH0sXG4gICAgICB3b3JkcHJlc3M6IHtcbiAgICAgICAgdmVyc2lvbjogc2l0ZS53b3JkUHJlc3NWZXJzaW9uLFxuICAgICAgICBtdWx0aXNpdGU6IHNpdGUubXVsdGlzaXRlIHx8IGZhbHNlLFxuICAgICAgfSxcbiAgICAgIGhvc3RDb25uZWN0aW9uczogc2l0ZS5ob3N0Q29ubmVjdGlvbnMgfHwgW10sXG4gICAgfTtcblxuICAgIHJldHVybiB7XG4gICAgICBjb250ZW50OiBbeyB0eXBlOiAndGV4dCcsIHRleHQ6IEpTT04uc3RyaW5naWZ5KHNpdGVJbmZvLCBudWxsLCAyKSB9XSxcbiAgICB9O1xuICB9IGNhdGNoIChlcnJvcjogYW55KSB7XG4gICAgcmV0dXJuIHtcbiAgICAgIGNvbnRlbnQ6IFt7IHR5cGU6ICd0ZXh0JywgdGV4dDogYEZhaWxlZCB0byBnZXQgc2l0ZSBpbmZvOiAke2Vycm9yLm1lc3NhZ2V9YCB9XSxcbiAgICAgIGlzRXJyb3I6IHRydWUsXG4gICAgfTtcbiAgfVxufVxuIl19
@@ -0,0 +1,7 @@
1
+ /**
2
+ * get_site_logs Tool
3
+ * Retrieve log files for a site
4
+ */
5
+ import { McpToolDefinition, McpToolResult, LocalServices } from '../../../common/types';
6
+ export declare const getSiteLogsDefinition: McpToolDefinition;
7
+ export declare function getSiteLogs(args: Record<string, unknown>, services: LocalServices): Promise<McpToolResult>;
@@ -0,0 +1,149 @@
1
+ "use strict";
2
+ /**
3
+ * get_site_logs Tool
4
+ * Retrieve log files for a site
5
+ */
6
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
7
+ if (k2 === undefined) k2 = k;
8
+ var desc = Object.getOwnPropertyDescriptor(m, k);
9
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
10
+ desc = { enumerable: true, get: function() { return m[k]; } };
11
+ }
12
+ Object.defineProperty(o, k2, desc);
13
+ }) : (function(o, m, k, k2) {
14
+ if (k2 === undefined) k2 = k;
15
+ o[k2] = m[k];
16
+ }));
17
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
18
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
19
+ }) : function(o, v) {
20
+ o["default"] = v;
21
+ });
22
+ var __importStar = (this && this.__importStar) || (function () {
23
+ var ownKeys = function(o) {
24
+ ownKeys = Object.getOwnPropertyNames || function (o) {
25
+ var ar = [];
26
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
27
+ return ar;
28
+ };
29
+ return ownKeys(o);
30
+ };
31
+ return function (mod) {
32
+ if (mod && mod.__esModule) return mod;
33
+ var result = {};
34
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
35
+ __setModuleDefault(result, mod);
36
+ return result;
37
+ };
38
+ })();
39
+ Object.defineProperty(exports, "__esModule", { value: true });
40
+ exports.getSiteLogsDefinition = void 0;
41
+ exports.getSiteLogs = getSiteLogs;
42
+ const helpers_1 = require("./helpers");
43
+ const fs = __importStar(require("fs"));
44
+ const path = __importStar(require("path"));
45
+ exports.getSiteLogsDefinition = {
46
+ name: 'get_site_logs',
47
+ description: 'Get log file contents for a site (PHP errors, access logs, etc.)',
48
+ inputSchema: {
49
+ type: 'object',
50
+ properties: {
51
+ site: {
52
+ type: 'string',
53
+ description: 'Site name or ID',
54
+ },
55
+ logType: {
56
+ type: 'string',
57
+ enum: ['php', 'nginx', 'mysql', 'all'],
58
+ description: 'Type of logs to retrieve (default: php)',
59
+ },
60
+ lines: {
61
+ type: 'number',
62
+ description: 'Number of lines to return from end of log (default: 100)',
63
+ },
64
+ },
65
+ required: ['site'],
66
+ },
67
+ };
68
+ function readLastLines(filePath, numLines) {
69
+ try {
70
+ if (!fs.existsSync(filePath)) {
71
+ return `[Log file not found: ${path.basename(filePath)}]`;
72
+ }
73
+ const content = fs.readFileSync(filePath, 'utf-8');
74
+ const lines = content.split('\n');
75
+ const lastLines = lines.slice(-numLines).join('\n');
76
+ return lastLines || '[Log file is empty]';
77
+ }
78
+ catch (error) {
79
+ return `[Error reading log: ${error.message}]`;
80
+ }
81
+ }
82
+ function findLogFiles(logsDir, serviceType) {
83
+ const servicePath = path.join(logsDir, serviceType);
84
+ const logFiles = [];
85
+ try {
86
+ if (fs.existsSync(servicePath) && fs.statSync(servicePath).isDirectory()) {
87
+ const files = fs.readdirSync(servicePath);
88
+ for (const file of files) {
89
+ if (file.endsWith('.log')) {
90
+ logFiles.push(path.join(servicePath, file));
91
+ }
92
+ }
93
+ }
94
+ }
95
+ catch {
96
+ // Directory doesn't exist or can't be read
97
+ }
98
+ return logFiles;
99
+ }
100
+ async function getSiteLogs(args, services) {
101
+ const { site: siteQuery, logType = 'php', lines = 100 } = args;
102
+ const paramError = (0, helpers_1.validateRequiredParam)(siteQuery, 'site');
103
+ if (paramError)
104
+ return paramError;
105
+ const siteResult = (0, helpers_1.findSiteOrError)(siteQuery, services.siteData);
106
+ if ('error' in siteResult)
107
+ return siteResult.error;
108
+ const { site } = siteResult;
109
+ try {
110
+ // Get logs directory from site paths
111
+ const logsDir = site.paths?.logs || path.join(site.path, 'logs');
112
+ if (!fs.existsSync(logsDir)) {
113
+ return {
114
+ content: [{ type: 'text', text: `Logs directory not found: ${logsDir}` }],
115
+ isError: true,
116
+ };
117
+ }
118
+ const results = [];
119
+ const numLines = Math.min(Math.max(lines, 10), 1000); // Clamp between 10 and 1000
120
+ const serviceTypes = logType === 'all' ? ['php', 'nginx', 'mysql'] : [logType];
121
+ for (const serviceType of serviceTypes) {
122
+ const logFiles = findLogFiles(logsDir, serviceType);
123
+ if (logFiles.length === 0) {
124
+ results.push(`=== ${serviceType.toUpperCase()} ===\n[No log files found]`);
125
+ continue;
126
+ }
127
+ for (const logFile of logFiles) {
128
+ const fileName = path.basename(logFile);
129
+ const content = readLastLines(logFile, numLines);
130
+ results.push(`=== ${serviceType.toUpperCase()}: ${fileName} ===\n${content}`);
131
+ }
132
+ }
133
+ return {
134
+ content: [
135
+ {
136
+ type: 'text',
137
+ text: `Logs for "${site.name}" (last ${numLines} lines):\n\n${results.join('\n\n')}`,
138
+ },
139
+ ],
140
+ };
141
+ }
142
+ catch (error) {
143
+ return {
144
+ content: [{ type: 'text', text: `Failed to get site logs: ${error.message}` }],
145
+ isError: true,
146
+ };
147
+ }
148
+ }
149
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"getSiteLogs.js","sourceRoot":"","sources":["../../../../src/main/mcp/tools/getSiteLogs.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * get_site_logs Tool\n * Retrieve log files for a site\n */\n\nimport { McpToolDefinition, McpToolResult, LocalServices } from '../../../common/types';\nimport { validateRequiredParam, findSiteOrError } from './helpers';\nimport * as fs from 'fs';\nimport * as path from 'path';\n\nexport const getSiteLogsDefinition: McpToolDefinition = {\n  name: 'get_site_logs',\n  description: 'Get log file contents for a site (PHP errors, access logs, etc.)',\n  inputSchema: {\n    type: 'object',\n    properties: {\n      site: {\n        type: 'string',\n        description: 'Site name or ID',\n      },\n      logType: {\n        type: 'string',\n        enum: ['php', 'nginx', 'mysql', 'all'],\n        description: 'Type of logs to retrieve (default: php)',\n      },\n      lines: {\n        type: 'number',\n        description: 'Number of lines to return from end of log (default: 100)',\n      },\n    },\n    required: ['site'],\n  },\n};\n\ninterface GetSiteLogsArgs {\n  site: string;\n  logType?: 'php' | 'nginx' | 'mysql' | 'all';\n  lines?: number;\n}\n\nfunction readLastLines(filePath: string, numLines: number): string {\n  try {\n    if (!fs.existsSync(filePath)) {\n      return `[Log file not found: ${path.basename(filePath)}]`;\n    }\n\n    const content = fs.readFileSync(filePath, 'utf-8');\n    const lines = content.split('\\n');\n    const lastLines = lines.slice(-numLines).join('\\n');\n    return lastLines || '[Log file is empty]';\n  } catch (error: any) {\n    return `[Error reading log: ${error.message}]`;\n  }\n}\n\nfunction findLogFiles(logsDir: string, serviceType: string): string[] {\n  const servicePath = path.join(logsDir, serviceType);\n  const logFiles: string[] = [];\n\n  try {\n    if (fs.existsSync(servicePath) && fs.statSync(servicePath).isDirectory()) {\n      const files = fs.readdirSync(servicePath);\n      for (const file of files) {\n        if (file.endsWith('.log')) {\n          logFiles.push(path.join(servicePath, file));\n        }\n      }\n    }\n  } catch {\n    // Directory doesn't exist or can't be read\n  }\n\n  return logFiles;\n}\n\nexport async function getSiteLogs(\n  args: Record<string, unknown>,\n  services: LocalServices\n): Promise<McpToolResult> {\n  const { site: siteQuery, logType = 'php', lines = 100 } = args as unknown as GetSiteLogsArgs;\n\n  const paramError = validateRequiredParam(siteQuery, 'site');\n  if (paramError) return paramError;\n\n  const siteResult = findSiteOrError(siteQuery, services.siteData);\n  if ('error' in siteResult) return siteResult.error;\n  const { site } = siteResult;\n\n  try {\n    // Get logs directory from site paths\n    const logsDir = site.paths?.logs || path.join(site.path, 'logs');\n\n    if (!fs.existsSync(logsDir)) {\n      return {\n        content: [{ type: 'text', text: `Logs directory not found: ${logsDir}` }],\n        isError: true,\n      };\n    }\n\n    const results: string[] = [];\n    const numLines = Math.min(Math.max(lines, 10), 1000); // Clamp between 10 and 1000\n\n    const serviceTypes = logType === 'all' ? ['php', 'nginx', 'mysql'] : [logType];\n\n    for (const serviceType of serviceTypes) {\n      const logFiles = findLogFiles(logsDir, serviceType);\n\n      if (logFiles.length === 0) {\n        results.push(`=== ${serviceType.toUpperCase()} ===\\n[No log files found]`);\n        continue;\n      }\n\n      for (const logFile of logFiles) {\n        const fileName = path.basename(logFile);\n        const content = readLastLines(logFile, numLines);\n        results.push(`=== ${serviceType.toUpperCase()}: ${fileName} ===\\n${content}`);\n      }\n    }\n\n    return {\n      content: [\n        {\n          type: 'text',\n          text: `Logs for \"${site.name}\" (last ${numLines} lines):\\n\\n${results.join('\\n\\n')}`,\n        },\n      ],\n    };\n  } catch (error: any) {\n    return {\n      content: [{ type: 'text', text: `Failed to get site logs: ${error.message}` }],\n      isError: true,\n    };\n  }\n}\n"]}
@@ -0,0 +1,59 @@
1
+ /**
2
+ * Tool Helper Functions
3
+ */
4
+ /**
5
+ * Blocked WP-CLI commands that could allow arbitrary code execution
6
+ */
7
+ export declare const BLOCKED_WP_COMMANDS: string[];
8
+ /**
9
+ * Check if a WP-CLI command is blocked for security reasons
10
+ * Returns the blocked command name if blocked, null if safe
11
+ */
12
+ export declare function isBlockedWpCommand(command: string[]): string | null;
13
+ /**
14
+ * Validate that a file path is safe (no path traversal)
15
+ * Returns true if the path is within allowed directories
16
+ */
17
+ export declare function isValidFilePath(filePath: string, allowedDirs?: string[]): boolean;
18
+ /**
19
+ * Validate that a SQL file path is safe
20
+ * Must end in .sql and be in an allowed directory
21
+ */
22
+ export declare function isValidSqlPath(sqlPath: unknown): boolean;
23
+ /**
24
+ * MCP Tool Result type for error responses
25
+ */
26
+ export interface McpErrorResult {
27
+ content: Array<{
28
+ type: 'text';
29
+ text: string;
30
+ }>;
31
+ isError: true;
32
+ }
33
+ /**
34
+ * Create a standardized error result for MCP tools
35
+ */
36
+ export declare function createErrorResult(message: string): McpErrorResult;
37
+ /**
38
+ * Validate that a required parameter is present
39
+ * Returns an error result if missing, null if valid
40
+ */
41
+ export declare function validateRequiredParam(value: unknown, paramName: string): McpErrorResult | null;
42
+ /**
43
+ * Find a site by name or ID
44
+ * Supports partial name matching (case-insensitive)
45
+ */
46
+ export declare function findSite(query: string, siteData: any): any | undefined;
47
+ /**
48
+ * Get all site names as a comma-separated string
49
+ */
50
+ export declare function getAllSiteNames(siteData: any): string;
51
+ /**
52
+ * Find a site or return an error result
53
+ * Combines site lookup with standardized error handling
54
+ */
55
+ export declare function findSiteOrError(query: string, siteData: any): {
56
+ site: any;
57
+ } | {
58
+ error: McpErrorResult;
59
+ };
@@ -0,0 +1,179 @@
1
+ "use strict";
2
+ /**
3
+ * Tool Helper Functions
4
+ */
5
+ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
6
+ if (k2 === undefined) k2 = k;
7
+ var desc = Object.getOwnPropertyDescriptor(m, k);
8
+ if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
9
+ desc = { enumerable: true, get: function() { return m[k]; } };
10
+ }
11
+ Object.defineProperty(o, k2, desc);
12
+ }) : (function(o, m, k, k2) {
13
+ if (k2 === undefined) k2 = k;
14
+ o[k2] = m[k];
15
+ }));
16
+ var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
17
+ Object.defineProperty(o, "default", { enumerable: true, value: v });
18
+ }) : function(o, v) {
19
+ o["default"] = v;
20
+ });
21
+ var __importStar = (this && this.__importStar) || (function () {
22
+ var ownKeys = function(o) {
23
+ ownKeys = Object.getOwnPropertyNames || function (o) {
24
+ var ar = [];
25
+ for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
26
+ return ar;
27
+ };
28
+ return ownKeys(o);
29
+ };
30
+ return function (mod) {
31
+ if (mod && mod.__esModule) return mod;
32
+ var result = {};
33
+ if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
34
+ __setModuleDefault(result, mod);
35
+ return result;
36
+ };
37
+ })();
38
+ Object.defineProperty(exports, "__esModule", { value: true });
39
+ exports.BLOCKED_WP_COMMANDS = void 0;
40
+ exports.isBlockedWpCommand = isBlockedWpCommand;
41
+ exports.isValidFilePath = isValidFilePath;
42
+ exports.isValidSqlPath = isValidSqlPath;
43
+ exports.createErrorResult = createErrorResult;
44
+ exports.validateRequiredParam = validateRequiredParam;
45
+ exports.findSite = findSite;
46
+ exports.getAllSiteNames = getAllSiteNames;
47
+ exports.findSiteOrError = findSiteOrError;
48
+ const path = __importStar(require("path"));
49
+ const os = __importStar(require("os"));
50
+ /**
51
+ * Blocked WP-CLI commands that could allow arbitrary code execution
52
+ */
53
+ exports.BLOCKED_WP_COMMANDS = [
54
+ 'eval', // Execute arbitrary PHP code
55
+ 'eval-file', // Execute PHP from file
56
+ 'shell', // Interactive PHP shell
57
+ 'db query', // Raw SQL execution
58
+ 'db cli', // MySQL CLI access
59
+ ];
60
+ /**
61
+ * Check if a WP-CLI command is blocked for security reasons
62
+ * Returns the blocked command name if blocked, null if safe
63
+ */
64
+ function isBlockedWpCommand(command) {
65
+ const commandStr = command.join(' ').toLowerCase();
66
+ for (const blocked of exports.BLOCKED_WP_COMMANDS) {
67
+ if (commandStr.includes(blocked)) {
68
+ return blocked;
69
+ }
70
+ }
71
+ return null;
72
+ }
73
+ /**
74
+ * Validate that a file path is safe (no path traversal)
75
+ * Returns true if the path is within allowed directories
76
+ */
77
+ function isValidFilePath(filePath, allowedDirs) {
78
+ // Resolve to absolute path
79
+ const resolvedPath = path.resolve(filePath);
80
+ // Default allowed directories: home, tmp, and common paths
81
+ const homeDir = os.homedir();
82
+ const defaultAllowed = [
83
+ homeDir,
84
+ os.tmpdir(),
85
+ '/tmp',
86
+ '/var/tmp',
87
+ ];
88
+ const allowedDirectories = allowedDirs || defaultAllowed;
89
+ // Check if path is within any allowed directory
90
+ for (const allowedDir of allowedDirectories) {
91
+ const resolvedAllowed = path.resolve(allowedDir);
92
+ if (resolvedPath.startsWith(resolvedAllowed + path.sep) || resolvedPath === resolvedAllowed) {
93
+ return true;
94
+ }
95
+ }
96
+ return false;
97
+ }
98
+ /**
99
+ * Validate that a SQL file path is safe
100
+ * Must end in .sql and be in an allowed directory
101
+ */
102
+ function isValidSqlPath(sqlPath) {
103
+ if (typeof sqlPath !== 'string' || !sqlPath) {
104
+ return false;
105
+ }
106
+ // Must end in .sql
107
+ if (!sqlPath.toLowerCase().endsWith('.sql')) {
108
+ return false;
109
+ }
110
+ // Must be in an allowed directory
111
+ return isValidFilePath(sqlPath);
112
+ }
113
+ /**
114
+ * Create a standardized error result for MCP tools
115
+ */
116
+ function createErrorResult(message) {
117
+ return {
118
+ content: [{ type: 'text', text: message }],
119
+ isError: true,
120
+ };
121
+ }
122
+ /**
123
+ * Validate that a required parameter is present
124
+ * Returns an error result if missing, null if valid
125
+ */
126
+ function validateRequiredParam(value, paramName) {
127
+ if (!value) {
128
+ return createErrorResult(`Error: ${paramName} parameter is required`);
129
+ }
130
+ return null;
131
+ }
132
+ /**
133
+ * Find a site by name or ID
134
+ * Supports partial name matching (case-insensitive)
135
+ */
136
+ function findSite(query, siteData) {
137
+ const sitesMap = siteData.getSites();
138
+ const sites = Object.values(sitesMap);
139
+ // Try exact ID match first
140
+ const byId = sites.find((s) => s.id === query);
141
+ if (byId)
142
+ return byId;
143
+ // Try exact name match (case-insensitive)
144
+ const byExactName = sites.find((s) => s.name.toLowerCase() === query.toLowerCase());
145
+ if (byExactName)
146
+ return byExactName;
147
+ // Try partial name match (case-insensitive)
148
+ const byPartialName = sites.find((s) => s.name.toLowerCase().includes(query.toLowerCase()));
149
+ if (byPartialName)
150
+ return byPartialName;
151
+ // Try domain match
152
+ const byDomain = sites.find((s) => s.domain?.toLowerCase() === query.toLowerCase());
153
+ if (byDomain)
154
+ return byDomain;
155
+ return undefined;
156
+ }
157
+ /**
158
+ * Get all site names as a comma-separated string
159
+ */
160
+ function getAllSiteNames(siteData) {
161
+ const sitesMap = siteData.getSites();
162
+ const sites = Object.values(sitesMap);
163
+ return sites.map((s) => s.name).join(', ') || 'none';
164
+ }
165
+ /**
166
+ * Find a site or return an error result
167
+ * Combines site lookup with standardized error handling
168
+ */
169
+ function findSiteOrError(query, siteData) {
170
+ const site = findSite(query, siteData);
171
+ if (!site) {
172
+ const siteNames = getAllSiteNames(siteData);
173
+ return {
174
+ error: createErrorResult(`Site not found: "${query}". Available sites: ${siteNames}`),
175
+ };
176
+ }
177
+ return { site };
178
+ }
179
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"helpers.js","sourceRoot":"","sources":["../../../../src/main/mcp/tools/helpers.ts"],"names":[],"mappings":";AAAA;;GAEG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAoBH,gDAQC;AAMD,0CAwBC;AAMD,wCAYC;AAaD,8CAKC;AAMD,sDAQC;AAMD,4BAqBC;AAKD,0CAIC;AAMD,0CAcC;AAlKD,2CAA6B;AAC7B,uCAAyB;AAEzB;;GAEG;AACU,QAAA,mBAAmB,GAAG;IACjC,MAAM,EAAS,6BAA6B;IAC5C,WAAW,EAAI,wBAAwB;IACvC,OAAO,EAAQ,wBAAwB;IACvC,UAAU,EAAK,oBAAoB;IACnC,QAAQ,EAAO,mBAAmB;CACnC,CAAC;AAEF;;;GAGG;AACH,SAAgB,kBAAkB,CAAC,OAAiB;IAClD,MAAM,UAAU,GAAG,OAAO,CAAC,IAAI,CAAC,GAAG,CAAC,CAAC,WAAW,EAAE,CAAC;IACnD,KAAK,MAAM,OAAO,IAAI,2BAAmB,EAAE,CAAC;QAC1C,IAAI,UAAU,CAAC,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC;YACjC,OAAO,OAAO,CAAC;QACjB,CAAC;IACH,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AAED;;;GAGG;AACH,SAAgB,eAAe,CAAC,QAAgB,EAAE,WAAsB;IACtE,2BAA2B;IAC3B,MAAM,YAAY,GAAG,IAAI,CAAC,OAAO,CAAC,QAAQ,CAAC,CAAC;IAE5C,2DAA2D;IAC3D,MAAM,OAAO,GAAG,EAAE,CAAC,OAAO,EAAE,CAAC;IAC7B,MAAM,cAAc,GAAG;QACrB,OAAO;QACP,EAAE,CAAC,MAAM,EAAE;QACX,MAAM;QACN,UAAU;KACX,CAAC;IAEF,MAAM,kBAAkB,GAAG,WAAW,IAAI,cAAc,CAAC;IAEzD,gDAAgD;IAChD,KAAK,MAAM,UAAU,IAAI,kBAAkB,EAAE,CAAC;QAC5C,MAAM,eAAe,GAAG,IAAI,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;QACjD,IAAI,YAAY,CAAC,UAAU,CAAC,eAAe,GAAG,IAAI,CAAC,GAAG,CAAC,IAAI,YAAY,KAAK,eAAe,EAAE,CAAC;YAC5F,OAAO,IAAI,CAAC;QACd,CAAC;IACH,CAAC;IAED,OAAO,KAAK,CAAC;AACf,CAAC;AAED;;;GAGG;AACH,SAAgB,cAAc,CAAC,OAAgB;IAC7C,IAAI,OAAO,OAAO,KAAK,QAAQ,IAAI,CAAC,OAAO,EAAE,CAAC;QAC5C,OAAO,KAAK,CAAC;IACf,CAAC;IAED,mBAAmB;IACnB,IAAI,CAAC,OAAO,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;QAC5C,OAAO,KAAK,CAAC;IACf,CAAC;IAED,kCAAkC;IAClC,OAAO,eAAe,CAAC,OAAO,CAAC,CAAC;AAClC,CAAC;AAUD;;GAEG;AACH,SAAgB,iBAAiB,CAAC,OAAe;IAC/C,OAAO;QACL,OAAO,EAAE,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC;QAC1C,OAAO,EAAE,IAAI;KACd,CAAC;AACJ,CAAC;AAED;;;GAGG;AACH,SAAgB,qBAAqB,CACnC,KAAc,EACd,SAAiB;IAEjB,IAAI,CAAC,KAAK,EAAE,CAAC;QACX,OAAO,iBAAiB,CAAC,UAAU,SAAS,wBAAwB,CAAC,CAAC;IACxE,CAAC;IACD,OAAO,IAAI,CAAC;AACd,CAAC;AAED;;;GAGG;AACH,SAAgB,QAAQ,CAAC,KAAa,EAAE,QAAa;IACnD,MAAM,QAAQ,GAAG,QAAQ,CAAC,QAAQ,EAAE,CAAC;IACrC,MAAM,KAAK,GAAG,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAU,CAAC;IAE/C,2BAA2B;IAC3B,MAAM,IAAI,GAAG,KAAK,CAAC,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,KAAK,CAAC,CAAC;IACpD,IAAI,IAAI;QAAE,OAAO,IAAI,CAAC;IAEtB,0CAA0C;IAC1C,MAAM,WAAW,GAAG,KAAK,CAAC,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,KAAK,KAAK,CAAC,WAAW,EAAE,CAAC,CAAC;IACzF,IAAI,WAAW;QAAE,OAAO,WAAW,CAAC;IAEpC,4CAA4C;IAC5C,MAAM,aAAa,GAAG,KAAK,CAAC,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,WAAW,EAAE,CAAC,QAAQ,CAAC,KAAK,CAAC,WAAW,EAAE,CAAC,CAAC,CAAC;IACjG,IAAI,aAAa;QAAE,OAAO,aAAa,CAAC;IAExC,mBAAmB;IACnB,MAAM,QAAQ,GAAG,KAAK,CAAC,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,EAAE,WAAW,EAAE,KAAK,KAAK,CAAC,WAAW,EAAE,CAAC,CAAC;IACzF,IAAI,QAAQ;QAAE,OAAO,QAAQ,CAAC;IAE9B,OAAO,SAAS,CAAC;AACnB,CAAC;AAED;;GAEG;AACH,SAAgB,eAAe,CAAC,QAAa;IAC3C,MAAM,QAAQ,GAAG,QAAQ,CAAC,QAAQ,EAAE,CAAC;IACrC,MAAM,KAAK,GAAG,MAAM,CAAC,MAAM,CAAC,QAAQ,CAAU,CAAC;IAC/C,OAAO,KAAK,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,MAAM,CAAC;AAC5D,CAAC;AAED;;;GAGG;AACH,SAAgB,eAAe,CAC7B,KAAa,EACb,QAAa;IAEb,MAAM,IAAI,GAAG,QAAQ,CAAC,KAAK,EAAE,QAAQ,CAAC,CAAC;IACvC,IAAI,CAAC,IAAI,EAAE,CAAC;QACV,MAAM,SAAS,GAAG,eAAe,CAAC,QAAQ,CAAC,CAAC;QAC5C,OAAO;YACL,KAAK,EAAE,iBAAiB,CACtB,oBAAoB,KAAK,uBAAuB,SAAS,EAAE,CAC5D;SACF,CAAC;IACJ,CAAC;IACD,OAAO,EAAE,IAAI,EAAE,CAAC;AAClB,CAAC","sourcesContent":["/**\n * Tool Helper Functions\n */\n\nimport * as path from 'path';\nimport * as os from 'os';\n\n/**\n * Blocked WP-CLI commands that could allow arbitrary code execution\n */\nexport const BLOCKED_WP_COMMANDS = [\n  'eval',        // Execute arbitrary PHP code\n  'eval-file',   // Execute PHP from file\n  'shell',       // Interactive PHP shell\n  'db query',    // Raw SQL execution\n  'db cli',      // MySQL CLI access\n];\n\n/**\n * Check if a WP-CLI command is blocked for security reasons\n * Returns the blocked command name if blocked, null if safe\n */\nexport function isBlockedWpCommand(command: string[]): string | null {\n  const commandStr = command.join(' ').toLowerCase();\n  for (const blocked of BLOCKED_WP_COMMANDS) {\n    if (commandStr.includes(blocked)) {\n      return blocked;\n    }\n  }\n  return null;\n}\n\n/**\n * Validate that a file path is safe (no path traversal)\n * Returns true if the path is within allowed directories\n */\nexport function isValidFilePath(filePath: string, allowedDirs?: string[]): boolean {\n  // Resolve to absolute path\n  const resolvedPath = path.resolve(filePath);\n\n  // Default allowed directories: home, tmp, and common paths\n  const homeDir = os.homedir();\n  const defaultAllowed = [\n    homeDir,\n    os.tmpdir(),\n    '/tmp',\n    '/var/tmp',\n  ];\n\n  const allowedDirectories = allowedDirs || defaultAllowed;\n\n  // Check if path is within any allowed directory\n  for (const allowedDir of allowedDirectories) {\n    const resolvedAllowed = path.resolve(allowedDir);\n    if (resolvedPath.startsWith(resolvedAllowed + path.sep) || resolvedPath === resolvedAllowed) {\n      return true;\n    }\n  }\n\n  return false;\n}\n\n/**\n * Validate that a SQL file path is safe\n * Must end in .sql and be in an allowed directory\n */\nexport function isValidSqlPath(sqlPath: unknown): boolean {\n  if (typeof sqlPath !== 'string' || !sqlPath) {\n    return false;\n  }\n\n  // Must end in .sql\n  if (!sqlPath.toLowerCase().endsWith('.sql')) {\n    return false;\n  }\n\n  // Must be in an allowed directory\n  return isValidFilePath(sqlPath);\n}\n\n/**\n * MCP Tool Result type for error responses\n */\nexport interface McpErrorResult {\n  content: Array<{ type: 'text'; text: string }>;\n  isError: true;\n}\n\n/**\n * Create a standardized error result for MCP tools\n */\nexport function createErrorResult(message: string): McpErrorResult {\n  return {\n    content: [{ type: 'text', text: message }],\n    isError: true,\n  };\n}\n\n/**\n * Validate that a required parameter is present\n * Returns an error result if missing, null if valid\n */\nexport function validateRequiredParam(\n  value: unknown,\n  paramName: string\n): McpErrorResult | null {\n  if (!value) {\n    return createErrorResult(`Error: ${paramName} parameter is required`);\n  }\n  return null;\n}\n\n/**\n * Find a site by name or ID\n * Supports partial name matching (case-insensitive)\n */\nexport function findSite(query: string, siteData: any): any | undefined {\n  const sitesMap = siteData.getSites();\n  const sites = Object.values(sitesMap) as any[];\n\n  // Try exact ID match first\n  const byId = sites.find((s: any) => s.id === query);\n  if (byId) return byId;\n\n  // Try exact name match (case-insensitive)\n  const byExactName = sites.find((s: any) => s.name.toLowerCase() === query.toLowerCase());\n  if (byExactName) return byExactName;\n\n  // Try partial name match (case-insensitive)\n  const byPartialName = sites.find((s: any) => s.name.toLowerCase().includes(query.toLowerCase()));\n  if (byPartialName) return byPartialName;\n\n  // Try domain match\n  const byDomain = sites.find((s: any) => s.domain?.toLowerCase() === query.toLowerCase());\n  if (byDomain) return byDomain;\n\n  return undefined;\n}\n\n/**\n * Get all site names as a comma-separated string\n */\nexport function getAllSiteNames(siteData: any): string {\n  const sitesMap = siteData.getSites();\n  const sites = Object.values(sitesMap) as any[];\n  return sites.map((s: any) => s.name).join(', ') || 'none';\n}\n\n/**\n * Find a site or return an error result\n * Combines site lookup with standardized error handling\n */\nexport function findSiteOrError(\n  query: string,\n  siteData: any\n): { site: any } | { error: McpErrorResult } {\n  const site = findSite(query, siteData);\n  if (!site) {\n    const siteNames = getAllSiteNames(siteData);\n    return {\n      error: createErrorResult(\n        `Site not found: \"${query}\". Available sites: ${siteNames}`\n      ),\n    };\n  }\n  return { site };\n}\n"]}
@@ -0,0 +1,7 @@
1
+ /**
2
+ * import_database Tool
3
+ * Import SQL file into site database using WP-CLI
4
+ */
5
+ import { McpToolDefinition, McpToolResult, LocalServices } from '../../../common/types';
6
+ export declare const importDatabaseDefinition: McpToolDefinition;
7
+ export declare function importDatabase(args: Record<string, unknown>, services: LocalServices): Promise<McpToolResult>;