@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,479 @@
1
+ "use strict";
2
+ /**
3
+ * MCP Server Addon - Renderer Entry Point
4
+ * Adds MCP Server preferences panel to Local
5
+ *
6
+ * Uses class components for compatibility with Local's Electron environment.
7
+ */
8
+ var __importDefault = (this && this.__importDefault) || function (mod) {
9
+ return (mod && mod.__esModule) ? mod : { "default": mod };
10
+ };
11
+ Object.defineProperty(exports, "__esModule", { value: true });
12
+ const react_1 = __importDefault(require("react"));
13
+ const theme_1 = require("../common/theme");
14
+ // MCP Preferences Panel Component (Class-based for Local compatibility)
15
+ class McpPreferencesPanel extends react_1.default.Component {
16
+ constructor() {
17
+ super(...arguments);
18
+ this.state = {
19
+ status: null,
20
+ connectionInfo: null,
21
+ loading: true,
22
+ copied: false,
23
+ actionInProgress: false,
24
+ activeTab: 'status',
25
+ colors: (0, theme_1.getThemeColors)(),
26
+ };
27
+ this.fetchStatus = async () => {
28
+ try {
29
+ const result = (await this.props.electron?.ipcRenderer?.invoke('mcp:getStatus'));
30
+ if (result) {
31
+ this.setState({ status: result });
32
+ }
33
+ }
34
+ catch (error) {
35
+ console.error('Failed to fetch MCP status:', error);
36
+ }
37
+ };
38
+ this.fetchConnectionInfo = async () => {
39
+ try {
40
+ const result = (await this.props.electron?.ipcRenderer?.invoke('mcp:getConnectionInfo'));
41
+ if (result) {
42
+ this.setState({ connectionInfo: result });
43
+ }
44
+ }
45
+ catch (error) {
46
+ console.error('Failed to fetch connection info:', error);
47
+ }
48
+ };
49
+ this.handleCopyStdioConfig = () => {
50
+ const config = {
51
+ mcpServers: {
52
+ local: {
53
+ type: 'stdio',
54
+ command: 'node',
55
+ args: ['/path/to/local-addon-mcp-server/bin/mcp-stdio.js'],
56
+ },
57
+ },
58
+ };
59
+ navigator.clipboard.writeText(JSON.stringify(config, null, 2));
60
+ this.setState({ copied: true });
61
+ setTimeout(() => this.setState({ copied: false }), 2000);
62
+ };
63
+ this.handleCopySseConfig = () => {
64
+ const { connectionInfo } = this.state;
65
+ if (connectionInfo) {
66
+ const config = {
67
+ mcpServers: {
68
+ local: {
69
+ url: `${connectionInfo.url}/mcp/sse`,
70
+ transport: 'sse',
71
+ headers: {
72
+ Authorization: `Bearer ${connectionInfo.authToken}`,
73
+ },
74
+ },
75
+ },
76
+ };
77
+ navigator.clipboard.writeText(JSON.stringify(config, null, 2));
78
+ this.setState({ copied: true });
79
+ setTimeout(() => this.setState({ copied: false }), 2000);
80
+ }
81
+ };
82
+ this.handleTestConnection = async () => {
83
+ const { connectionInfo } = this.state;
84
+ try {
85
+ const response = await fetch(`http://127.0.0.1:${connectionInfo?.port}/health`);
86
+ const data = await response.json();
87
+ if (data.status === 'ok') {
88
+ alert('Connection successful! MCP server is healthy.');
89
+ }
90
+ else {
91
+ alert('Connection failed: Server returned unexpected response');
92
+ }
93
+ }
94
+ catch (error) {
95
+ const message = error instanceof Error ? error.message : 'Unknown error';
96
+ alert(`Connection failed: ${message}`);
97
+ }
98
+ };
99
+ this.handleStartStop = async () => {
100
+ const { status } = this.state;
101
+ this.setState({ actionInProgress: true });
102
+ try {
103
+ if (status?.running) {
104
+ await this.props.electron?.ipcRenderer?.invoke('mcp:stop');
105
+ }
106
+ else {
107
+ await this.props.electron?.ipcRenderer?.invoke('mcp:start');
108
+ }
109
+ await this.fetchStatus();
110
+ await this.fetchConnectionInfo();
111
+ }
112
+ catch (error) {
113
+ const message = error instanceof Error ? error.message : 'Unknown error';
114
+ alert(`Action failed: ${message}`);
115
+ }
116
+ finally {
117
+ this.setState({ actionInProgress: false });
118
+ }
119
+ };
120
+ this.handleRestart = async () => {
121
+ this.setState({ actionInProgress: true });
122
+ try {
123
+ await this.props.electron?.ipcRenderer?.invoke('mcp:restart');
124
+ await this.fetchStatus();
125
+ await this.fetchConnectionInfo();
126
+ }
127
+ catch (error) {
128
+ const message = error instanceof Error ? error.message : 'Unknown error';
129
+ alert(`Restart failed: ${message}`);
130
+ }
131
+ finally {
132
+ this.setState({ actionInProgress: false });
133
+ }
134
+ };
135
+ this.handleRegenerateToken = async () => {
136
+ if (!confirm('Regenerate authentication token? You will need to update your AI tool configuration.')) {
137
+ return;
138
+ }
139
+ this.setState({ actionInProgress: true });
140
+ try {
141
+ const result = (await this.props.electron?.ipcRenderer?.invoke('mcp:regenerateToken'));
142
+ if (result?.success) {
143
+ await this.fetchConnectionInfo();
144
+ alert('Token regenerated successfully. Please update your AI tool configuration.');
145
+ }
146
+ else {
147
+ alert(`Failed to regenerate token: ${result?.error}`);
148
+ }
149
+ }
150
+ catch (error) {
151
+ const message = error instanceof Error ? error.message : 'Unknown error';
152
+ alert(`Failed to regenerate token: ${message}`);
153
+ }
154
+ finally {
155
+ this.setState({ actionInProgress: false });
156
+ }
157
+ };
158
+ }
159
+ componentDidMount() {
160
+ this.init();
161
+ this.statusInterval = setInterval(() => this.fetchStatus(), 5000);
162
+ this.themeCleanup = (0, theme_1.onThemeChange)(() => {
163
+ this.setState({ colors: (0, theme_1.getThemeColors)() });
164
+ });
165
+ }
166
+ componentWillUnmount() {
167
+ if (this.statusInterval) {
168
+ clearInterval(this.statusInterval);
169
+ }
170
+ if (this.themeCleanup) {
171
+ this.themeCleanup();
172
+ }
173
+ }
174
+ async init() {
175
+ await Promise.all([this.fetchStatus(), this.fetchConnectionInfo()]);
176
+ this.setState({ loading: false });
177
+ }
178
+ getStatusColor() {
179
+ const { status, colors } = this.state;
180
+ if (!status)
181
+ return colors.textMuted;
182
+ if (status.running)
183
+ return colors.successText;
184
+ return colors.errorText;
185
+ }
186
+ getStatusText() {
187
+ const { loading, status } = this.state;
188
+ if (loading)
189
+ return 'Loading...';
190
+ if (!status)
191
+ return 'Unknown';
192
+ if (status.running)
193
+ return 'Running';
194
+ return 'Stopped';
195
+ }
196
+ formatUptime(seconds) {
197
+ if (seconds < 60)
198
+ return `${Math.floor(seconds)}s`;
199
+ if (seconds < 3600)
200
+ return `${Math.floor(seconds / 60)}m`;
201
+ return `${Math.floor(seconds / 3600)}h ${Math.floor((seconds % 3600) / 60)}m`;
202
+ }
203
+ getTabStyle(isActive) {
204
+ const { colors } = this.state;
205
+ return {
206
+ padding: '8px 16px',
207
+ border: 'none',
208
+ borderBottom: isActive ? `2px solid ${colors.primary}` : '2px solid transparent',
209
+ backgroundColor: 'transparent',
210
+ color: isActive ? colors.primary : colors.textSecondary,
211
+ cursor: 'pointer',
212
+ fontSize: '14px',
213
+ fontWeight: isActive ? 600 : 400,
214
+ };
215
+ }
216
+ getButtonStyle(bgColor, disabled) {
217
+ return {
218
+ padding: '8px 16px',
219
+ backgroundColor: bgColor,
220
+ color: 'white',
221
+ border: 'none',
222
+ borderRadius: '4px',
223
+ cursor: disabled ? 'not-allowed' : 'pointer',
224
+ fontSize: '13px',
225
+ opacity: disabled ? 0.5 : 1,
226
+ };
227
+ }
228
+ renderStatusTab() {
229
+ const { status, connectionInfo, actionInProgress, colors } = this.state;
230
+ return (react_1.default.createElement(react_1.default.Fragment, null,
231
+ react_1.default.createElement("div", { style: { marginBottom: '24px' } },
232
+ react_1.default.createElement("h3", { style: {
233
+ fontSize: '14px',
234
+ fontWeight: 600,
235
+ marginBottom: '12px',
236
+ color: colors.textPrimary,
237
+ } }, "Server Status"),
238
+ react_1.default.createElement("div", { style: {
239
+ display: 'flex',
240
+ alignItems: 'center',
241
+ gap: '12px',
242
+ padding: '12px',
243
+ backgroundColor: colors.panelBgSecondary,
244
+ borderRadius: '6px',
245
+ } },
246
+ react_1.default.createElement("span", { style: {
247
+ width: '12px',
248
+ height: '12px',
249
+ borderRadius: '50%',
250
+ backgroundColor: this.getStatusColor(),
251
+ } }),
252
+ react_1.default.createElement("span", { style: { fontWeight: 500, color: colors.textPrimary } }, this.getStatusText()),
253
+ status?.running && (react_1.default.createElement(react_1.default.Fragment, null,
254
+ react_1.default.createElement("span", { style: { color: colors.textSecondary } }, "|"),
255
+ react_1.default.createElement("span", { style: { color: colors.textSecondary } },
256
+ "Port: ",
257
+ status.port),
258
+ react_1.default.createElement("span", { style: { color: colors.textSecondary } }, "|"),
259
+ react_1.default.createElement("span", { style: { color: colors.textSecondary } },
260
+ "Uptime: ",
261
+ this.formatUptime(status.uptime)))))),
262
+ react_1.default.createElement("div", { style: { marginBottom: '24px' } },
263
+ react_1.default.createElement("h3", { style: {
264
+ fontSize: '14px',
265
+ fontWeight: 600,
266
+ marginBottom: '12px',
267
+ color: colors.textPrimary,
268
+ } }, "Server Controls"),
269
+ react_1.default.createElement("div", { style: { display: 'flex', gap: '12px', flexWrap: 'wrap' } },
270
+ react_1.default.createElement("button", { onClick: this.handleStartStop, disabled: actionInProgress, style: this.getButtonStyle(status?.running ? colors.errorText : colors.successText, actionInProgress) }, status?.running ? 'Stop Server' : 'Start Server'),
271
+ react_1.default.createElement("button", { onClick: this.handleRestart, disabled: actionInProgress || !status?.running, style: this.getButtonStyle('#6c757d', actionInProgress || !status?.running) }, "Restart Server"),
272
+ react_1.default.createElement("button", { onClick: this.handleTestConnection, disabled: !status?.running, style: this.getButtonStyle(status?.running ? '#17a2b8' : colors.textMuted, !status?.running) }, "Test Connection"))),
273
+ connectionInfo && (react_1.default.createElement("div", { style: { marginBottom: '24px' } },
274
+ react_1.default.createElement("h3", { style: {
275
+ fontSize: '14px',
276
+ fontWeight: 600,
277
+ marginBottom: '12px',
278
+ color: colors.textPrimary,
279
+ } }, "Connection Info"),
280
+ react_1.default.createElement("div", { style: {
281
+ padding: '12px',
282
+ backgroundColor: colors.panelBgSecondary,
283
+ borderRadius: '6px',
284
+ fontFamily: 'monospace',
285
+ fontSize: '12px',
286
+ color: colors.textPrimary,
287
+ } },
288
+ react_1.default.createElement("div", { style: { marginBottom: '8px' } },
289
+ react_1.default.createElement("strong", null, "URL:"),
290
+ " ",
291
+ connectionInfo.url),
292
+ react_1.default.createElement("div", { style: { marginBottom: '8px' } },
293
+ react_1.default.createElement("strong", null, "stdio script:"),
294
+ " bin/mcp-stdio.js"),
295
+ react_1.default.createElement("div", { style: { marginBottom: '8px' } },
296
+ react_1.default.createElement("strong", null, "Version:"),
297
+ " ",
298
+ connectionInfo.version),
299
+ react_1.default.createElement("div", null,
300
+ react_1.default.createElement("strong", null, "Tools:"),
301
+ " ",
302
+ connectionInfo.tools.join(', '))))),
303
+ react_1.default.createElement("div", { style: { marginBottom: '24px' } },
304
+ react_1.default.createElement("h3", { style: {
305
+ fontSize: '14px',
306
+ fontWeight: 600,
307
+ marginBottom: '12px',
308
+ color: colors.textPrimary,
309
+ } }, "Security"),
310
+ react_1.default.createElement("div", { style: { display: 'flex', gap: '12px', flexWrap: 'wrap' } },
311
+ react_1.default.createElement("button", { onClick: this.handleRegenerateToken, disabled: actionInProgress || !status?.running, style: {
312
+ padding: '8px 16px',
313
+ backgroundColor: '#ffc107',
314
+ color: '#212529',
315
+ border: 'none',
316
+ borderRadius: '4px',
317
+ cursor: actionInProgress || !status?.running ? 'not-allowed' : 'pointer',
318
+ fontSize: '13px',
319
+ opacity: actionInProgress || !status?.running ? 0.5 : 1,
320
+ } }, "Regenerate Token")),
321
+ react_1.default.createElement("p", { style: { fontSize: '12px', color: colors.textSecondary, marginTop: '8px' } }, "Regenerating the token will require you to update your AI tool configuration."))));
322
+ }
323
+ renderSetupTab() {
324
+ const { connectionInfo, copied, colors } = this.state;
325
+ return (react_1.default.createElement(react_1.default.Fragment, null,
326
+ react_1.default.createElement("div", { style: { marginBottom: '24px' } },
327
+ react_1.default.createElement("h3", { style: {
328
+ fontSize: '14px',
329
+ fontWeight: 600,
330
+ marginBottom: '12px',
331
+ color: colors.textPrimary,
332
+ } }, "Claude Code (Recommended)"),
333
+ react_1.default.createElement("div", { style: {
334
+ padding: '16px',
335
+ backgroundColor: colors.panelBgSecondary,
336
+ borderRadius: '6px',
337
+ } },
338
+ react_1.default.createElement("p", { style: { margin: '0 0 12px 0', fontSize: '13px', color: colors.textPrimary } },
339
+ "Add the following to your",
340
+ ' ',
341
+ react_1.default.createElement("code", { style: {
342
+ backgroundColor: colors.panelBgCode,
343
+ padding: '2px 4px',
344
+ borderRadius: '3px',
345
+ color: '#f8f8f2',
346
+ } }, "~/.claude.json"),
347
+ ' ',
348
+ "file:"),
349
+ react_1.default.createElement("pre", { style: {
350
+ backgroundColor: colors.panelBgCode,
351
+ color: '#f8f8f2',
352
+ padding: '12px',
353
+ borderRadius: '4px',
354
+ fontSize: '11px',
355
+ overflow: 'auto',
356
+ margin: '0 0 12px 0',
357
+ } }, `{
358
+ "mcpServers": {
359
+ "local": {
360
+ "type": "stdio",
361
+ "command": "node",
362
+ "args": ["/path/to/local-addon-mcp-server/bin/mcp-stdio.js"]
363
+ }
364
+ }
365
+ }`),
366
+ react_1.default.createElement("button", { onClick: this.handleCopyStdioConfig, style: this.getButtonStyle(colors.primary) }, copied ? 'Copied!' : 'Copy Config'))),
367
+ react_1.default.createElement("div", { style: { marginBottom: '24px' } },
368
+ react_1.default.createElement("h3", { style: {
369
+ fontSize: '14px',
370
+ fontWeight: 600,
371
+ marginBottom: '12px',
372
+ color: colors.textPrimary,
373
+ } }, "Claude.ai / ChatGPT / Other AI Tools"),
374
+ react_1.default.createElement("div", { style: {
375
+ padding: '16px',
376
+ backgroundColor: colors.panelBgSecondary,
377
+ borderRadius: '6px',
378
+ } },
379
+ react_1.default.createElement("p", { style: { margin: '0 0 12px 0', fontSize: '13px', color: colors.textPrimary } }, "For tools that support SSE transport, use this configuration:"),
380
+ react_1.default.createElement("pre", { style: {
381
+ backgroundColor: colors.panelBgCode,
382
+ color: '#f8f8f2',
383
+ padding: '12px',
384
+ borderRadius: '4px',
385
+ fontSize: '11px',
386
+ overflow: 'auto',
387
+ margin: '0 0 12px 0',
388
+ } }, connectionInfo
389
+ ? `{
390
+ "mcpServers": {
391
+ "local": {
392
+ "url": "${connectionInfo.url}/mcp/sse",
393
+ "transport": "sse",
394
+ "headers": {
395
+ "Authorization": "Bearer ${connectionInfo.authToken.substring(0, 20)}..."
396
+ }
397
+ }
398
+ }
399
+ }`
400
+ : 'Server not running'),
401
+ react_1.default.createElement("button", { onClick: this.handleCopySseConfig, disabled: !connectionInfo, style: this.getButtonStyle(connectionInfo ? colors.primary : colors.textMuted, !connectionInfo) }, copied ? 'Copied!' : 'Copy SSE Config'))),
402
+ react_1.default.createElement("div", { style: { marginBottom: '24px' } },
403
+ react_1.default.createElement("h3", { style: {
404
+ fontSize: '14px',
405
+ fontWeight: 600,
406
+ marginBottom: '12px',
407
+ color: colors.textPrimary,
408
+ } }, "Example Commands"),
409
+ react_1.default.createElement("div", { style: {
410
+ padding: '16px',
411
+ backgroundColor: colors.infoBg,
412
+ borderRadius: '6px',
413
+ border: `1px solid ${colors.infoBorder}`,
414
+ } },
415
+ react_1.default.createElement("p", { style: {
416
+ margin: '0 0 8px 0',
417
+ fontSize: '13px',
418
+ fontWeight: 500,
419
+ color: colors.infoText,
420
+ } }, "Try saying to your AI assistant:"),
421
+ react_1.default.createElement("ul", { style: { margin: 0, paddingLeft: '20px', fontSize: '13px', color: colors.infoText } },
422
+ react_1.default.createElement("li", null, "\"List my Local sites\""),
423
+ react_1.default.createElement("li", null, "\"Start the blog site\""),
424
+ react_1.default.createElement("li", null, "\"Create a new site called test-project\""),
425
+ react_1.default.createElement("li", null, "\"Run wp plugin list on my-site\""),
426
+ react_1.default.createElement("li", null, "\"Stop all running sites\""),
427
+ react_1.default.createElement("li", null, "\"What plugins are installed on my-site?\""))))));
428
+ }
429
+ render() {
430
+ const { activeTab, colors } = this.state;
431
+ return (react_1.default.createElement("div", { style: { padding: '20px', color: colors.textPrimary } },
432
+ react_1.default.createElement("h2", { style: {
433
+ marginBottom: '20px',
434
+ fontSize: '18px',
435
+ fontWeight: 600,
436
+ color: colors.textPrimary,
437
+ } }, "MCP Server"),
438
+ react_1.default.createElement("p", { style: { color: colors.textSecondary, marginBottom: '24px' } }, "The MCP (Model Context Protocol) server enables AI tools like Claude Code to control your Local sites."),
439
+ react_1.default.createElement("div", { style: { borderBottom: `1px solid ${colors.border}`, marginBottom: '24px' } },
440
+ react_1.default.createElement("button", { style: this.getTabStyle(activeTab === 'status'), onClick: () => this.setState({ activeTab: 'status' }) }, "Status & Controls"),
441
+ react_1.default.createElement("button", { style: this.getTabStyle(activeTab === 'setup'), onClick: () => this.setState({ activeTab: 'setup' }) }, "AI Tool Setup")),
442
+ activeTab === 'status' && this.renderStatusTab(),
443
+ activeTab === 'setup' && this.renderSetupTab()));
444
+ }
445
+ }
446
+ // Export as a constructor function that Local can instantiate
447
+ module.exports = function McpServerRenderer(context) {
448
+ // CLI-only mode - preferences panel disabled
449
+ // To enable MCP preferences UI, uncomment the following:
450
+ /*
451
+ const { hooks, electron } = context as {
452
+ hooks: { addFilter: (name: string, callback: (items: unknown[]) => unknown[]) => void };
453
+ electron: McpPreferencesPanelProps['electron'];
454
+ };
455
+
456
+ console.log('[MCP Server] Renderer loading...');
457
+
458
+ // Add MCP Server to preferences menu
459
+ hooks.addFilter('preferencesMenuItems', (menuItems: unknown[]) => {
460
+ console.log('[MCP Server] Adding MCP Server to preferences');
461
+
462
+ (menuItems as Array<Record<string, unknown>>).push({
463
+ path: 'mcp-server',
464
+ displayName: 'MCP Server',
465
+ sections: () => <McpPreferencesPanel electron={electron} />,
466
+ onApply: () => {
467
+ // No apply action needed for read-only display
468
+ },
469
+ });
470
+
471
+ return menuItems;
472
+ });
473
+
474
+ console.log('[MCP Server] Renderer setup complete');
475
+ */
476
+ // Silence unused variable warnings
477
+ void context;
478
+ };
479
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/renderer/index.tsx"],"names":[],"mappings":"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","sourcesContent":["/**\n * MCP Server Addon - Renderer Entry Point\n * Adds MCP Server preferences panel to Local\n *\n * Uses class components for compatibility with Local's Electron environment.\n */\n\nimport React from 'react';\nimport { getThemeColors, onThemeChange, ThemeColors } from '../common/theme';\n\n// Type definitions\ninterface McpStatus {\n  running: boolean;\n  port: number;\n  uptime: number;\n  error?: string;\n}\n\ninterface ConnectionInfo {\n  url: string;\n  authToken: string;\n  port: number;\n  version: string;\n  tools: string[];\n}\n\ninterface McpPreferencesPanelProps {\n  electron: {\n    ipcRenderer?: {\n      invoke: (channel: string, ...args: unknown[]) => Promise<unknown>;\n    };\n  };\n}\n\ninterface McpPreferencesPanelState {\n  status: McpStatus | null;\n  connectionInfo: ConnectionInfo | null;\n  loading: boolean;\n  copied: boolean;\n  actionInProgress: boolean;\n  activeTab: 'status' | 'setup';\n  colors: ThemeColors;\n}\n\n// MCP Preferences Panel Component (Class-based for Local compatibility)\nclass McpPreferencesPanel extends React.Component<\n  McpPreferencesPanelProps,\n  McpPreferencesPanelState\n> {\n  private statusInterval?: ReturnType<typeof setInterval>;\n  private themeCleanup?: () => void;\n\n  state: McpPreferencesPanelState = {\n    status: null,\n    connectionInfo: null,\n    loading: true,\n    copied: false,\n    actionInProgress: false,\n    activeTab: 'status',\n    colors: getThemeColors(),\n  };\n\n  componentDidMount(): void {\n    this.init();\n    this.statusInterval = setInterval(() => this.fetchStatus(), 5000);\n    this.themeCleanup = onThemeChange(() => {\n      this.setState({ colors: getThemeColors() });\n    });\n  }\n\n  componentWillUnmount(): void {\n    if (this.statusInterval) {\n      clearInterval(this.statusInterval);\n    }\n    if (this.themeCleanup) {\n      this.themeCleanup();\n    }\n  }\n\n  private async init(): Promise<void> {\n    await Promise.all([this.fetchStatus(), this.fetchConnectionInfo()]);\n    this.setState({ loading: false });\n  }\n\n  private fetchStatus = async (): Promise<void> => {\n    try {\n      const result = (await this.props.electron?.ipcRenderer?.invoke('mcp:getStatus')) as\n        | McpStatus\n        | undefined;\n      if (result) {\n        this.setState({ status: result });\n      }\n    } catch (error) {\n      console.error('Failed to fetch MCP status:', error);\n    }\n  };\n\n  private fetchConnectionInfo = async (): Promise<void> => {\n    try {\n      const result = (await this.props.electron?.ipcRenderer?.invoke('mcp:getConnectionInfo')) as\n        | ConnectionInfo\n        | undefined;\n      if (result) {\n        this.setState({ connectionInfo: result });\n      }\n    } catch (error) {\n      console.error('Failed to fetch connection info:', error);\n    }\n  };\n\n  private handleCopyStdioConfig = (): void => {\n    const config = {\n      mcpServers: {\n        local: {\n          type: 'stdio',\n          command: 'node',\n          args: ['/path/to/local-addon-mcp-server/bin/mcp-stdio.js'],\n        },\n      },\n    };\n    navigator.clipboard.writeText(JSON.stringify(config, null, 2));\n    this.setState({ copied: true });\n    setTimeout(() => this.setState({ copied: false }), 2000);\n  };\n\n  private handleCopySseConfig = (): void => {\n    const { connectionInfo } = this.state;\n    if (connectionInfo) {\n      const config = {\n        mcpServers: {\n          local: {\n            url: `${connectionInfo.url}/mcp/sse`,\n            transport: 'sse',\n            headers: {\n              Authorization: `Bearer ${connectionInfo.authToken}`,\n            },\n          },\n        },\n      };\n      navigator.clipboard.writeText(JSON.stringify(config, null, 2));\n      this.setState({ copied: true });\n      setTimeout(() => this.setState({ copied: false }), 2000);\n    }\n  };\n\n  private handleTestConnection = async (): Promise<void> => {\n    const { connectionInfo } = this.state;\n    try {\n      const response = await fetch(`http://127.0.0.1:${connectionInfo?.port}/health`);\n      const data = await response.json();\n      if (data.status === 'ok') {\n        alert('Connection successful! MCP server is healthy.');\n      } else {\n        alert('Connection failed: Server returned unexpected response');\n      }\n    } catch (error) {\n      const message = error instanceof Error ? error.message : 'Unknown error';\n      alert(`Connection failed: ${message}`);\n    }\n  };\n\n  private handleStartStop = async (): Promise<void> => {\n    const { status } = this.state;\n    this.setState({ actionInProgress: true });\n    try {\n      if (status?.running) {\n        await this.props.electron?.ipcRenderer?.invoke('mcp:stop');\n      } else {\n        await this.props.electron?.ipcRenderer?.invoke('mcp:start');\n      }\n      await this.fetchStatus();\n      await this.fetchConnectionInfo();\n    } catch (error) {\n      const message = error instanceof Error ? error.message : 'Unknown error';\n      alert(`Action failed: ${message}`);\n    } finally {\n      this.setState({ actionInProgress: false });\n    }\n  };\n\n  private handleRestart = async (): Promise<void> => {\n    this.setState({ actionInProgress: true });\n    try {\n      await this.props.electron?.ipcRenderer?.invoke('mcp:restart');\n      await this.fetchStatus();\n      await this.fetchConnectionInfo();\n    } catch (error) {\n      const message = error instanceof Error ? error.message : 'Unknown error';\n      alert(`Restart failed: ${message}`);\n    } finally {\n      this.setState({ actionInProgress: false });\n    }\n  };\n\n  private handleRegenerateToken = async (): Promise<void> => {\n    if (\n      !confirm(\n        'Regenerate authentication token? You will need to update your AI tool configuration.'\n      )\n    ) {\n      return;\n    }\n    this.setState({ actionInProgress: true });\n    try {\n      const result = (await this.props.electron?.ipcRenderer?.invoke('mcp:regenerateToken')) as {\n        success?: boolean;\n        error?: string;\n      };\n      if (result?.success) {\n        await this.fetchConnectionInfo();\n        alert('Token regenerated successfully. Please update your AI tool configuration.');\n      } else {\n        alert(`Failed to regenerate token: ${result?.error}`);\n      }\n    } catch (error) {\n      const message = error instanceof Error ? error.message : 'Unknown error';\n      alert(`Failed to regenerate token: ${message}`);\n    } finally {\n      this.setState({ actionInProgress: false });\n    }\n  };\n\n  private getStatusColor(): string {\n    const { status, colors } = this.state;\n    if (!status) return colors.textMuted;\n    if (status.running) return colors.successText;\n    return colors.errorText;\n  }\n\n  private getStatusText(): string {\n    const { loading, status } = this.state;\n    if (loading) return 'Loading...';\n    if (!status) return 'Unknown';\n    if (status.running) return 'Running';\n    return 'Stopped';\n  }\n\n  private formatUptime(seconds: number): string {\n    if (seconds < 60) return `${Math.floor(seconds)}s`;\n    if (seconds < 3600) return `${Math.floor(seconds / 60)}m`;\n    return `${Math.floor(seconds / 3600)}h ${Math.floor((seconds % 3600) / 60)}m`;\n  }\n\n  private getTabStyle(isActive: boolean): React.CSSProperties {\n    const { colors } = this.state;\n    return {\n      padding: '8px 16px',\n      border: 'none',\n      borderBottom: isActive ? `2px solid ${colors.primary}` : '2px solid transparent',\n      backgroundColor: 'transparent',\n      color: isActive ? colors.primary : colors.textSecondary,\n      cursor: 'pointer',\n      fontSize: '14px',\n      fontWeight: isActive ? 600 : 400,\n    };\n  }\n\n  private getButtonStyle(bgColor: string, disabled?: boolean): React.CSSProperties {\n    return {\n      padding: '8px 16px',\n      backgroundColor: bgColor,\n      color: 'white',\n      border: 'none',\n      borderRadius: '4px',\n      cursor: disabled ? 'not-allowed' : 'pointer',\n      fontSize: '13px',\n      opacity: disabled ? 0.5 : 1,\n    };\n  }\n\n  private renderStatusTab(): React.ReactNode {\n    const { status, connectionInfo, actionInProgress, colors } = this.state;\n\n    return (\n      <>\n        {/* Status Section */}\n        <div style={{ marginBottom: '24px' }}>\n          <h3\n            style={{\n              fontSize: '14px',\n              fontWeight: 600,\n              marginBottom: '12px',\n              color: colors.textPrimary,\n            }}\n          >\n            Server Status\n          </h3>\n          <div\n            style={{\n              display: 'flex',\n              alignItems: 'center',\n              gap: '12px',\n              padding: '12px',\n              backgroundColor: colors.panelBgSecondary,\n              borderRadius: '6px',\n            }}\n          >\n            <span\n              style={{\n                width: '12px',\n                height: '12px',\n                borderRadius: '50%',\n                backgroundColor: this.getStatusColor(),\n              }}\n            />\n            <span style={{ fontWeight: 500, color: colors.textPrimary }}>\n              {this.getStatusText()}\n            </span>\n            {status?.running && (\n              <>\n                <span style={{ color: colors.textSecondary }}>|</span>\n                <span style={{ color: colors.textSecondary }}>Port: {status.port}</span>\n                <span style={{ color: colors.textSecondary }}>|</span>\n                <span style={{ color: colors.textSecondary }}>\n                  Uptime: {this.formatUptime(status.uptime)}\n                </span>\n              </>\n            )}\n          </div>\n        </div>\n\n        {/* Server Controls */}\n        <div style={{ marginBottom: '24px' }}>\n          <h3\n            style={{\n              fontSize: '14px',\n              fontWeight: 600,\n              marginBottom: '12px',\n              color: colors.textPrimary,\n            }}\n          >\n            Server Controls\n          </h3>\n          <div style={{ display: 'flex', gap: '12px', flexWrap: 'wrap' }}>\n            <button\n              onClick={this.handleStartStop}\n              disabled={actionInProgress}\n              style={this.getButtonStyle(\n                status?.running ? colors.errorText : colors.successText,\n                actionInProgress\n              )}\n            >\n              {status?.running ? 'Stop Server' : 'Start Server'}\n            </button>\n            <button\n              onClick={this.handleRestart}\n              disabled={actionInProgress || !status?.running}\n              style={this.getButtonStyle('#6c757d', actionInProgress || !status?.running)}\n            >\n              Restart Server\n            </button>\n            <button\n              onClick={this.handleTestConnection}\n              disabled={!status?.running}\n              style={this.getButtonStyle(\n                status?.running ? '#17a2b8' : colors.textMuted,\n                !status?.running\n              )}\n            >\n              Test Connection\n            </button>\n          </div>\n        </div>\n\n        {/* Connection Info Section */}\n        {connectionInfo && (\n          <div style={{ marginBottom: '24px' }}>\n            <h3\n              style={{\n                fontSize: '14px',\n                fontWeight: 600,\n                marginBottom: '12px',\n                color: colors.textPrimary,\n              }}\n            >\n              Connection Info\n            </h3>\n            <div\n              style={{\n                padding: '12px',\n                backgroundColor: colors.panelBgSecondary,\n                borderRadius: '6px',\n                fontFamily: 'monospace',\n                fontSize: '12px',\n                color: colors.textPrimary,\n              }}\n            >\n              <div style={{ marginBottom: '8px' }}>\n                <strong>URL:</strong> {connectionInfo.url}\n              </div>\n              <div style={{ marginBottom: '8px' }}>\n                <strong>stdio script:</strong> bin/mcp-stdio.js\n              </div>\n              <div style={{ marginBottom: '8px' }}>\n                <strong>Version:</strong> {connectionInfo.version}\n              </div>\n              <div>\n                <strong>Tools:</strong> {connectionInfo.tools.join(', ')}\n              </div>\n            </div>\n          </div>\n        )}\n\n        {/* Security Section */}\n        <div style={{ marginBottom: '24px' }}>\n          <h3\n            style={{\n              fontSize: '14px',\n              fontWeight: 600,\n              marginBottom: '12px',\n              color: colors.textPrimary,\n            }}\n          >\n            Security\n          </h3>\n          <div style={{ display: 'flex', gap: '12px', flexWrap: 'wrap' }}>\n            <button\n              onClick={this.handleRegenerateToken}\n              disabled={actionInProgress || !status?.running}\n              style={{\n                padding: '8px 16px',\n                backgroundColor: '#ffc107',\n                color: '#212529',\n                border: 'none',\n                borderRadius: '4px',\n                cursor: actionInProgress || !status?.running ? 'not-allowed' : 'pointer',\n                fontSize: '13px',\n                opacity: actionInProgress || !status?.running ? 0.5 : 1,\n              }}\n            >\n              Regenerate Token\n            </button>\n          </div>\n          <p style={{ fontSize: '12px', color: colors.textSecondary, marginTop: '8px' }}>\n            Regenerating the token will require you to update your AI tool configuration.\n          </p>\n        </div>\n      </>\n    );\n  }\n\n  private renderSetupTab(): React.ReactNode {\n    const { connectionInfo, copied, colors } = this.state;\n\n    return (\n      <>\n        {/* Claude Code Setup */}\n        <div style={{ marginBottom: '24px' }}>\n          <h3\n            style={{\n              fontSize: '14px',\n              fontWeight: 600,\n              marginBottom: '12px',\n              color: colors.textPrimary,\n            }}\n          >\n            Claude Code (Recommended)\n          </h3>\n          <div\n            style={{\n              padding: '16px',\n              backgroundColor: colors.panelBgSecondary,\n              borderRadius: '6px',\n            }}\n          >\n            <p style={{ margin: '0 0 12px 0', fontSize: '13px', color: colors.textPrimary }}>\n              Add the following to your{' '}\n              <code\n                style={{\n                  backgroundColor: colors.panelBgCode,\n                  padding: '2px 4px',\n                  borderRadius: '3px',\n                  color: '#f8f8f2',\n                }}\n              >\n                ~/.claude.json\n              </code>{' '}\n              file:\n            </p>\n            <pre\n              style={{\n                backgroundColor: colors.panelBgCode,\n                color: '#f8f8f2',\n                padding: '12px',\n                borderRadius: '4px',\n                fontSize: '11px',\n                overflow: 'auto',\n                margin: '0 0 12px 0',\n              }}\n            >\n              {`{\n  \"mcpServers\": {\n    \"local\": {\n      \"type\": \"stdio\",\n      \"command\": \"node\",\n      \"args\": [\"/path/to/local-addon-mcp-server/bin/mcp-stdio.js\"]\n    }\n  }\n}`}\n            </pre>\n            <button\n              onClick={this.handleCopyStdioConfig}\n              style={this.getButtonStyle(colors.primary)}\n            >\n              {copied ? 'Copied!' : 'Copy Config'}\n            </button>\n          </div>\n        </div>\n\n        {/* Claude.ai / ChatGPT Setup */}\n        <div style={{ marginBottom: '24px' }}>\n          <h3\n            style={{\n              fontSize: '14px',\n              fontWeight: 600,\n              marginBottom: '12px',\n              color: colors.textPrimary,\n            }}\n          >\n            Claude.ai / ChatGPT / Other AI Tools\n          </h3>\n          <div\n            style={{\n              padding: '16px',\n              backgroundColor: colors.panelBgSecondary,\n              borderRadius: '6px',\n            }}\n          >\n            <p style={{ margin: '0 0 12px 0', fontSize: '13px', color: colors.textPrimary }}>\n              For tools that support SSE transport, use this configuration:\n            </p>\n            <pre\n              style={{\n                backgroundColor: colors.panelBgCode,\n                color: '#f8f8f2',\n                padding: '12px',\n                borderRadius: '4px',\n                fontSize: '11px',\n                overflow: 'auto',\n                margin: '0 0 12px 0',\n              }}\n            >\n              {connectionInfo\n                ? `{\n  \"mcpServers\": {\n    \"local\": {\n      \"url\": \"${connectionInfo.url}/mcp/sse\",\n      \"transport\": \"sse\",\n      \"headers\": {\n        \"Authorization\": \"Bearer ${connectionInfo.authToken.substring(0, 20)}...\"\n      }\n    }\n  }\n}`\n                : 'Server not running'}\n            </pre>\n            <button\n              onClick={this.handleCopySseConfig}\n              disabled={!connectionInfo}\n              style={this.getButtonStyle(\n                connectionInfo ? colors.primary : colors.textMuted,\n                !connectionInfo\n              )}\n            >\n              {copied ? 'Copied!' : 'Copy SSE Config'}\n            </button>\n          </div>\n        </div>\n\n        {/* Available Commands */}\n        <div style={{ marginBottom: '24px' }}>\n          <h3\n            style={{\n              fontSize: '14px',\n              fontWeight: 600,\n              marginBottom: '12px',\n              color: colors.textPrimary,\n            }}\n          >\n            Example Commands\n          </h3>\n          <div\n            style={{\n              padding: '16px',\n              backgroundColor: colors.infoBg,\n              borderRadius: '6px',\n              border: `1px solid ${colors.infoBorder}`,\n            }}\n          >\n            <p\n              style={{\n                margin: '0 0 8px 0',\n                fontSize: '13px',\n                fontWeight: 500,\n                color: colors.infoText,\n              }}\n            >\n              Try saying to your AI assistant:\n            </p>\n            <ul\n              style={{ margin: 0, paddingLeft: '20px', fontSize: '13px', color: colors.infoText }}\n            >\n              <li>&quot;List my Local sites&quot;</li>\n              <li>&quot;Start the blog site&quot;</li>\n              <li>&quot;Create a new site called test-project&quot;</li>\n              <li>&quot;Run wp plugin list on my-site&quot;</li>\n              <li>&quot;Stop all running sites&quot;</li>\n              <li>&quot;What plugins are installed on my-site?&quot;</li>\n            </ul>\n          </div>\n        </div>\n      </>\n    );\n  }\n\n  render(): React.ReactNode {\n    const { activeTab, colors } = this.state;\n\n    return (\n      <div style={{ padding: '20px', color: colors.textPrimary }}>\n        <h2\n          style={{\n            marginBottom: '20px',\n            fontSize: '18px',\n            fontWeight: 600,\n            color: colors.textPrimary,\n          }}\n        >\n          MCP Server\n        </h2>\n\n        <p style={{ color: colors.textSecondary, marginBottom: '24px' }}>\n          The MCP (Model Context Protocol) server enables AI tools like Claude Code to control your\n          Local sites.\n        </p>\n\n        {/* Tab Navigation */}\n        <div style={{ borderBottom: `1px solid ${colors.border}`, marginBottom: '24px' }}>\n          <button\n            style={this.getTabStyle(activeTab === 'status')}\n            onClick={() => this.setState({ activeTab: 'status' })}\n          >\n            Status &amp; Controls\n          </button>\n          <button\n            style={this.getTabStyle(activeTab === 'setup')}\n            onClick={() => this.setState({ activeTab: 'setup' })}\n          >\n            AI Tool Setup\n          </button>\n        </div>\n\n        {activeTab === 'status' && this.renderStatusTab()}\n        {activeTab === 'setup' && this.renderSetupTab()}\n      </div>\n    );\n  }\n}\n\n// Export as a constructor function that Local can instantiate\nmodule.exports = function McpServerRenderer(context: { hooks: unknown; electron: unknown }): void {\n  // CLI-only mode - preferences panel disabled\n  // To enable MCP preferences UI, uncomment the following:\n  /*\n  const { hooks, electron } = context as {\n    hooks: { addFilter: (name: string, callback: (items: unknown[]) => unknown[]) => void };\n    electron: McpPreferencesPanelProps['electron'];\n  };\n\n  console.log('[MCP Server] Renderer loading...');\n\n  // Add MCP Server to preferences menu\n  hooks.addFilter('preferencesMenuItems', (menuItems: unknown[]) => {\n    console.log('[MCP Server] Adding MCP Server to preferences');\n\n    (menuItems as Array<Record<string, unknown>>).push({\n      path: 'mcp-server',\n      displayName: 'MCP Server',\n      sections: () => <McpPreferencesPanel electron={electron} />,\n      onApply: () => {\n        // No apply action needed for read-only display\n      },\n    });\n\n    return menuItems;\n  });\n\n  console.log('[MCP Server] Renderer setup complete');\n  */\n\n  // Silence unused variable warnings\n  void context;\n};\n"]}
@@ -0,0 +1,73 @@
1
+ {
2
+ "name": "@local-labs-jpollock/local-addon-cli",
3
+ "productName": "Local CLI",
4
+ "version": "0.0.1",
5
+ "description": "Command-line interface for Local WordPress development",
6
+ "main": "lib/main/index.js",
7
+ "renderer": "lib/renderer/index.js",
8
+ "bin": {
9
+ "local-mcp": "./bin/mcp-stdio.js"
10
+ },
11
+ "scripts": {
12
+ "build": "npm run clean && tsc",
13
+ "watch": "tsc --watch",
14
+ "clean": "rm -rf lib coverage",
15
+ "lint": "eslint src --ext .ts,.tsx --max-warnings=0",
16
+ "lint:fix": "eslint src --ext .ts,.tsx --fix",
17
+ "format": "prettier --write \"src/**/*.{ts,tsx}\"",
18
+ "format:check": "prettier --check \"src/**/*.{ts,tsx}\"",
19
+ "typecheck": "tsc --noEmit",
20
+ "test": "jest",
21
+ "test:watch": "jest --watch",
22
+ "test:coverage": "jest --coverage"
23
+ },
24
+ "keywords": [
25
+ "local",
26
+ "addon",
27
+ "cli",
28
+ "wordpress",
29
+ "development"
30
+ ],
31
+ "author": "Jeremy Pollock <jpollock911@gmail.com>",
32
+ "license": "MIT",
33
+ "repository": {
34
+ "type": "git",
35
+ "url": "https://github.com/jpollock/local-addon-cli.git",
36
+ "directory": "packages/addon"
37
+ },
38
+ "engines": {
39
+ "node": ">=18.0.0"
40
+ },
41
+ "files": [
42
+ "lib/",
43
+ "bin/",
44
+ "README.md",
45
+ "LICENSE"
46
+ ],
47
+ "devDependencies": {
48
+ "@getflywheel/local": "^9.0.0",
49
+ "@types/fs-extra": "^11.0.4",
50
+ "@types/jest": "^29.5.12",
51
+ "@types/node": "^20.0.0",
52
+ "@types/react": "^19.2.10",
53
+ "@typescript-eslint/eslint-plugin": "^7.0.0",
54
+ "@typescript-eslint/parser": "^7.0.0",
55
+ "eslint": "^8.57.0",
56
+ "eslint-config-prettier": "^9.1.0",
57
+ "eslint-plugin-react": "^7.34.0",
58
+ "eslint-plugin-react-hooks": "^4.6.0",
59
+ "graphql-tag": "^2.12.6",
60
+ "jest": "^29.7.0",
61
+ "prettier": "^3.2.5",
62
+ "ts-jest": "^29.1.2",
63
+ "typescript": "^5.0.0"
64
+ },
65
+ "localAddon": {
66
+ "minimumLocalVersion": "9.0.0",
67
+ "type": "addon",
68
+ "category": "developer-tools"
69
+ },
70
+ "dependencies": {
71
+ "node-stream-zip": "^1.15.0"
72
+ }
73
+ }