@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.
- package/addon-dist/bin/mcp-stdio.js +2808 -0
- package/addon-dist/lib/common/constants.d.ts +22 -0
- package/addon-dist/lib/common/constants.js +26 -0
- package/addon-dist/lib/common/theme.d.ts +68 -0
- package/addon-dist/lib/common/theme.js +126 -0
- package/addon-dist/lib/common/types.d.ts +298 -0
- package/addon-dist/lib/common/types.js +6 -0
- package/addon-dist/lib/main/config/ConnectionInfo.d.ts +25 -0
- package/addon-dist/lib/main/config/ConnectionInfo.js +82 -0
- package/addon-dist/lib/main/index.d.ts +12 -0
- package/addon-dist/lib/main/index.js +3322 -0
- package/addon-dist/lib/main/mcp/McpAuth.d.ts +37 -0
- package/addon-dist/lib/main/mcp/McpAuth.js +87 -0
- package/addon-dist/lib/main/mcp/McpServer.d.ts +67 -0
- package/addon-dist/lib/main/mcp/McpServer.js +343 -0
- package/addon-dist/lib/main/mcp/tools/changePhpVersion.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/changePhpVersion.js +81 -0
- package/addon-dist/lib/main/mcp/tools/cloneSite.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/cloneSite.js +66 -0
- package/addon-dist/lib/main/mcp/tools/createSite.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/createSite.js +137 -0
- package/addon-dist/lib/main/mcp/tools/deleteSite.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/deleteSite.js +72 -0
- package/addon-dist/lib/main/mcp/tools/exportDatabase.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/exportDatabase.js +72 -0
- package/addon-dist/lib/main/mcp/tools/exportSite.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/exportSite.js +103 -0
- package/addon-dist/lib/main/mcp/tools/getLocalInfo.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/getLocalInfo.js +72 -0
- package/addon-dist/lib/main/mcp/tools/getSite.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/getSite.js +68 -0
- package/addon-dist/lib/main/mcp/tools/getSiteLogs.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/getSiteLogs.js +149 -0
- package/addon-dist/lib/main/mcp/tools/helpers.d.ts +59 -0
- package/addon-dist/lib/main/mcp/tools/helpers.js +179 -0
- package/addon-dist/lib/main/mcp/tools/importDatabase.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/importDatabase.js +109 -0
- package/addon-dist/lib/main/mcp/tools/importSite.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/importSite.js +149 -0
- package/addon-dist/lib/main/mcp/tools/index.d.ts +26 -0
- package/addon-dist/lib/main/mcp/tools/index.js +117 -0
- package/addon-dist/lib/main/mcp/tools/listBlueprints.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/listBlueprints.js +54 -0
- package/addon-dist/lib/main/mcp/tools/listServices.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/listServices.js +112 -0
- package/addon-dist/lib/main/mcp/tools/listSites.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/listSites.js +62 -0
- package/addon-dist/lib/main/mcp/tools/openAdminer.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/openAdminer.js +59 -0
- package/addon-dist/lib/main/mcp/tools/openSite.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/openSite.js +62 -0
- package/addon-dist/lib/main/mcp/tools/renameSite.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/renameSite.js +70 -0
- package/addon-dist/lib/main/mcp/tools/restartSite.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/restartSite.js +56 -0
- package/addon-dist/lib/main/mcp/tools/saveBlueprint.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/saveBlueprint.js +89 -0
- package/addon-dist/lib/main/mcp/tools/startSite.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/startSite.js +54 -0
- package/addon-dist/lib/main/mcp/tools/stopSite.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/stopSite.js +54 -0
- package/addon-dist/lib/main/mcp/tools/toggleXdebug.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/toggleXdebug.js +69 -0
- package/addon-dist/lib/main/mcp/tools/trustSsl.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/trustSsl.js +59 -0
- package/addon-dist/lib/main/mcp/tools/wpCli.d.ts +7 -0
- package/addon-dist/lib/main/mcp/tools/wpCli.js +110 -0
- package/addon-dist/lib/main.d.ts +1 -0
- package/addon-dist/lib/main.js +10 -0
- package/addon-dist/lib/renderer/index.d.ts +7 -0
- package/addon-dist/lib/renderer/index.js +479 -0
- package/addon-dist/package.json +73 -0
- package/bin/lwp.js +10 -0
- package/lib/bootstrap/index.d.ts +98 -0
- package/lib/bootstrap/index.js +493 -0
- package/lib/bootstrap/paths.d.ts +28 -0
- package/lib/bootstrap/paths.js +96 -0
- package/lib/client/GraphQLClient.d.ts +38 -0
- package/lib/client/GraphQLClient.js +71 -0
- package/lib/client/index.d.ts +4 -0
- package/lib/client/index.js +10 -0
- package/lib/formatters/index.d.ts +75 -0
- package/lib/formatters/index.js +139 -0
- package/lib/index.d.ts +8 -0
- package/lib/index.js +1173 -0
- package/package.json +72 -0
|
@@ -0,0 +1,3322 @@
|
|
|
1
|
+
"use strict";
|
|
2
|
+
/**
|
|
3
|
+
* CLI Bridge Addon - Main Process Entry Point
|
|
4
|
+
*
|
|
5
|
+
* This addon extends Local's capabilities:
|
|
6
|
+
* - GraphQL mutations for deleteSite, wpCli (for local-cli)
|
|
7
|
+
* - MCP Server for AI tool integration (Claude Code, ChatGPT, etc.)
|
|
8
|
+
*/
|
|
9
|
+
var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
|
10
|
+
if (k2 === undefined) k2 = k;
|
|
11
|
+
var desc = Object.getOwnPropertyDescriptor(m, k);
|
|
12
|
+
if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
|
13
|
+
desc = { enumerable: true, get: function() { return m[k]; } };
|
|
14
|
+
}
|
|
15
|
+
Object.defineProperty(o, k2, desc);
|
|
16
|
+
}) : (function(o, m, k, k2) {
|
|
17
|
+
if (k2 === undefined) k2 = k;
|
|
18
|
+
o[k2] = m[k];
|
|
19
|
+
}));
|
|
20
|
+
var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
|
21
|
+
Object.defineProperty(o, "default", { enumerable: true, value: v });
|
|
22
|
+
}) : function(o, v) {
|
|
23
|
+
o["default"] = v;
|
|
24
|
+
});
|
|
25
|
+
var __importStar = (this && this.__importStar) || (function () {
|
|
26
|
+
var ownKeys = function(o) {
|
|
27
|
+
ownKeys = Object.getOwnPropertyNames || function (o) {
|
|
28
|
+
var ar = [];
|
|
29
|
+
for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
|
|
30
|
+
return ar;
|
|
31
|
+
};
|
|
32
|
+
return ownKeys(o);
|
|
33
|
+
};
|
|
34
|
+
return function (mod) {
|
|
35
|
+
if (mod && mod.__esModule) return mod;
|
|
36
|
+
var result = {};
|
|
37
|
+
if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
|
|
38
|
+
__setModuleDefault(result, mod);
|
|
39
|
+
return result;
|
|
40
|
+
};
|
|
41
|
+
})();
|
|
42
|
+
var __importDefault = (this && this.__importDefault) || function (mod) {
|
|
43
|
+
return (mod && mod.__esModule) ? mod : { "default": mod };
|
|
44
|
+
};
|
|
45
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
46
|
+
exports.default = default_1;
|
|
47
|
+
const LocalMain = __importStar(require("@getflywheel/local/main"));
|
|
48
|
+
const electron_1 = require("electron");
|
|
49
|
+
const graphql_tag_1 = __importDefault(require("graphql-tag"));
|
|
50
|
+
const McpServer_1 = require("./mcp/McpServer");
|
|
51
|
+
const constants_1 = require("../common/constants");
|
|
52
|
+
const ADDON_NAME = 'MCP Server';
|
|
53
|
+
let mcpServer = null;
|
|
54
|
+
/**
|
|
55
|
+
* GraphQL type definitions for CLI Bridge
|
|
56
|
+
*/
|
|
57
|
+
const typeDefs = (0, graphql_tag_1.default) `
|
|
58
|
+
input DeleteSiteInput {
|
|
59
|
+
"The site ID to delete"
|
|
60
|
+
id: ID!
|
|
61
|
+
"Whether to move site files to trash (true) or just remove from Local (false)"
|
|
62
|
+
trashFiles: Boolean = true
|
|
63
|
+
"Whether to update the hosts file"
|
|
64
|
+
updateHosts: Boolean = true
|
|
65
|
+
}
|
|
66
|
+
|
|
67
|
+
type DeleteSiteResult {
|
|
68
|
+
"Whether the deletion was successful"
|
|
69
|
+
success: Boolean!
|
|
70
|
+
"Error message if deletion failed"
|
|
71
|
+
error: String
|
|
72
|
+
"The ID of the deleted site"
|
|
73
|
+
siteId: ID
|
|
74
|
+
}
|
|
75
|
+
|
|
76
|
+
input WpCliInput {
|
|
77
|
+
"The site ID to run WP-CLI against"
|
|
78
|
+
siteId: ID!
|
|
79
|
+
"WP-CLI command and arguments (e.g., ['plugin', 'list', '--format=json'])"
|
|
80
|
+
args: [String!]!
|
|
81
|
+
"Skip loading plugins (default: true)"
|
|
82
|
+
skipPlugins: Boolean = true
|
|
83
|
+
"Skip loading themes (default: true)"
|
|
84
|
+
skipThemes: Boolean = true
|
|
85
|
+
}
|
|
86
|
+
|
|
87
|
+
type WpCliResult {
|
|
88
|
+
"Whether the command executed successfully"
|
|
89
|
+
success: Boolean!
|
|
90
|
+
"Command output (stdout)"
|
|
91
|
+
output: String
|
|
92
|
+
"Error message if command failed"
|
|
93
|
+
error: String
|
|
94
|
+
}
|
|
95
|
+
|
|
96
|
+
input CreateSiteInput {
|
|
97
|
+
"Site name (required)"
|
|
98
|
+
name: String!
|
|
99
|
+
"PHP version (e.g., '8.2.10'). Uses Local default if not specified."
|
|
100
|
+
phpVersion: String
|
|
101
|
+
"Web server type"
|
|
102
|
+
webServer: String
|
|
103
|
+
"Database type"
|
|
104
|
+
database: String
|
|
105
|
+
"WordPress admin username (default: admin)"
|
|
106
|
+
wpAdminUsername: String
|
|
107
|
+
"WordPress admin password (default: password)"
|
|
108
|
+
wpAdminPassword: String
|
|
109
|
+
"WordPress admin email (default: admin@local.test)"
|
|
110
|
+
wpAdminEmail: String
|
|
111
|
+
"Blueprint name to create site from. Use list_blueprints to see available blueprints."
|
|
112
|
+
blueprint: String
|
|
113
|
+
}
|
|
114
|
+
|
|
115
|
+
type CreateSiteResult {
|
|
116
|
+
"Whether site creation was initiated successfully"
|
|
117
|
+
success: Boolean!
|
|
118
|
+
"Error message if creation failed"
|
|
119
|
+
error: String
|
|
120
|
+
"The created site ID"
|
|
121
|
+
siteId: ID
|
|
122
|
+
"The site name"
|
|
123
|
+
siteName: String
|
|
124
|
+
"The site domain"
|
|
125
|
+
siteDomain: String
|
|
126
|
+
}
|
|
127
|
+
|
|
128
|
+
input OpenSiteInput {
|
|
129
|
+
"The site ID to open"
|
|
130
|
+
siteId: ID!
|
|
131
|
+
"Path to open (default: /, use /wp-admin for admin)"
|
|
132
|
+
path: String = "/"
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
type OpenSiteResult {
|
|
136
|
+
"Whether the site was opened successfully"
|
|
137
|
+
success: Boolean!
|
|
138
|
+
"Error message if failed"
|
|
139
|
+
error: String
|
|
140
|
+
"The URL that was opened"
|
|
141
|
+
url: String
|
|
142
|
+
}
|
|
143
|
+
|
|
144
|
+
input CloneSiteInput {
|
|
145
|
+
"The site ID to clone"
|
|
146
|
+
siteId: ID!
|
|
147
|
+
"Name for the cloned site"
|
|
148
|
+
newName: String!
|
|
149
|
+
}
|
|
150
|
+
|
|
151
|
+
type CloneSiteResult {
|
|
152
|
+
"Whether cloning was successful"
|
|
153
|
+
success: Boolean!
|
|
154
|
+
"Error message if failed"
|
|
155
|
+
error: String
|
|
156
|
+
"The new site ID"
|
|
157
|
+
newSiteId: ID
|
|
158
|
+
"The new site name"
|
|
159
|
+
newSiteName: String
|
|
160
|
+
"The new site domain"
|
|
161
|
+
newSiteDomain: String
|
|
162
|
+
}
|
|
163
|
+
|
|
164
|
+
input ExportSiteInput {
|
|
165
|
+
"The site ID to export"
|
|
166
|
+
siteId: ID!
|
|
167
|
+
"Output directory path (default: ~/Downloads)"
|
|
168
|
+
outputPath: String
|
|
169
|
+
}
|
|
170
|
+
|
|
171
|
+
type ExportSiteResult {
|
|
172
|
+
"Whether export was successful"
|
|
173
|
+
success: Boolean!
|
|
174
|
+
"Error message if failed"
|
|
175
|
+
error: String
|
|
176
|
+
"Path to the exported zip file"
|
|
177
|
+
exportPath: String
|
|
178
|
+
}
|
|
179
|
+
|
|
180
|
+
type Blueprint {
|
|
181
|
+
"Blueprint name"
|
|
182
|
+
name: String!
|
|
183
|
+
"Last modified date"
|
|
184
|
+
lastModified: String
|
|
185
|
+
"PHP version"
|
|
186
|
+
phpVersion: String
|
|
187
|
+
"Web server type"
|
|
188
|
+
webServer: String
|
|
189
|
+
"Database type"
|
|
190
|
+
database: String
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
type BlueprintsResult {
|
|
194
|
+
"Whether query was successful"
|
|
195
|
+
success: Boolean!
|
|
196
|
+
"Error message if failed"
|
|
197
|
+
error: String
|
|
198
|
+
"List of blueprints"
|
|
199
|
+
blueprints: [Blueprint!]
|
|
200
|
+
}
|
|
201
|
+
|
|
202
|
+
input SaveBlueprintInput {
|
|
203
|
+
"The site ID to save as blueprint"
|
|
204
|
+
siteId: ID!
|
|
205
|
+
"Name for the blueprint"
|
|
206
|
+
name: String!
|
|
207
|
+
}
|
|
208
|
+
|
|
209
|
+
type SaveBlueprintResult {
|
|
210
|
+
"Whether save was successful"
|
|
211
|
+
success: Boolean!
|
|
212
|
+
"Error message if failed"
|
|
213
|
+
error: String
|
|
214
|
+
"The blueprint name"
|
|
215
|
+
blueprintName: String
|
|
216
|
+
}
|
|
217
|
+
|
|
218
|
+
# Phase 8: WordPress Development Tools
|
|
219
|
+
input ExportDatabaseInput {
|
|
220
|
+
"The site ID"
|
|
221
|
+
siteId: ID!
|
|
222
|
+
"Output file path (optional, defaults to ~/Downloads/<site-name>.sql)"
|
|
223
|
+
outputPath: String
|
|
224
|
+
}
|
|
225
|
+
|
|
226
|
+
type ExportDatabaseResult {
|
|
227
|
+
"Whether export was successful"
|
|
228
|
+
success: Boolean!
|
|
229
|
+
"Error message if failed"
|
|
230
|
+
error: String
|
|
231
|
+
"Path to the exported SQL file"
|
|
232
|
+
outputPath: String
|
|
233
|
+
}
|
|
234
|
+
|
|
235
|
+
input ImportDatabaseInput {
|
|
236
|
+
"The site ID"
|
|
237
|
+
siteId: ID!
|
|
238
|
+
"Path to the SQL file to import"
|
|
239
|
+
sqlPath: String!
|
|
240
|
+
}
|
|
241
|
+
|
|
242
|
+
type ImportDatabaseResult {
|
|
243
|
+
"Whether import was successful"
|
|
244
|
+
success: Boolean!
|
|
245
|
+
"Error message if failed"
|
|
246
|
+
error: String
|
|
247
|
+
}
|
|
248
|
+
|
|
249
|
+
input OpenAdminerInput {
|
|
250
|
+
"The site ID"
|
|
251
|
+
siteId: ID!
|
|
252
|
+
}
|
|
253
|
+
|
|
254
|
+
type OpenAdminerResult {
|
|
255
|
+
"Whether opening was successful"
|
|
256
|
+
success: Boolean!
|
|
257
|
+
"Error message if failed"
|
|
258
|
+
error: String
|
|
259
|
+
}
|
|
260
|
+
|
|
261
|
+
input TrustSslInput {
|
|
262
|
+
"The site ID"
|
|
263
|
+
siteId: ID!
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
type TrustSslResult {
|
|
267
|
+
"Whether trust was successful"
|
|
268
|
+
success: Boolean!
|
|
269
|
+
"Error message if failed"
|
|
270
|
+
error: String
|
|
271
|
+
}
|
|
272
|
+
|
|
273
|
+
input McpRenameSiteInput {
|
|
274
|
+
"The site ID"
|
|
275
|
+
siteId: ID!
|
|
276
|
+
"New name for the site"
|
|
277
|
+
newName: String!
|
|
278
|
+
}
|
|
279
|
+
|
|
280
|
+
type McpRenameSiteResult {
|
|
281
|
+
"Whether rename was successful"
|
|
282
|
+
success: Boolean!
|
|
283
|
+
"Error message if failed"
|
|
284
|
+
error: String
|
|
285
|
+
"The new name"
|
|
286
|
+
newName: String
|
|
287
|
+
}
|
|
288
|
+
|
|
289
|
+
input ChangePhpVersionInput {
|
|
290
|
+
"The site ID"
|
|
291
|
+
siteId: ID!
|
|
292
|
+
"Target PHP version"
|
|
293
|
+
phpVersion: String!
|
|
294
|
+
}
|
|
295
|
+
|
|
296
|
+
type ChangePhpVersionResult {
|
|
297
|
+
"Whether change was successful"
|
|
298
|
+
success: Boolean!
|
|
299
|
+
"Error message if failed"
|
|
300
|
+
error: String
|
|
301
|
+
"The new PHP version"
|
|
302
|
+
phpVersion: String
|
|
303
|
+
}
|
|
304
|
+
|
|
305
|
+
input ImportSiteInput {
|
|
306
|
+
"Path to the zip file to import"
|
|
307
|
+
zipPath: String!
|
|
308
|
+
"Name for the imported site (optional)"
|
|
309
|
+
siteName: String
|
|
310
|
+
}
|
|
311
|
+
|
|
312
|
+
type ImportSiteResult {
|
|
313
|
+
"Whether import was successful"
|
|
314
|
+
success: Boolean!
|
|
315
|
+
"Error message if failed"
|
|
316
|
+
error: String
|
|
317
|
+
"The imported site ID"
|
|
318
|
+
siteId: ID
|
|
319
|
+
"The imported site name"
|
|
320
|
+
siteName: String
|
|
321
|
+
}
|
|
322
|
+
|
|
323
|
+
# Phase 9: Site Configuration & Dev Tools
|
|
324
|
+
input ToggleXdebugInput {
|
|
325
|
+
"The site ID"
|
|
326
|
+
siteId: ID!
|
|
327
|
+
"Whether to enable or disable Xdebug"
|
|
328
|
+
enabled: Boolean!
|
|
329
|
+
}
|
|
330
|
+
|
|
331
|
+
type ToggleXdebugResult {
|
|
332
|
+
"Whether toggle was successful"
|
|
333
|
+
success: Boolean!
|
|
334
|
+
"Error message if failed"
|
|
335
|
+
error: String
|
|
336
|
+
"Current Xdebug state"
|
|
337
|
+
enabled: Boolean
|
|
338
|
+
}
|
|
339
|
+
|
|
340
|
+
input GetSiteLogsInput {
|
|
341
|
+
"The site ID"
|
|
342
|
+
siteId: ID!
|
|
343
|
+
"Type of logs to retrieve (php, nginx, mysql, all)"
|
|
344
|
+
logType: String = "php"
|
|
345
|
+
"Number of lines to return"
|
|
346
|
+
lines: Int = 100
|
|
347
|
+
}
|
|
348
|
+
|
|
349
|
+
type LogEntry {
|
|
350
|
+
"Log type"
|
|
351
|
+
type: String!
|
|
352
|
+
"Log content"
|
|
353
|
+
content: String!
|
|
354
|
+
"Log file path"
|
|
355
|
+
path: String!
|
|
356
|
+
}
|
|
357
|
+
|
|
358
|
+
type GetSiteLogsResult {
|
|
359
|
+
"Whether retrieval was successful"
|
|
360
|
+
success: Boolean!
|
|
361
|
+
"Error message if failed"
|
|
362
|
+
error: String
|
|
363
|
+
"Log entries"
|
|
364
|
+
logs: [LogEntry!]
|
|
365
|
+
}
|
|
366
|
+
|
|
367
|
+
type ServiceInfo {
|
|
368
|
+
"Service role (php, database, webserver)"
|
|
369
|
+
role: String!
|
|
370
|
+
"Service name"
|
|
371
|
+
name: String!
|
|
372
|
+
"Service version"
|
|
373
|
+
version: String!
|
|
374
|
+
}
|
|
375
|
+
|
|
376
|
+
type ListServicesResult {
|
|
377
|
+
"Whether listing was successful"
|
|
378
|
+
success: Boolean!
|
|
379
|
+
"Error message if failed"
|
|
380
|
+
error: String
|
|
381
|
+
"Available services"
|
|
382
|
+
services: [ServiceInfo!]
|
|
383
|
+
}
|
|
384
|
+
|
|
385
|
+
extend type Mutation {
|
|
386
|
+
"Create a new WordPress site with full WordPress installation"
|
|
387
|
+
createSite(input: CreateSiteInput!): CreateSiteResult!
|
|
388
|
+
|
|
389
|
+
"Delete a site from Local"
|
|
390
|
+
deleteSite(input: DeleteSiteInput!): DeleteSiteResult!
|
|
391
|
+
|
|
392
|
+
"Delete multiple sites from Local"
|
|
393
|
+
deleteSites(ids: [ID!]!, trashFiles: Boolean = true): DeleteSiteResult!
|
|
394
|
+
|
|
395
|
+
"Run a WP-CLI command against a site"
|
|
396
|
+
wpCli(input: WpCliInput!): WpCliResult!
|
|
397
|
+
|
|
398
|
+
"Open a site in the default browser"
|
|
399
|
+
openSite(input: OpenSiteInput!): OpenSiteResult!
|
|
400
|
+
|
|
401
|
+
"Clone an existing site"
|
|
402
|
+
cloneSite(input: CloneSiteInput!): CloneSiteResult!
|
|
403
|
+
|
|
404
|
+
"Export a site to a zip file"
|
|
405
|
+
exportSite(input: ExportSiteInput!): ExportSiteResult!
|
|
406
|
+
|
|
407
|
+
"Save a site as a blueprint"
|
|
408
|
+
saveBlueprint(input: SaveBlueprintInput!): SaveBlueprintResult!
|
|
409
|
+
|
|
410
|
+
# Phase 8: WordPress Development Tools
|
|
411
|
+
"Export site database to SQL file"
|
|
412
|
+
exportDatabase(input: ExportDatabaseInput!): ExportDatabaseResult!
|
|
413
|
+
|
|
414
|
+
"Import SQL file into site database"
|
|
415
|
+
importDatabase(input: ImportDatabaseInput!): ImportDatabaseResult!
|
|
416
|
+
|
|
417
|
+
"Open Adminer database management UI"
|
|
418
|
+
openAdminer(input: OpenAdminerInput!): OpenAdminerResult!
|
|
419
|
+
|
|
420
|
+
"Trust site SSL certificate"
|
|
421
|
+
trustSsl(input: TrustSslInput!): TrustSslResult!
|
|
422
|
+
|
|
423
|
+
"Rename a site (MCP version)"
|
|
424
|
+
mcpRenameSite(input: McpRenameSiteInput!): McpRenameSiteResult!
|
|
425
|
+
|
|
426
|
+
"Change site PHP version"
|
|
427
|
+
changePhpVersion(input: ChangePhpVersionInput!): ChangePhpVersionResult!
|
|
428
|
+
|
|
429
|
+
"Import site from zip file"
|
|
430
|
+
importSite(input: ImportSiteInput!): ImportSiteResult!
|
|
431
|
+
|
|
432
|
+
# Phase 9: Site Configuration & Dev Tools
|
|
433
|
+
"Toggle Xdebug for a site"
|
|
434
|
+
toggleXdebug(input: ToggleXdebugInput!): ToggleXdebugResult!
|
|
435
|
+
|
|
436
|
+
"Get site log files"
|
|
437
|
+
getSiteLogs(input: GetSiteLogsInput!): GetSiteLogsResult!
|
|
438
|
+
}
|
|
439
|
+
|
|
440
|
+
extend type Query {
|
|
441
|
+
"Run a WP-CLI command against a site (read-only operations)"
|
|
442
|
+
wpCliQuery(input: WpCliInput!): WpCliResult!
|
|
443
|
+
|
|
444
|
+
"List all available blueprints"
|
|
445
|
+
blueprints: BlueprintsResult!
|
|
446
|
+
|
|
447
|
+
"List available service versions"
|
|
448
|
+
listServices(type: String): ListServicesResult!
|
|
449
|
+
|
|
450
|
+
# Phase 11: WP Engine Connect
|
|
451
|
+
"Check WP Engine authentication status"
|
|
452
|
+
wpeStatus: WpeAuthStatus!
|
|
453
|
+
|
|
454
|
+
"List all sites from WP Engine account"
|
|
455
|
+
listWpeSites(accountId: String): ListWpeSitesResult!
|
|
456
|
+
|
|
457
|
+
# Phase 11b: Site Linking
|
|
458
|
+
"Get WP Engine connection details for a local site"
|
|
459
|
+
getWpeLink(siteId: ID!): GetWpeLinkResult!
|
|
460
|
+
}
|
|
461
|
+
|
|
462
|
+
# Phase 11: WP Engine Connect Types
|
|
463
|
+
type WpeAuthStatus {
|
|
464
|
+
"Whether authenticated with WP Engine"
|
|
465
|
+
authenticated: Boolean!
|
|
466
|
+
"User email if authenticated"
|
|
467
|
+
email: String
|
|
468
|
+
"Account ID if authenticated"
|
|
469
|
+
accountId: String
|
|
470
|
+
"Account name if authenticated"
|
|
471
|
+
accountName: String
|
|
472
|
+
"Token expiry time"
|
|
473
|
+
tokenExpiry: String
|
|
474
|
+
"Error message if status check failed"
|
|
475
|
+
error: String
|
|
476
|
+
}
|
|
477
|
+
|
|
478
|
+
type WpeAuthResult {
|
|
479
|
+
"Whether authentication was successful"
|
|
480
|
+
success: Boolean!
|
|
481
|
+
"User email if successful"
|
|
482
|
+
email: String
|
|
483
|
+
"Message about the authentication result"
|
|
484
|
+
message: String
|
|
485
|
+
"Error message if failed"
|
|
486
|
+
error: String
|
|
487
|
+
}
|
|
488
|
+
|
|
489
|
+
type WpeLogoutResult {
|
|
490
|
+
"Whether logout was successful"
|
|
491
|
+
success: Boolean!
|
|
492
|
+
"Message about the logout result"
|
|
493
|
+
message: String
|
|
494
|
+
"Error message if failed"
|
|
495
|
+
error: String
|
|
496
|
+
}
|
|
497
|
+
|
|
498
|
+
type WpeSite {
|
|
499
|
+
"Install ID"
|
|
500
|
+
id: String!
|
|
501
|
+
"Install name"
|
|
502
|
+
name: String!
|
|
503
|
+
"Environment (production, staging, development)"
|
|
504
|
+
environment: String!
|
|
505
|
+
"PHP version"
|
|
506
|
+
phpVersion: String
|
|
507
|
+
"Primary domain"
|
|
508
|
+
primaryDomain: String
|
|
509
|
+
"Account ID"
|
|
510
|
+
accountId: String
|
|
511
|
+
"Account name"
|
|
512
|
+
accountName: String
|
|
513
|
+
"SFTP host"
|
|
514
|
+
sftpHost: String
|
|
515
|
+
"SFTP user"
|
|
516
|
+
sftpUser: String
|
|
517
|
+
}
|
|
518
|
+
|
|
519
|
+
type ListWpeSitesResult {
|
|
520
|
+
"Whether query was successful"
|
|
521
|
+
success: Boolean!
|
|
522
|
+
"Error message if failed"
|
|
523
|
+
error: String
|
|
524
|
+
"List of WP Engine sites"
|
|
525
|
+
sites: [WpeSite!]
|
|
526
|
+
"Total count of sites"
|
|
527
|
+
count: Int
|
|
528
|
+
}
|
|
529
|
+
|
|
530
|
+
# Phase 11b: Site Linking Types
|
|
531
|
+
type WpeConnection {
|
|
532
|
+
"Remote install ID (UUID from WP Engine)"
|
|
533
|
+
remoteInstallId: String!
|
|
534
|
+
"Install name (human-readable, used in portal URLs)"
|
|
535
|
+
installName: String
|
|
536
|
+
"Environment (production, staging, development)"
|
|
537
|
+
environment: String
|
|
538
|
+
"Account ID"
|
|
539
|
+
accountId: String
|
|
540
|
+
"WP Engine portal URL"
|
|
541
|
+
portalUrl: String
|
|
542
|
+
"Primary domain/CNAME"
|
|
543
|
+
primaryDomain: String
|
|
544
|
+
}
|
|
545
|
+
|
|
546
|
+
"Sync capabilities available for WPE-connected sites"
|
|
547
|
+
type WpeSyncCapabilities {
|
|
548
|
+
"Whether user can push to WP Engine"
|
|
549
|
+
canPush: Boolean!
|
|
550
|
+
"Whether user can pull from WP Engine"
|
|
551
|
+
canPull: Boolean!
|
|
552
|
+
"Available sync modes"
|
|
553
|
+
syncModes: [String!]!
|
|
554
|
+
"Whether Magic Sync (select files) is available"
|
|
555
|
+
magicSyncAvailable: Boolean!
|
|
556
|
+
"Whether database sync is available"
|
|
557
|
+
databaseSyncAvailable: Boolean!
|
|
558
|
+
}
|
|
559
|
+
|
|
560
|
+
type GetWpeLinkResult {
|
|
561
|
+
"Whether site is linked to WP Engine"
|
|
562
|
+
linked: Boolean!
|
|
563
|
+
"Site name"
|
|
564
|
+
siteName: String
|
|
565
|
+
"WP Engine connections"
|
|
566
|
+
connections: [WpeConnection!]
|
|
567
|
+
"Number of connections"
|
|
568
|
+
connectionCount: Int
|
|
569
|
+
"Sync capabilities (only present if linked)"
|
|
570
|
+
capabilities: WpeSyncCapabilities
|
|
571
|
+
"Message (for unlinked sites)"
|
|
572
|
+
message: String
|
|
573
|
+
"Error message if failed"
|
|
574
|
+
error: String
|
|
575
|
+
}
|
|
576
|
+
|
|
577
|
+
# Phase 11c: Sync Operations Types
|
|
578
|
+
type SyncHistoryEvent {
|
|
579
|
+
"Remote install name"
|
|
580
|
+
remoteInstallName: String
|
|
581
|
+
"Unix timestamp"
|
|
582
|
+
timestamp: Float!
|
|
583
|
+
"Environment (production, staging, development)"
|
|
584
|
+
environment: String!
|
|
585
|
+
"Sync direction"
|
|
586
|
+
direction: String!
|
|
587
|
+
"Sync status"
|
|
588
|
+
status: String
|
|
589
|
+
}
|
|
590
|
+
|
|
591
|
+
type GetSyncHistoryResult {
|
|
592
|
+
"Whether the query was successful"
|
|
593
|
+
success: Boolean!
|
|
594
|
+
"Site name"
|
|
595
|
+
siteName: String
|
|
596
|
+
"Sync history events"
|
|
597
|
+
events: [SyncHistoryEvent!]
|
|
598
|
+
"Number of events"
|
|
599
|
+
count: Int
|
|
600
|
+
"Error message if failed"
|
|
601
|
+
error: String
|
|
602
|
+
}
|
|
603
|
+
|
|
604
|
+
type SyncResult {
|
|
605
|
+
"Whether the sync was initiated successfully"
|
|
606
|
+
success: Boolean!
|
|
607
|
+
"Status message"
|
|
608
|
+
message: String
|
|
609
|
+
"Error message if failed"
|
|
610
|
+
error: String
|
|
611
|
+
}
|
|
612
|
+
|
|
613
|
+
# File change detection
|
|
614
|
+
type FileChange {
|
|
615
|
+
"File path relative to site root"
|
|
616
|
+
path: String!
|
|
617
|
+
"Change type: create, upload, download, delete, modify"
|
|
618
|
+
instruction: String!
|
|
619
|
+
"File size in bytes"
|
|
620
|
+
size: Int
|
|
621
|
+
"File type: - (file) or d (directory)"
|
|
622
|
+
type: String
|
|
623
|
+
}
|
|
624
|
+
|
|
625
|
+
type GetSiteChangesResult {
|
|
626
|
+
"Whether the query was successful"
|
|
627
|
+
success: Boolean!
|
|
628
|
+
"Site name"
|
|
629
|
+
siteName: String
|
|
630
|
+
"Direction of comparison"
|
|
631
|
+
direction: String
|
|
632
|
+
"Files that would be added/uploaded"
|
|
633
|
+
added: [FileChange!]
|
|
634
|
+
"Files that would be modified"
|
|
635
|
+
modified: [FileChange!]
|
|
636
|
+
"Files that would be deleted"
|
|
637
|
+
deleted: [FileChange!]
|
|
638
|
+
"Total number of changes"
|
|
639
|
+
totalChanges: Int
|
|
640
|
+
"Summary message"
|
|
641
|
+
message: String
|
|
642
|
+
"Error message if failed"
|
|
643
|
+
error: String
|
|
644
|
+
}
|
|
645
|
+
|
|
646
|
+
# Phase 10: Cloud Backup Types
|
|
647
|
+
type BackupProviderStatus {
|
|
648
|
+
"Whether authenticated with provider"
|
|
649
|
+
authenticated: Boolean!
|
|
650
|
+
"Account ID"
|
|
651
|
+
accountId: String
|
|
652
|
+
"Account email"
|
|
653
|
+
email: String
|
|
654
|
+
}
|
|
655
|
+
|
|
656
|
+
type BackupStatusResult {
|
|
657
|
+
"Whether backups are available"
|
|
658
|
+
available: Boolean!
|
|
659
|
+
"Whether the feature is enabled"
|
|
660
|
+
featureEnabled: Boolean!
|
|
661
|
+
"Dropbox authentication status"
|
|
662
|
+
dropbox: BackupProviderStatus
|
|
663
|
+
"Google Drive authentication status"
|
|
664
|
+
googleDrive: BackupProviderStatus
|
|
665
|
+
"Message if backups unavailable"
|
|
666
|
+
message: String
|
|
667
|
+
"Error message if failed"
|
|
668
|
+
error: String
|
|
669
|
+
}
|
|
670
|
+
|
|
671
|
+
type BackupMetadata {
|
|
672
|
+
"Snapshot ID"
|
|
673
|
+
snapshotId: String!
|
|
674
|
+
"Backup timestamp (ISO format)"
|
|
675
|
+
timestamp: String
|
|
676
|
+
"Backup note/description"
|
|
677
|
+
note: String
|
|
678
|
+
"Site domain"
|
|
679
|
+
siteDomain: String
|
|
680
|
+
"Services info (JSON)"
|
|
681
|
+
services: String
|
|
682
|
+
}
|
|
683
|
+
|
|
684
|
+
type ListBackupsResult {
|
|
685
|
+
"Whether query was successful"
|
|
686
|
+
success: Boolean!
|
|
687
|
+
"Site name"
|
|
688
|
+
siteName: String
|
|
689
|
+
"Backup provider"
|
|
690
|
+
provider: String
|
|
691
|
+
"List of backups"
|
|
692
|
+
backups: [BackupMetadata!]
|
|
693
|
+
"Number of backups"
|
|
694
|
+
count: Int
|
|
695
|
+
"Error message if failed"
|
|
696
|
+
error: String
|
|
697
|
+
}
|
|
698
|
+
|
|
699
|
+
type CreateBackupResult {
|
|
700
|
+
"Whether backup was created successfully"
|
|
701
|
+
success: Boolean!
|
|
702
|
+
"Snapshot ID"
|
|
703
|
+
snapshotId: String
|
|
704
|
+
"Backup timestamp"
|
|
705
|
+
timestamp: String
|
|
706
|
+
"Status message"
|
|
707
|
+
message: String
|
|
708
|
+
"Error message if failed"
|
|
709
|
+
error: String
|
|
710
|
+
}
|
|
711
|
+
|
|
712
|
+
type RestoreBackupResult {
|
|
713
|
+
"Whether restore was successful"
|
|
714
|
+
success: Boolean!
|
|
715
|
+
"Status message"
|
|
716
|
+
message: String
|
|
717
|
+
"Error message if failed"
|
|
718
|
+
error: String
|
|
719
|
+
}
|
|
720
|
+
|
|
721
|
+
type DeleteBackupResult {
|
|
722
|
+
"Whether deletion was successful"
|
|
723
|
+
success: Boolean!
|
|
724
|
+
"Deleted snapshot ID"
|
|
725
|
+
deletedSnapshotId: String
|
|
726
|
+
"Status message"
|
|
727
|
+
message: String
|
|
728
|
+
"Error message if failed"
|
|
729
|
+
error: String
|
|
730
|
+
}
|
|
731
|
+
|
|
732
|
+
type DownloadBackupResult {
|
|
733
|
+
"Whether download was successful"
|
|
734
|
+
success: Boolean!
|
|
735
|
+
"Path to downloaded file"
|
|
736
|
+
filePath: String
|
|
737
|
+
"Status message"
|
|
738
|
+
message: String
|
|
739
|
+
"Error message if failed"
|
|
740
|
+
error: String
|
|
741
|
+
}
|
|
742
|
+
|
|
743
|
+
type EditBackupNoteResult {
|
|
744
|
+
"Whether edit was successful"
|
|
745
|
+
success: Boolean!
|
|
746
|
+
"Updated snapshot ID"
|
|
747
|
+
snapshotId: String
|
|
748
|
+
"Updated note"
|
|
749
|
+
note: String
|
|
750
|
+
"Error message if failed"
|
|
751
|
+
error: String
|
|
752
|
+
}
|
|
753
|
+
|
|
754
|
+
extend type Query {
|
|
755
|
+
# Phase 10: Cloud Backups
|
|
756
|
+
"Check if cloud backups are available and authenticated"
|
|
757
|
+
backupStatus: BackupStatusResult!
|
|
758
|
+
|
|
759
|
+
"List all backups for a site"
|
|
760
|
+
listBackups(siteId: ID!, provider: String!): ListBackupsResult!
|
|
761
|
+
|
|
762
|
+
# Phase 11c: Sync Operations
|
|
763
|
+
"Get sync history for a local site"
|
|
764
|
+
getSyncHistory(siteId: ID!, limit: Int): GetSyncHistoryResult!
|
|
765
|
+
|
|
766
|
+
"Get file changes between local site and WP Engine (dry-run comparison)"
|
|
767
|
+
getSiteChanges(siteId: ID!, direction: String = "push"): GetSiteChangesResult!
|
|
768
|
+
}
|
|
769
|
+
|
|
770
|
+
extend type Mutation {
|
|
771
|
+
# Phase 10: Cloud Backups
|
|
772
|
+
"Create a backup of a site to cloud storage"
|
|
773
|
+
createBackup(siteId: ID!, provider: String!, note: String): CreateBackupResult!
|
|
774
|
+
|
|
775
|
+
"Restore a site from a cloud backup"
|
|
776
|
+
restoreBackup(
|
|
777
|
+
siteId: ID!
|
|
778
|
+
provider: String!
|
|
779
|
+
snapshotId: String!
|
|
780
|
+
confirm: Boolean = false
|
|
781
|
+
): RestoreBackupResult!
|
|
782
|
+
|
|
783
|
+
"Delete a backup from cloud storage"
|
|
784
|
+
deleteBackup(
|
|
785
|
+
siteId: ID!
|
|
786
|
+
provider: String!
|
|
787
|
+
snapshotId: String!
|
|
788
|
+
confirm: Boolean = false
|
|
789
|
+
): DeleteBackupResult!
|
|
790
|
+
|
|
791
|
+
"Download a backup as a ZIP file"
|
|
792
|
+
downloadBackup(siteId: ID!, provider: String!, snapshotId: String!): DownloadBackupResult!
|
|
793
|
+
|
|
794
|
+
"Update the note/description for a backup"
|
|
795
|
+
editBackupNote(
|
|
796
|
+
siteId: ID!
|
|
797
|
+
provider: String!
|
|
798
|
+
snapshotId: String!
|
|
799
|
+
note: String!
|
|
800
|
+
): EditBackupNoteResult!
|
|
801
|
+
|
|
802
|
+
# Phase 11: WP Engine Connect
|
|
803
|
+
"Authenticate with WP Engine (opens browser for OAuth)"
|
|
804
|
+
wpeAuthenticate: WpeAuthResult!
|
|
805
|
+
|
|
806
|
+
"Logout from WP Engine"
|
|
807
|
+
wpeLogout: WpeLogoutResult!
|
|
808
|
+
|
|
809
|
+
# Phase 11c: Sync Operations
|
|
810
|
+
"Push local site to WP Engine"
|
|
811
|
+
pushToWpe(
|
|
812
|
+
localSiteId: ID!
|
|
813
|
+
remoteInstallId: ID!
|
|
814
|
+
includeSql: Boolean = false
|
|
815
|
+
confirm: Boolean = false
|
|
816
|
+
): SyncResult!
|
|
817
|
+
|
|
818
|
+
"Pull from WP Engine to local site"
|
|
819
|
+
pullFromWpe(localSiteId: ID!, remoteInstallId: ID!, includeSql: Boolean = false): SyncResult!
|
|
820
|
+
}
|
|
821
|
+
`;
|
|
822
|
+
/**
|
|
823
|
+
* Create GraphQL resolvers that use Local's internal services
|
|
824
|
+
*/
|
|
825
|
+
function createResolvers(services) {
|
|
826
|
+
const { deleteSite: deleteSiteService, siteData, localLogger, wpCli, siteProcessManager, addSite: addSiteService, cloneSite: cloneSiteService, exportSite: exportSiteService, blueprints: blueprintsService, browserManager, adminer, x509Cert, siteProvisioner, importSite: importSiteService, lightningServices, siteDatabase, importSQLFile: importSQLFileService,
|
|
827
|
+
// Phase 11: WP Engine Connect
|
|
828
|
+
wpeOAuth: wpeOAuthService, capi: capiService,
|
|
829
|
+
// Phase 11c: Sync services
|
|
830
|
+
wpePush: wpePushService, wpePull: wpePullService, connectHistory: connectHistoryService, wpeConnectBase: wpeConnectBaseService,
|
|
831
|
+
// Note: Phase 10 Cloud Backup services are accessed via IPC to the Cloud Backups addon
|
|
832
|
+
// (backupService, dropbox, googleDrive, featureFlags, userData)
|
|
833
|
+
} = services;
|
|
834
|
+
// Helper to invoke IPC calls to the Cloud Backups addon
|
|
835
|
+
// This uses the same pattern as the BackupAIBridge
|
|
836
|
+
// Timeout constants for backup operations (in milliseconds)
|
|
837
|
+
const BACKUP_IPC_TIMEOUT = 600000; // 10 minutes for backup operations
|
|
838
|
+
const DEFAULT_IPC_TIMEOUT = 30000; // 30 seconds for quick operations
|
|
839
|
+
const invokeBackupIPC = async (channel, timeoutMs = BACKUP_IPC_TIMEOUT, ...args) => {
|
|
840
|
+
return new Promise((resolve, reject) => {
|
|
841
|
+
const timestamp = Date.now();
|
|
842
|
+
const random = Math.random().toString(36).substr(2, 9);
|
|
843
|
+
const successReplyChannel = `${channel}-success-${timestamp}-${random}`;
|
|
844
|
+
const errorReplyChannel = `${channel}-error-${timestamp}-${random}`;
|
|
845
|
+
const timeoutSeconds = Math.round(timeoutMs / 1000);
|
|
846
|
+
const timeout = setTimeout(() => {
|
|
847
|
+
electron_1.ipcMain.removeAllListeners(successReplyChannel);
|
|
848
|
+
electron_1.ipcMain.removeAllListeners(errorReplyChannel);
|
|
849
|
+
reject(new Error(`IPC call to ${channel} timed out after ${timeoutSeconds} seconds`));
|
|
850
|
+
}, timeoutMs);
|
|
851
|
+
electron_1.ipcMain.once(successReplyChannel, (_event, result) => {
|
|
852
|
+
clearTimeout(timeout);
|
|
853
|
+
electron_1.ipcMain.removeAllListeners(errorReplyChannel);
|
|
854
|
+
localLogger.info(`[${ADDON_NAME}] IPC success from ${channel}`);
|
|
855
|
+
resolve({ result });
|
|
856
|
+
});
|
|
857
|
+
electron_1.ipcMain.once(errorReplyChannel, (_event, error) => {
|
|
858
|
+
clearTimeout(timeout);
|
|
859
|
+
electron_1.ipcMain.removeAllListeners(successReplyChannel);
|
|
860
|
+
localLogger.error(`[${ADDON_NAME}] IPC error from ${channel}: ${error?.message}`);
|
|
861
|
+
resolve({ error });
|
|
862
|
+
});
|
|
863
|
+
const mockEvent = {
|
|
864
|
+
reply: (replyChannel, data) => {
|
|
865
|
+
electron_1.ipcMain.emit(replyChannel, null, data);
|
|
866
|
+
},
|
|
867
|
+
sender: {
|
|
868
|
+
send: (replyChannel, data) => {
|
|
869
|
+
electron_1.ipcMain.emit(replyChannel, null, data);
|
|
870
|
+
},
|
|
871
|
+
},
|
|
872
|
+
};
|
|
873
|
+
const replyChannels = { successReplyChannel, errorReplyChannel };
|
|
874
|
+
localLogger.info(`[${ADDON_NAME}] Invoking backup IPC: ${channel}`);
|
|
875
|
+
electron_1.ipcMain.emit(channel, mockEvent, replyChannels, ...args);
|
|
876
|
+
});
|
|
877
|
+
};
|
|
878
|
+
// Helper to get backup providers from the Cloud Backups addon
|
|
879
|
+
const getBackupProviders = async () => {
|
|
880
|
+
try {
|
|
881
|
+
const result = await invokeBackupIPC('backups:enabled-providers', DEFAULT_IPC_TIMEOUT);
|
|
882
|
+
localLogger.info(`[${ADDON_NAME}] Raw IPC result: ${JSON.stringify(result)}`);
|
|
883
|
+
if (result.error) {
|
|
884
|
+
localLogger.error(`[${ADDON_NAME}] Failed to get backup providers: ${result.error.message}`);
|
|
885
|
+
return [];
|
|
886
|
+
}
|
|
887
|
+
// The response is double-nested: result.result.result contains the array
|
|
888
|
+
// Structure: { result: { result: [providers...] } }
|
|
889
|
+
let providers = result.result;
|
|
890
|
+
// Unwrap nested result if present
|
|
891
|
+
if (providers && typeof providers === 'object' && !Array.isArray(providers)) {
|
|
892
|
+
if (Array.isArray(providers.result)) {
|
|
893
|
+
providers = providers.result;
|
|
894
|
+
}
|
|
895
|
+
else if (providers.result && typeof providers.result === 'object') {
|
|
896
|
+
// Even deeper nesting
|
|
897
|
+
providers = providers.result;
|
|
898
|
+
}
|
|
899
|
+
}
|
|
900
|
+
localLogger.info(`[${ADDON_NAME}] Extracted providers: ${JSON.stringify(providers)}`);
|
|
901
|
+
if (Array.isArray(providers)) {
|
|
902
|
+
localLogger.info(`[${ADDON_NAME}] Got ${providers.length} backup providers`);
|
|
903
|
+
return providers;
|
|
904
|
+
}
|
|
905
|
+
localLogger.warn(`[${ADDON_NAME}] Unexpected providers format after unwrapping: ${typeof providers}`);
|
|
906
|
+
return [];
|
|
907
|
+
}
|
|
908
|
+
catch (error) {
|
|
909
|
+
localLogger.error(`[${ADDON_NAME}] Error getting backup providers: ${error.message}`);
|
|
910
|
+
return [];
|
|
911
|
+
}
|
|
912
|
+
};
|
|
913
|
+
// Shared WP-CLI execution logic
|
|
914
|
+
const executeWpCli = async (_parent, args) => {
|
|
915
|
+
const { siteId, args: wpArgs, skipPlugins = true, skipThemes = true } = args.input;
|
|
916
|
+
try {
|
|
917
|
+
localLogger.info(`[${ADDON_NAME}] Running WP-CLI: wp ${wpArgs.join(' ')}`);
|
|
918
|
+
const site = siteData.getSite(siteId);
|
|
919
|
+
if (!site) {
|
|
920
|
+
return {
|
|
921
|
+
success: false,
|
|
922
|
+
output: null,
|
|
923
|
+
error: `Site not found: ${siteId}`,
|
|
924
|
+
};
|
|
925
|
+
}
|
|
926
|
+
const status = await siteProcessManager.getSiteStatus(site);
|
|
927
|
+
if (status !== 'running') {
|
|
928
|
+
return {
|
|
929
|
+
success: false,
|
|
930
|
+
output: null,
|
|
931
|
+
error: `Site "${site.name}" is not running. Start it first with: local-cli start ${site.name}`,
|
|
932
|
+
};
|
|
933
|
+
}
|
|
934
|
+
const output = await wpCli.run(site, wpArgs, {
|
|
935
|
+
skipPlugins,
|
|
936
|
+
skipThemes,
|
|
937
|
+
ignoreErrors: false,
|
|
938
|
+
});
|
|
939
|
+
localLogger.info(`[${ADDON_NAME}] WP-CLI completed successfully`);
|
|
940
|
+
return {
|
|
941
|
+
success: true,
|
|
942
|
+
output: output?.trim() || '',
|
|
943
|
+
error: null,
|
|
944
|
+
};
|
|
945
|
+
}
|
|
946
|
+
catch (error) {
|
|
947
|
+
localLogger.error(`[${ADDON_NAME}] WP-CLI failed:`, error);
|
|
948
|
+
return {
|
|
949
|
+
success: false,
|
|
950
|
+
output: null,
|
|
951
|
+
error: error.message || 'Unknown error',
|
|
952
|
+
};
|
|
953
|
+
}
|
|
954
|
+
};
|
|
955
|
+
return {
|
|
956
|
+
Query: {
|
|
957
|
+
wpCliQuery: executeWpCli,
|
|
958
|
+
blueprints: async () => {
|
|
959
|
+
try {
|
|
960
|
+
localLogger.info(`[${ADDON_NAME}] Fetching blueprints`);
|
|
961
|
+
const blueprintsList = await blueprintsService.getBlueprints();
|
|
962
|
+
return {
|
|
963
|
+
success: true,
|
|
964
|
+
error: null,
|
|
965
|
+
blueprints: blueprintsList.map((bp) => ({
|
|
966
|
+
name: bp.name,
|
|
967
|
+
lastModified: bp.lastModified,
|
|
968
|
+
// Handle nested objects - extract just the name/type string
|
|
969
|
+
phpVersion: typeof bp.phpVersion === 'object'
|
|
970
|
+
? bp.phpVersion?.name || bp.phpVersion?.version
|
|
971
|
+
: bp.phpVersion,
|
|
972
|
+
webServer: typeof bp.webServer === 'object'
|
|
973
|
+
? bp.webServer?.name || bp.webServer?.type
|
|
974
|
+
: bp.webServer,
|
|
975
|
+
database: typeof bp.database === 'object'
|
|
976
|
+
? bp.database?.name || bp.database?.type
|
|
977
|
+
: bp.database,
|
|
978
|
+
})),
|
|
979
|
+
};
|
|
980
|
+
}
|
|
981
|
+
catch (error) {
|
|
982
|
+
localLogger.error(`[${ADDON_NAME}] Failed to fetch blueprints:`, error);
|
|
983
|
+
return {
|
|
984
|
+
success: false,
|
|
985
|
+
error: error.message || 'Unknown error',
|
|
986
|
+
blueprints: [],
|
|
987
|
+
};
|
|
988
|
+
}
|
|
989
|
+
},
|
|
990
|
+
listServices: async (_parent, args) => {
|
|
991
|
+
const { type = 'all' } = args;
|
|
992
|
+
try {
|
|
993
|
+
localLogger.info(`[${ADDON_NAME}] Listing services (type: ${type})`);
|
|
994
|
+
if (!lightningServices) {
|
|
995
|
+
return {
|
|
996
|
+
success: false,
|
|
997
|
+
error: 'Lightning services not available',
|
|
998
|
+
services: [],
|
|
999
|
+
};
|
|
1000
|
+
}
|
|
1001
|
+
const roleMap = {
|
|
1002
|
+
php: 'php',
|
|
1003
|
+
database: 'mysql',
|
|
1004
|
+
webserver: 'nginx',
|
|
1005
|
+
};
|
|
1006
|
+
const roleFilter = type !== 'all' ? roleMap[type] : undefined;
|
|
1007
|
+
const registeredServices = lightningServices.getRegisteredServices(roleFilter);
|
|
1008
|
+
const serviceList = [];
|
|
1009
|
+
for (const [role, versions] of Object.entries(registeredServices)) {
|
|
1010
|
+
for (const [version, info] of Object.entries(versions)) {
|
|
1011
|
+
serviceList.push({
|
|
1012
|
+
role,
|
|
1013
|
+
name: info?.name || role,
|
|
1014
|
+
version,
|
|
1015
|
+
});
|
|
1016
|
+
}
|
|
1017
|
+
}
|
|
1018
|
+
return {
|
|
1019
|
+
success: true,
|
|
1020
|
+
error: null,
|
|
1021
|
+
services: serviceList,
|
|
1022
|
+
};
|
|
1023
|
+
}
|
|
1024
|
+
catch (error) {
|
|
1025
|
+
localLogger.error(`[${ADDON_NAME}] Failed to list services:`, error);
|
|
1026
|
+
return {
|
|
1027
|
+
success: false,
|
|
1028
|
+
error: error.message || 'Unknown error',
|
|
1029
|
+
services: [],
|
|
1030
|
+
};
|
|
1031
|
+
}
|
|
1032
|
+
},
|
|
1033
|
+
// Phase 11: WP Engine Connect
|
|
1034
|
+
wpeStatus: async () => {
|
|
1035
|
+
try {
|
|
1036
|
+
localLogger.info(`[${ADDON_NAME}] Checking WP Engine authentication status`);
|
|
1037
|
+
if (!wpeOAuthService) {
|
|
1038
|
+
return {
|
|
1039
|
+
authenticated: false,
|
|
1040
|
+
email: null,
|
|
1041
|
+
accountId: null,
|
|
1042
|
+
accountName: null,
|
|
1043
|
+
tokenExpiry: null,
|
|
1044
|
+
error: 'WP Engine OAuth service not available',
|
|
1045
|
+
};
|
|
1046
|
+
}
|
|
1047
|
+
// Check if we have valid credentials by trying to get access token
|
|
1048
|
+
const accessToken = await wpeOAuthService.getAccessToken();
|
|
1049
|
+
if (!accessToken) {
|
|
1050
|
+
return {
|
|
1051
|
+
authenticated: false,
|
|
1052
|
+
email: null,
|
|
1053
|
+
accountId: null,
|
|
1054
|
+
accountName: null,
|
|
1055
|
+
tokenExpiry: null,
|
|
1056
|
+
error: null,
|
|
1057
|
+
};
|
|
1058
|
+
}
|
|
1059
|
+
// Try to get user info from CAPI if available
|
|
1060
|
+
let email = null;
|
|
1061
|
+
if (capiService) {
|
|
1062
|
+
try {
|
|
1063
|
+
const currentUser = await capiService.getCurrentUser();
|
|
1064
|
+
email = currentUser?.email || null;
|
|
1065
|
+
}
|
|
1066
|
+
catch {
|
|
1067
|
+
// User info not available, but still authenticated
|
|
1068
|
+
}
|
|
1069
|
+
}
|
|
1070
|
+
return {
|
|
1071
|
+
authenticated: true,
|
|
1072
|
+
email,
|
|
1073
|
+
accountId: null,
|
|
1074
|
+
accountName: null,
|
|
1075
|
+
tokenExpiry: null,
|
|
1076
|
+
error: null,
|
|
1077
|
+
};
|
|
1078
|
+
}
|
|
1079
|
+
catch (error) {
|
|
1080
|
+
localLogger.error(`[${ADDON_NAME}] Failed to check WPE status:`, error);
|
|
1081
|
+
return {
|
|
1082
|
+
authenticated: false,
|
|
1083
|
+
email: null,
|
|
1084
|
+
accountId: null,
|
|
1085
|
+
accountName: null,
|
|
1086
|
+
tokenExpiry: null,
|
|
1087
|
+
error: error.message || 'Unknown error',
|
|
1088
|
+
};
|
|
1089
|
+
}
|
|
1090
|
+
},
|
|
1091
|
+
listWpeSites: async (_parent, args) => {
|
|
1092
|
+
const { accountId } = args;
|
|
1093
|
+
try {
|
|
1094
|
+
localLogger.info(`[${ADDON_NAME}] Listing WP Engine sites${accountId ? ` for account ${accountId}` : ''}`);
|
|
1095
|
+
if (!wpeOAuthService) {
|
|
1096
|
+
return {
|
|
1097
|
+
success: false,
|
|
1098
|
+
error: 'WP Engine OAuth service not available',
|
|
1099
|
+
sites: [],
|
|
1100
|
+
count: 0,
|
|
1101
|
+
};
|
|
1102
|
+
}
|
|
1103
|
+
// Check if authenticated by trying to get access token
|
|
1104
|
+
const accessToken = await wpeOAuthService.getAccessToken();
|
|
1105
|
+
if (!accessToken) {
|
|
1106
|
+
return {
|
|
1107
|
+
success: false,
|
|
1108
|
+
error: 'Not authenticated with WP Engine. Use wpe_authenticate first.',
|
|
1109
|
+
sites: [],
|
|
1110
|
+
count: 0,
|
|
1111
|
+
};
|
|
1112
|
+
}
|
|
1113
|
+
if (!capiService) {
|
|
1114
|
+
return {
|
|
1115
|
+
success: false,
|
|
1116
|
+
error: 'WP Engine CAPI service not available',
|
|
1117
|
+
sites: [],
|
|
1118
|
+
count: 0,
|
|
1119
|
+
};
|
|
1120
|
+
}
|
|
1121
|
+
// Get installs from CAPI using getInstallList
|
|
1122
|
+
const installs = await capiService.getInstallList();
|
|
1123
|
+
if (!installs) {
|
|
1124
|
+
return {
|
|
1125
|
+
success: true,
|
|
1126
|
+
error: null,
|
|
1127
|
+
sites: [],
|
|
1128
|
+
count: 0,
|
|
1129
|
+
};
|
|
1130
|
+
}
|
|
1131
|
+
const sites = installs.map((install) => ({
|
|
1132
|
+
id: install.id,
|
|
1133
|
+
name: install.name,
|
|
1134
|
+
environment: install.environment || 'production',
|
|
1135
|
+
phpVersion: install.phpVersion || null,
|
|
1136
|
+
primaryDomain: install.primaryDomain || install.cname || null,
|
|
1137
|
+
accountId: install.accountId || accountId || null,
|
|
1138
|
+
accountName: install.accountName || null,
|
|
1139
|
+
sftpHost: `${install.name}.ssh.wpengine.net`,
|
|
1140
|
+
sftpUser: install.name,
|
|
1141
|
+
}));
|
|
1142
|
+
return {
|
|
1143
|
+
success: true,
|
|
1144
|
+
error: null,
|
|
1145
|
+
sites,
|
|
1146
|
+
count: sites.length,
|
|
1147
|
+
};
|
|
1148
|
+
}
|
|
1149
|
+
catch (error) {
|
|
1150
|
+
localLogger.error(`[${ADDON_NAME}] Failed to list WPE sites:`, error);
|
|
1151
|
+
return {
|
|
1152
|
+
success: false,
|
|
1153
|
+
error: error.message || 'Unknown error',
|
|
1154
|
+
sites: [],
|
|
1155
|
+
count: 0,
|
|
1156
|
+
};
|
|
1157
|
+
}
|
|
1158
|
+
},
|
|
1159
|
+
// Phase 11b: Site Linking
|
|
1160
|
+
getWpeLink: async (_parent, args) => {
|
|
1161
|
+
const { siteId } = args;
|
|
1162
|
+
try {
|
|
1163
|
+
localLogger.info(`[${ADDON_NAME}] Getting WP Engine link for site ${siteId}`);
|
|
1164
|
+
// Get site from siteData
|
|
1165
|
+
const site = siteData.getSite(siteId);
|
|
1166
|
+
if (!site) {
|
|
1167
|
+
return {
|
|
1168
|
+
linked: false,
|
|
1169
|
+
siteName: null,
|
|
1170
|
+
connections: [],
|
|
1171
|
+
connectionCount: 0,
|
|
1172
|
+
message: null,
|
|
1173
|
+
error: `Site not found: ${siteId}`,
|
|
1174
|
+
};
|
|
1175
|
+
}
|
|
1176
|
+
// Get hostConnections from site
|
|
1177
|
+
const hostConnections = site.hostConnections || [];
|
|
1178
|
+
const wpeConnections = hostConnections.filter((c) => c.hostId === 'wpe');
|
|
1179
|
+
if (wpeConnections.length === 0) {
|
|
1180
|
+
return {
|
|
1181
|
+
linked: false,
|
|
1182
|
+
siteName: site.name,
|
|
1183
|
+
connections: [],
|
|
1184
|
+
connectionCount: 0,
|
|
1185
|
+
message: 'Site is not linked to any WP Engine environment. Use Connect in Local to pull a site from WPE.',
|
|
1186
|
+
error: null,
|
|
1187
|
+
};
|
|
1188
|
+
}
|
|
1189
|
+
// Transform connections for output, enriching with CAPI data if available
|
|
1190
|
+
const connections = await Promise.all(wpeConnections.map(async (c) => {
|
|
1191
|
+
let installName = c.remoteSiteId; // Default to UUID
|
|
1192
|
+
let portalUrl = null;
|
|
1193
|
+
let primaryDomain = null;
|
|
1194
|
+
// Try to get install details from CAPI to get the actual name
|
|
1195
|
+
// remoteSiteId matches install.site.id (WPE Site ID), not install.id
|
|
1196
|
+
if (capiService && typeof capiService.getInstallList === 'function') {
|
|
1197
|
+
try {
|
|
1198
|
+
localLogger.info(`[${ADDON_NAME}] Looking for install with site.id=${c.remoteSiteId}, env=${c.remoteSiteEnv}`);
|
|
1199
|
+
const installs = await capiService.getInstallList();
|
|
1200
|
+
localLogger.info(`[${ADDON_NAME}] Got ${installs?.length || 0} installs from CAPI`);
|
|
1201
|
+
if (installs && installs.length > 0) {
|
|
1202
|
+
// Log first install structure for debugging
|
|
1203
|
+
localLogger.info(`[${ADDON_NAME}] Sample install structure: ${JSON.stringify(installs[0], null, 2)}`);
|
|
1204
|
+
// Match by site.id (remoteSiteId is the WPE Site ID, not Install ID)
|
|
1205
|
+
// Also filter by environment if available
|
|
1206
|
+
const matchingInstall = installs.find((i) => i.site?.id === c.remoteSiteId &&
|
|
1207
|
+
(!c.remoteSiteEnv || i.environment === c.remoteSiteEnv));
|
|
1208
|
+
if (matchingInstall) {
|
|
1209
|
+
localLogger.info(`[${ADDON_NAME}] Found match: ${matchingInstall.name}`);
|
|
1210
|
+
installName = matchingInstall.name;
|
|
1211
|
+
portalUrl = `https://my.wpengine.com/installs/${matchingInstall.name}`;
|
|
1212
|
+
primaryDomain =
|
|
1213
|
+
matchingInstall.primary_domain || matchingInstall.cname || null;
|
|
1214
|
+
}
|
|
1215
|
+
else {
|
|
1216
|
+
localLogger.warn(`[${ADDON_NAME}] No matching install found for site.id=${c.remoteSiteId}`);
|
|
1217
|
+
}
|
|
1218
|
+
}
|
|
1219
|
+
}
|
|
1220
|
+
catch (e) {
|
|
1221
|
+
localLogger.warn(`[${ADDON_NAME}] Could not look up install from CAPI: ${e.message}`);
|
|
1222
|
+
}
|
|
1223
|
+
}
|
|
1224
|
+
else {
|
|
1225
|
+
localLogger.warn(`[${ADDON_NAME}] capiService or getInstallList not available`);
|
|
1226
|
+
}
|
|
1227
|
+
return {
|
|
1228
|
+
remoteInstallId: c.remoteSiteId,
|
|
1229
|
+
installName,
|
|
1230
|
+
environment: c.remoteSiteEnv || null,
|
|
1231
|
+
accountId: c.accountId || null,
|
|
1232
|
+
portalUrl,
|
|
1233
|
+
primaryDomain,
|
|
1234
|
+
};
|
|
1235
|
+
}));
|
|
1236
|
+
// Capabilities are always the same for WPE-connected sites
|
|
1237
|
+
const capabilities = {
|
|
1238
|
+
canPush: true,
|
|
1239
|
+
canPull: true,
|
|
1240
|
+
syncModes: ['all_files', 'select_files', 'database_only'],
|
|
1241
|
+
magicSyncAvailable: true,
|
|
1242
|
+
databaseSyncAvailable: true,
|
|
1243
|
+
};
|
|
1244
|
+
return {
|
|
1245
|
+
linked: true,
|
|
1246
|
+
siteName: site.name,
|
|
1247
|
+
connections,
|
|
1248
|
+
connectionCount: connections.length,
|
|
1249
|
+
capabilities,
|
|
1250
|
+
message: null,
|
|
1251
|
+
error: null,
|
|
1252
|
+
};
|
|
1253
|
+
}
|
|
1254
|
+
catch (error) {
|
|
1255
|
+
localLogger.error(`[${ADDON_NAME}] Failed to get WPE link:`, error);
|
|
1256
|
+
return {
|
|
1257
|
+
linked: false,
|
|
1258
|
+
siteName: null,
|
|
1259
|
+
connections: [],
|
|
1260
|
+
connectionCount: 0,
|
|
1261
|
+
message: null,
|
|
1262
|
+
error: error.message || 'Unknown error',
|
|
1263
|
+
};
|
|
1264
|
+
}
|
|
1265
|
+
},
|
|
1266
|
+
// Phase 10: Cloud Backups
|
|
1267
|
+
backupStatus: async () => {
|
|
1268
|
+
try {
|
|
1269
|
+
localLogger.info(`[${ADDON_NAME}] Checking backup status`);
|
|
1270
|
+
// Get providers from Cloud Backups addon via IPC
|
|
1271
|
+
const providers = await getBackupProviders();
|
|
1272
|
+
localLogger.info(`[${ADDON_NAME}] Got ${providers.length} backup providers`);
|
|
1273
|
+
if (providers.length === 0) {
|
|
1274
|
+
return {
|
|
1275
|
+
available: false,
|
|
1276
|
+
featureEnabled: false,
|
|
1277
|
+
dropbox: null,
|
|
1278
|
+
googleDrive: null,
|
|
1279
|
+
message: 'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub (hub.localwp.com/addons/cloud-backups).',
|
|
1280
|
+
error: null,
|
|
1281
|
+
};
|
|
1282
|
+
}
|
|
1283
|
+
// Map provider info to our response format
|
|
1284
|
+
const dropboxProvider = providers.find((p) => p.id === 'dropbox' || p.name?.toLowerCase().includes('dropbox'));
|
|
1285
|
+
const googleProvider = providers.find((p) => p.id === 'google' || p.name?.toLowerCase().includes('google'));
|
|
1286
|
+
const dropboxStatus = dropboxProvider
|
|
1287
|
+
? {
|
|
1288
|
+
authenticated: true,
|
|
1289
|
+
accountId: dropboxProvider.id,
|
|
1290
|
+
email: dropboxProvider.email || null,
|
|
1291
|
+
}
|
|
1292
|
+
: {
|
|
1293
|
+
authenticated: false,
|
|
1294
|
+
accountId: null,
|
|
1295
|
+
email: null,
|
|
1296
|
+
};
|
|
1297
|
+
const googleDriveStatus = googleProvider
|
|
1298
|
+
? {
|
|
1299
|
+
authenticated: true,
|
|
1300
|
+
accountId: googleProvider.id,
|
|
1301
|
+
email: googleProvider.email || null,
|
|
1302
|
+
}
|
|
1303
|
+
: {
|
|
1304
|
+
authenticated: false,
|
|
1305
|
+
accountId: null,
|
|
1306
|
+
email: null,
|
|
1307
|
+
};
|
|
1308
|
+
const hasProvider = providers.length > 0;
|
|
1309
|
+
return {
|
|
1310
|
+
available: hasProvider,
|
|
1311
|
+
featureEnabled: true,
|
|
1312
|
+
dropbox: dropboxStatus,
|
|
1313
|
+
googleDrive: googleDriveStatus,
|
|
1314
|
+
message: hasProvider
|
|
1315
|
+
? null
|
|
1316
|
+
: 'No cloud storage provider authenticated. Connect Dropbox or Google Drive in Local settings.',
|
|
1317
|
+
error: null,
|
|
1318
|
+
};
|
|
1319
|
+
}
|
|
1320
|
+
catch (error) {
|
|
1321
|
+
localLogger.error(`[${ADDON_NAME}] Failed to check backup status:`, error);
|
|
1322
|
+
return {
|
|
1323
|
+
available: false,
|
|
1324
|
+
featureEnabled: false,
|
|
1325
|
+
dropbox: null,
|
|
1326
|
+
googleDrive: null,
|
|
1327
|
+
message: null,
|
|
1328
|
+
error: error.message || 'Unknown error',
|
|
1329
|
+
};
|
|
1330
|
+
}
|
|
1331
|
+
},
|
|
1332
|
+
listBackups: async (_parent, args) => {
|
|
1333
|
+
const { siteId, provider } = args;
|
|
1334
|
+
try {
|
|
1335
|
+
localLogger.info(`[${ADDON_NAME}] Listing backups for site ${siteId} from ${provider}`);
|
|
1336
|
+
// Get site
|
|
1337
|
+
const site = siteData.getSite(siteId);
|
|
1338
|
+
if (!site) {
|
|
1339
|
+
return {
|
|
1340
|
+
success: false,
|
|
1341
|
+
siteName: null,
|
|
1342
|
+
provider,
|
|
1343
|
+
backups: [],
|
|
1344
|
+
count: 0,
|
|
1345
|
+
error: `Site not found: ${siteId}`,
|
|
1346
|
+
};
|
|
1347
|
+
}
|
|
1348
|
+
// Get providers from Cloud Backups addon
|
|
1349
|
+
const providers = await getBackupProviders();
|
|
1350
|
+
if (providers.length === 0) {
|
|
1351
|
+
return {
|
|
1352
|
+
success: false,
|
|
1353
|
+
siteName: site.name,
|
|
1354
|
+
provider,
|
|
1355
|
+
backups: [],
|
|
1356
|
+
count: 0,
|
|
1357
|
+
error: 'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub.',
|
|
1358
|
+
};
|
|
1359
|
+
}
|
|
1360
|
+
// Find the matching provider (map 'googleDrive' to 'google' for the addon)
|
|
1361
|
+
const providerMap = { googleDrive: 'google', dropbox: 'dropbox' };
|
|
1362
|
+
const providerId = providerMap[provider] || provider;
|
|
1363
|
+
const matchedProvider = providers.find((p) => p.id === providerId);
|
|
1364
|
+
if (!matchedProvider) {
|
|
1365
|
+
return {
|
|
1366
|
+
success: false,
|
|
1367
|
+
siteName: site.name,
|
|
1368
|
+
provider,
|
|
1369
|
+
backups: [],
|
|
1370
|
+
count: 0,
|
|
1371
|
+
error: `Provider '${provider}' not configured. Available: ${providers.map((p) => p.name).join(', ')}`,
|
|
1372
|
+
};
|
|
1373
|
+
}
|
|
1374
|
+
// For listing snapshots, use the Hub provider ID directly (e.g., 'google')
|
|
1375
|
+
// NOT the rclone backend name ('drive') - the Hub queries expect the OAuth provider name
|
|
1376
|
+
// Also pass pageOffset parameter (0 for first page)
|
|
1377
|
+
const result = await invokeBackupIPC('backups:provider-snapshots', DEFAULT_IPC_TIMEOUT, siteId, matchedProvider.id, 0);
|
|
1378
|
+
localLogger.info(`[${ADDON_NAME}] Provider snapshots raw result: ${JSON.stringify(result)}`);
|
|
1379
|
+
if (result.error) {
|
|
1380
|
+
return {
|
|
1381
|
+
success: false,
|
|
1382
|
+
siteName: site.name,
|
|
1383
|
+
provider,
|
|
1384
|
+
backups: [],
|
|
1385
|
+
count: 0,
|
|
1386
|
+
error: result.error.message || 'Failed to list backups',
|
|
1387
|
+
};
|
|
1388
|
+
}
|
|
1389
|
+
// Unwrap nested result structure (similar to providers)
|
|
1390
|
+
let backupsData = result.result;
|
|
1391
|
+
if (backupsData && typeof backupsData === 'object' && !Array.isArray(backupsData)) {
|
|
1392
|
+
// Check for nested result or snapshots array
|
|
1393
|
+
if (Array.isArray(backupsData.result)) {
|
|
1394
|
+
backupsData = backupsData.result;
|
|
1395
|
+
}
|
|
1396
|
+
else if (Array.isArray(backupsData.snapshots)) {
|
|
1397
|
+
backupsData = backupsData.snapshots;
|
|
1398
|
+
}
|
|
1399
|
+
else if (backupsData.result && Array.isArray(backupsData.result.snapshots)) {
|
|
1400
|
+
backupsData = backupsData.result.snapshots;
|
|
1401
|
+
}
|
|
1402
|
+
}
|
|
1403
|
+
const backups = Array.isArray(backupsData) ? backupsData : [];
|
|
1404
|
+
localLogger.info(`[${ADDON_NAME}] Extracted ${backups.length} backups`);
|
|
1405
|
+
return {
|
|
1406
|
+
success: true,
|
|
1407
|
+
siteName: site.name,
|
|
1408
|
+
provider,
|
|
1409
|
+
backups: backups.map((b) => ({
|
|
1410
|
+
// Use hash for snapshotId as that's what restic uses for restore/delete operations
|
|
1411
|
+
// The Hub ID (b.id) is just a database identifier
|
|
1412
|
+
snapshotId: b.hash || b.snapshotId || b.short_id,
|
|
1413
|
+
timestamp: b.updatedAt || b.createdAt || b.timestamp || b.time || b.created,
|
|
1414
|
+
note: b.configObject?.description || b.note || b.description || b.tags?.description || '',
|
|
1415
|
+
siteDomain: b.configObject?.name
|
|
1416
|
+
? `${b.configObject.name}.local`
|
|
1417
|
+
: b.siteDomain || site.domain,
|
|
1418
|
+
services: JSON.stringify(b.configObject?.services || b.services || {}),
|
|
1419
|
+
})),
|
|
1420
|
+
count: backups.length,
|
|
1421
|
+
error: null,
|
|
1422
|
+
};
|
|
1423
|
+
}
|
|
1424
|
+
catch (error) {
|
|
1425
|
+
localLogger.error(`[${ADDON_NAME}] Failed to list backups:`, error);
|
|
1426
|
+
return {
|
|
1427
|
+
success: false,
|
|
1428
|
+
siteName: null,
|
|
1429
|
+
provider,
|
|
1430
|
+
backups: [],
|
|
1431
|
+
count: 0,
|
|
1432
|
+
error: error.message || 'Unknown error',
|
|
1433
|
+
};
|
|
1434
|
+
}
|
|
1435
|
+
},
|
|
1436
|
+
// Phase 11c: Sync History
|
|
1437
|
+
getSyncHistory: async (_parent, args) => {
|
|
1438
|
+
const { siteId, limit = 30 } = args;
|
|
1439
|
+
try {
|
|
1440
|
+
localLogger.info(`[${ADDON_NAME}] Getting sync history for site ${siteId}`);
|
|
1441
|
+
// Get site to verify it exists
|
|
1442
|
+
const site = siteData.getSite(siteId);
|
|
1443
|
+
if (!site) {
|
|
1444
|
+
return {
|
|
1445
|
+
success: false,
|
|
1446
|
+
siteName: null,
|
|
1447
|
+
events: [],
|
|
1448
|
+
count: 0,
|
|
1449
|
+
error: `Site not found: ${siteId}`,
|
|
1450
|
+
};
|
|
1451
|
+
}
|
|
1452
|
+
// Check if connectHistory service is available
|
|
1453
|
+
if (!connectHistoryService || typeof connectHistoryService.getEvents !== 'function') {
|
|
1454
|
+
return {
|
|
1455
|
+
success: false,
|
|
1456
|
+
siteName: site.name,
|
|
1457
|
+
events: [],
|
|
1458
|
+
count: 0,
|
|
1459
|
+
error: 'Sync history service not available',
|
|
1460
|
+
};
|
|
1461
|
+
}
|
|
1462
|
+
const events = connectHistoryService.getEvents(siteId);
|
|
1463
|
+
const limitedEvents = events.slice(0, limit);
|
|
1464
|
+
return {
|
|
1465
|
+
success: true,
|
|
1466
|
+
siteName: site.name,
|
|
1467
|
+
events: limitedEvents.map((e) => ({
|
|
1468
|
+
remoteInstallName: e.remoteInstallName || null,
|
|
1469
|
+
timestamp: e.timestamp,
|
|
1470
|
+
environment: e.environment,
|
|
1471
|
+
direction: e.direction,
|
|
1472
|
+
status: e.status || null,
|
|
1473
|
+
})),
|
|
1474
|
+
count: limitedEvents.length,
|
|
1475
|
+
error: null,
|
|
1476
|
+
};
|
|
1477
|
+
}
|
|
1478
|
+
catch (error) {
|
|
1479
|
+
localLogger.error(`[${ADDON_NAME}] Failed to get sync history:`, error);
|
|
1480
|
+
return {
|
|
1481
|
+
success: false,
|
|
1482
|
+
siteName: null,
|
|
1483
|
+
events: [],
|
|
1484
|
+
count: 0,
|
|
1485
|
+
error: error.message || 'Unknown error',
|
|
1486
|
+
};
|
|
1487
|
+
}
|
|
1488
|
+
},
|
|
1489
|
+
// Get file changes between local and WPE (dry-run comparison)
|
|
1490
|
+
getSiteChanges: async (_parent, args) => {
|
|
1491
|
+
const { siteId, direction = 'push' } = args;
|
|
1492
|
+
try {
|
|
1493
|
+
localLogger.info(`[${ADDON_NAME}] Getting site changes for ${siteId}, direction=${direction}`);
|
|
1494
|
+
// Validate direction
|
|
1495
|
+
if (direction !== 'push' && direction !== 'pull') {
|
|
1496
|
+
return {
|
|
1497
|
+
success: false,
|
|
1498
|
+
siteName: null,
|
|
1499
|
+
direction,
|
|
1500
|
+
added: [],
|
|
1501
|
+
modified: [],
|
|
1502
|
+
deleted: [],
|
|
1503
|
+
totalChanges: 0,
|
|
1504
|
+
message: null,
|
|
1505
|
+
error: 'Invalid direction. Must be "push" or "pull".',
|
|
1506
|
+
};
|
|
1507
|
+
}
|
|
1508
|
+
// Get site
|
|
1509
|
+
const site = siteData.getSite(siteId);
|
|
1510
|
+
if (!site) {
|
|
1511
|
+
return {
|
|
1512
|
+
success: false,
|
|
1513
|
+
siteName: null,
|
|
1514
|
+
direction,
|
|
1515
|
+
added: [],
|
|
1516
|
+
modified: [],
|
|
1517
|
+
deleted: [],
|
|
1518
|
+
totalChanges: 0,
|
|
1519
|
+
message: null,
|
|
1520
|
+
error: `Site not found: ${siteId}`,
|
|
1521
|
+
};
|
|
1522
|
+
}
|
|
1523
|
+
// Check WPE connection
|
|
1524
|
+
const wpeConnection = site.hostConnections?.find((c) => c.hostId === 'wpe');
|
|
1525
|
+
if (!wpeConnection) {
|
|
1526
|
+
return {
|
|
1527
|
+
success: false,
|
|
1528
|
+
siteName: site.name,
|
|
1529
|
+
direction,
|
|
1530
|
+
added: [],
|
|
1531
|
+
modified: [],
|
|
1532
|
+
deleted: [],
|
|
1533
|
+
totalChanges: 0,
|
|
1534
|
+
message: null,
|
|
1535
|
+
error: 'Site is not linked to WP Engine. Use Connect in Local to link the site first.',
|
|
1536
|
+
};
|
|
1537
|
+
}
|
|
1538
|
+
// Check service availability
|
|
1539
|
+
if (!wpeConnectBaseService ||
|
|
1540
|
+
typeof wpeConnectBaseService.listModifications !== 'function') {
|
|
1541
|
+
return {
|
|
1542
|
+
success: false,
|
|
1543
|
+
siteName: site.name,
|
|
1544
|
+
direction,
|
|
1545
|
+
added: [],
|
|
1546
|
+
modified: [],
|
|
1547
|
+
deleted: [],
|
|
1548
|
+
totalChanges: 0,
|
|
1549
|
+
message: null,
|
|
1550
|
+
error: 'WPE Connect service not available',
|
|
1551
|
+
};
|
|
1552
|
+
}
|
|
1553
|
+
// Get install details from CAPI
|
|
1554
|
+
let installName = wpeConnection.remoteSiteId;
|
|
1555
|
+
let primaryDomain = '';
|
|
1556
|
+
let installId = '';
|
|
1557
|
+
if (capiService && typeof capiService.getInstallList === 'function') {
|
|
1558
|
+
const installs = await capiService.getInstallList();
|
|
1559
|
+
const matchingInstall = installs?.find((i) => i.site?.id === wpeConnection.remoteSiteId &&
|
|
1560
|
+
(!wpeConnection.remoteSiteEnv || i.environment === wpeConnection.remoteSiteEnv));
|
|
1561
|
+
if (matchingInstall) {
|
|
1562
|
+
installName = matchingInstall.name;
|
|
1563
|
+
primaryDomain =
|
|
1564
|
+
matchingInstall.primary_domain ||
|
|
1565
|
+
matchingInstall.cname ||
|
|
1566
|
+
`${matchingInstall.name}.wpengine.com`;
|
|
1567
|
+
installId = matchingInstall.id;
|
|
1568
|
+
}
|
|
1569
|
+
}
|
|
1570
|
+
if (!primaryDomain) {
|
|
1571
|
+
return {
|
|
1572
|
+
success: false,
|
|
1573
|
+
siteName: site.name,
|
|
1574
|
+
direction,
|
|
1575
|
+
added: [],
|
|
1576
|
+
modified: [],
|
|
1577
|
+
deleted: [],
|
|
1578
|
+
totalChanges: 0,
|
|
1579
|
+
message: null,
|
|
1580
|
+
error: 'Could not determine WP Engine install details. Please ensure you are authenticated.',
|
|
1581
|
+
};
|
|
1582
|
+
}
|
|
1583
|
+
// Call listModifications (dry-run rsync comparison)
|
|
1584
|
+
localLogger.info(`[${ADDON_NAME}] Calling listModifications for ${installName}`);
|
|
1585
|
+
const modifications = await wpeConnectBaseService.listModifications({
|
|
1586
|
+
connectArgs: {
|
|
1587
|
+
wpengineInstallName: installName,
|
|
1588
|
+
wpengineInstallId: installId,
|
|
1589
|
+
wpengineSiteId: wpeConnection.remoteSiteId,
|
|
1590
|
+
wpenginePrimaryDomain: primaryDomain,
|
|
1591
|
+
localSiteId: site.id,
|
|
1592
|
+
},
|
|
1593
|
+
direction: direction,
|
|
1594
|
+
includeIgnored: false,
|
|
1595
|
+
});
|
|
1596
|
+
// Categorize changes
|
|
1597
|
+
const added = modifications
|
|
1598
|
+
.filter((f) => f.instruction === 'create' ||
|
|
1599
|
+
f.instruction === 'upload' ||
|
|
1600
|
+
f.instruction === 'download')
|
|
1601
|
+
.map((f) => ({
|
|
1602
|
+
path: f.path,
|
|
1603
|
+
instruction: f.instruction,
|
|
1604
|
+
size: f.size,
|
|
1605
|
+
type: f.type,
|
|
1606
|
+
}));
|
|
1607
|
+
const modified = modifications
|
|
1608
|
+
.filter((f) => f.instruction === 'modify')
|
|
1609
|
+
.map((f) => ({
|
|
1610
|
+
path: f.path,
|
|
1611
|
+
instruction: f.instruction,
|
|
1612
|
+
size: f.size,
|
|
1613
|
+
type: f.type,
|
|
1614
|
+
}));
|
|
1615
|
+
const deleted = modifications
|
|
1616
|
+
.filter((f) => f.instruction === 'delete')
|
|
1617
|
+
.map((f) => ({
|
|
1618
|
+
path: f.path,
|
|
1619
|
+
instruction: f.instruction,
|
|
1620
|
+
size: f.size,
|
|
1621
|
+
type: f.type,
|
|
1622
|
+
}));
|
|
1623
|
+
const totalChanges = added.length + modified.length + deleted.length;
|
|
1624
|
+
const directionLabel = direction === 'push' ? 'local → WPE' : 'WPE → local';
|
|
1625
|
+
return {
|
|
1626
|
+
success: true,
|
|
1627
|
+
siteName: site.name,
|
|
1628
|
+
direction,
|
|
1629
|
+
added,
|
|
1630
|
+
modified,
|
|
1631
|
+
deleted,
|
|
1632
|
+
totalChanges,
|
|
1633
|
+
message: totalChanges > 0
|
|
1634
|
+
? `${totalChanges} file(s) changed (${directionLabel}): ${added.length} added, ${modified.length} modified, ${deleted.length} deleted`
|
|
1635
|
+
: `No changes detected (${directionLabel})`,
|
|
1636
|
+
error: null,
|
|
1637
|
+
};
|
|
1638
|
+
}
|
|
1639
|
+
catch (error) {
|
|
1640
|
+
localLogger.error(`[${ADDON_NAME}] Failed to get site changes:`, error);
|
|
1641
|
+
return {
|
|
1642
|
+
success: false,
|
|
1643
|
+
siteName: null,
|
|
1644
|
+
direction,
|
|
1645
|
+
added: [],
|
|
1646
|
+
modified: [],
|
|
1647
|
+
deleted: [],
|
|
1648
|
+
totalChanges: 0,
|
|
1649
|
+
message: null,
|
|
1650
|
+
error: error.message || 'Unknown error',
|
|
1651
|
+
};
|
|
1652
|
+
}
|
|
1653
|
+
},
|
|
1654
|
+
},
|
|
1655
|
+
Mutation: {
|
|
1656
|
+
wpCli: executeWpCli,
|
|
1657
|
+
createSite: async (_parent, args) => {
|
|
1658
|
+
// DEBUG: Log raw args received
|
|
1659
|
+
localLogger.info(`[${ADDON_NAME}] createSite called with args: ${JSON.stringify(args)}`);
|
|
1660
|
+
const { name, phpVersion, webServer = 'nginx', database = 'mysql', wpAdminUsername = 'admin', wpAdminPassword = 'password', wpAdminEmail = 'admin@local.test', blueprint, } = args.input;
|
|
1661
|
+
// DEBUG: Log destructured values
|
|
1662
|
+
localLogger.info(`[${ADDON_NAME}] Destructured - name: ${name}, blueprint: ${blueprint}, typeof blueprint: ${typeof blueprint}`);
|
|
1663
|
+
try {
|
|
1664
|
+
localLogger.info(`[${ADDON_NAME}] Creating site: ${name}${blueprint ? ` from blueprint: ${blueprint}` : ''}`);
|
|
1665
|
+
// Generate slug and domain from name
|
|
1666
|
+
const siteSlug = name
|
|
1667
|
+
.toLowerCase()
|
|
1668
|
+
.replace(/[^a-z0-9]+/g, '-')
|
|
1669
|
+
.replace(/^-|-$/g, '');
|
|
1670
|
+
const siteDomain = `${siteSlug}.local`;
|
|
1671
|
+
const os = require('os');
|
|
1672
|
+
const path = require('path');
|
|
1673
|
+
const fs = require('fs');
|
|
1674
|
+
const sitePath = path.join(os.homedir(), 'Local Sites', siteSlug);
|
|
1675
|
+
// If blueprint is provided, use importSiteService instead of addSiteService
|
|
1676
|
+
if (blueprint) {
|
|
1677
|
+
localLogger.info(`[${ADDON_NAME}] Blueprint parameter received: ${blueprint}`);
|
|
1678
|
+
// Get the userDataPath from electron app
|
|
1679
|
+
const { app } = require('electron');
|
|
1680
|
+
const userDataPath = app.getPath('userData');
|
|
1681
|
+
const blueprintZipPath = path.join(userDataPath, 'blueprints', `${blueprint}.zip`);
|
|
1682
|
+
localLogger.info(`[${ADDON_NAME}] Looking for blueprint at: ${blueprintZipPath}`);
|
|
1683
|
+
// Verify blueprint exists
|
|
1684
|
+
if (!fs.existsSync(blueprintZipPath)) {
|
|
1685
|
+
localLogger.error(`[${ADDON_NAME}] Blueprint not found at: ${blueprintZipPath}`);
|
|
1686
|
+
return {
|
|
1687
|
+
success: false,
|
|
1688
|
+
error: `Blueprint not found: ${blueprint}. Use list_blueprints to see available blueprints.`,
|
|
1689
|
+
siteId: null,
|
|
1690
|
+
siteName: name,
|
|
1691
|
+
siteDomain: null,
|
|
1692
|
+
};
|
|
1693
|
+
}
|
|
1694
|
+
localLogger.info(`[${ADDON_NAME}] Found blueprint at: ${blueprintZipPath}`);
|
|
1695
|
+
// Read the local-site.json from the blueprint zip to get manifest
|
|
1696
|
+
let localSiteJSON;
|
|
1697
|
+
try {
|
|
1698
|
+
const StreamZip = require('node-stream-zip');
|
|
1699
|
+
localLogger.info(`[${ADDON_NAME}] node-stream-zip loaded successfully`);
|
|
1700
|
+
const zip = new StreamZip.async({ file: blueprintZipPath });
|
|
1701
|
+
const entries = await zip.entries();
|
|
1702
|
+
localLogger.info(`[${ADDON_NAME}] Zip entries loaded, count: ${Object.keys(entries).length}`);
|
|
1703
|
+
const filename = entries['local-site.json']
|
|
1704
|
+
? 'local-site.json'
|
|
1705
|
+
: 'pressmatic-site.json';
|
|
1706
|
+
localLogger.info(`[${ADDON_NAME}] Reading manifest file: ${filename}`);
|
|
1707
|
+
const data = await zip.entryData(filename);
|
|
1708
|
+
localSiteJSON = JSON.parse(data.toString('utf8'));
|
|
1709
|
+
await zip.close();
|
|
1710
|
+
localLogger.info(`[${ADDON_NAME}] Successfully read manifest:`, JSON.stringify(localSiteJSON).substring(0, 200));
|
|
1711
|
+
}
|
|
1712
|
+
catch (zipError) {
|
|
1713
|
+
localLogger.error(`[${ADDON_NAME}] Failed to read blueprint zip: ${zipError.message}`, zipError);
|
|
1714
|
+
return {
|
|
1715
|
+
success: false,
|
|
1716
|
+
error: `Failed to read blueprint manifest: ${zipError.message}`,
|
|
1717
|
+
siteId: null,
|
|
1718
|
+
siteName: name,
|
|
1719
|
+
siteDomain: null,
|
|
1720
|
+
};
|
|
1721
|
+
}
|
|
1722
|
+
// Build import settings
|
|
1723
|
+
const importSettings = {
|
|
1724
|
+
siteName: name,
|
|
1725
|
+
siteDomain: siteDomain,
|
|
1726
|
+
sitePath: sitePath,
|
|
1727
|
+
zip: blueprintZipPath,
|
|
1728
|
+
importData: {
|
|
1729
|
+
type: 'local-blueprint',
|
|
1730
|
+
oldSite: localSiteJSON,
|
|
1731
|
+
},
|
|
1732
|
+
environment: localSiteJSON.environment || 'flywheel',
|
|
1733
|
+
blueprint: blueprint,
|
|
1734
|
+
};
|
|
1735
|
+
// Copy service versions from blueprint if available
|
|
1736
|
+
if (localSiteJSON.services) {
|
|
1737
|
+
// Extract PHP version
|
|
1738
|
+
const phpService = Object.values(localSiteJSON.services).find((s) => s.role === 'php');
|
|
1739
|
+
if (phpService) {
|
|
1740
|
+
importSettings.phpVersion = phpService.version;
|
|
1741
|
+
}
|
|
1742
|
+
// Extract database
|
|
1743
|
+
const dbService = Object.values(localSiteJSON.services).find((s) => s.role === 'database' || s.role === 'db');
|
|
1744
|
+
if (dbService) {
|
|
1745
|
+
importSettings.database = `${dbService.name}-${dbService.version}`;
|
|
1746
|
+
}
|
|
1747
|
+
// Extract web server
|
|
1748
|
+
const webService = Object.values(localSiteJSON.services).find((s) => s.role === 'http' || s.role === 'web');
|
|
1749
|
+
if (webService) {
|
|
1750
|
+
importSettings.webServer = `${webService.name}-${webService.version}`;
|
|
1751
|
+
}
|
|
1752
|
+
}
|
|
1753
|
+
else if (localSiteJSON.phpVersion) {
|
|
1754
|
+
importSettings.phpVersion = localSiteJSON.phpVersion;
|
|
1755
|
+
}
|
|
1756
|
+
localLogger.info(`[${ADDON_NAME}] Import settings prepared:`, JSON.stringify(importSettings).substring(0, 500));
|
|
1757
|
+
if (!importSiteService) {
|
|
1758
|
+
localLogger.error(`[${ADDON_NAME}] importSiteService is not available!`);
|
|
1759
|
+
return {
|
|
1760
|
+
success: false,
|
|
1761
|
+
error: 'Import service not available',
|
|
1762
|
+
siteId: null,
|
|
1763
|
+
siteName: name,
|
|
1764
|
+
siteDomain: null,
|
|
1765
|
+
};
|
|
1766
|
+
}
|
|
1767
|
+
localLogger.info(`[${ADDON_NAME}] Calling importSiteService.run()...`);
|
|
1768
|
+
// Use the importSiteService to create from blueprint
|
|
1769
|
+
const importResult = await importSiteService.run(importSettings);
|
|
1770
|
+
localLogger.info(`[${ADDON_NAME}] Import result:`, JSON.stringify(importResult || 'null').substring(0, 500));
|
|
1771
|
+
if (importResult && importResult.id) {
|
|
1772
|
+
localLogger.info(`[${ADDON_NAME}] Successfully created site from blueprint: ${name} (${importResult.id})`);
|
|
1773
|
+
return {
|
|
1774
|
+
success: true,
|
|
1775
|
+
error: null,
|
|
1776
|
+
siteId: importResult.id,
|
|
1777
|
+
siteName: name,
|
|
1778
|
+
siteDomain: siteDomain,
|
|
1779
|
+
};
|
|
1780
|
+
}
|
|
1781
|
+
else {
|
|
1782
|
+
localLogger.warn(`[${ADDON_NAME}] Import returned but no site ID found`);
|
|
1783
|
+
return {
|
|
1784
|
+
success: true,
|
|
1785
|
+
error: null,
|
|
1786
|
+
siteId: null,
|
|
1787
|
+
siteName: name,
|
|
1788
|
+
siteDomain: siteDomain,
|
|
1789
|
+
};
|
|
1790
|
+
}
|
|
1791
|
+
}
|
|
1792
|
+
// No blueprint - create a fresh site
|
|
1793
|
+
const newSiteInfo = {
|
|
1794
|
+
siteName: name,
|
|
1795
|
+
siteDomain: siteDomain,
|
|
1796
|
+
sitePath: sitePath,
|
|
1797
|
+
webServer: webServer,
|
|
1798
|
+
database: database,
|
|
1799
|
+
};
|
|
1800
|
+
if (phpVersion) {
|
|
1801
|
+
newSiteInfo.phpVersion = phpVersion;
|
|
1802
|
+
}
|
|
1803
|
+
const wpCredentials = {
|
|
1804
|
+
adminUsername: wpAdminUsername,
|
|
1805
|
+
adminPassword: wpAdminPassword,
|
|
1806
|
+
adminEmail: wpAdminEmail,
|
|
1807
|
+
};
|
|
1808
|
+
const site = await addSiteService.addSite({
|
|
1809
|
+
newSiteInfo,
|
|
1810
|
+
wpCredentials,
|
|
1811
|
+
goToSite: false,
|
|
1812
|
+
});
|
|
1813
|
+
localLogger.info(`[${ADDON_NAME}] Successfully created site: ${name} (${site.id})`);
|
|
1814
|
+
return {
|
|
1815
|
+
success: true,
|
|
1816
|
+
error: null,
|
|
1817
|
+
siteId: site.id,
|
|
1818
|
+
siteName: name,
|
|
1819
|
+
siteDomain: siteDomain,
|
|
1820
|
+
};
|
|
1821
|
+
}
|
|
1822
|
+
catch (error) {
|
|
1823
|
+
localLogger.error(`[${ADDON_NAME}] Failed to create site:`, error);
|
|
1824
|
+
return {
|
|
1825
|
+
success: false,
|
|
1826
|
+
error: error.message || 'Unknown error',
|
|
1827
|
+
siteId: null,
|
|
1828
|
+
siteName: name,
|
|
1829
|
+
siteDomain: null,
|
|
1830
|
+
};
|
|
1831
|
+
}
|
|
1832
|
+
},
|
|
1833
|
+
deleteSite: async (_parent, args) => {
|
|
1834
|
+
const { id, trashFiles = true, updateHosts = true } = args.input;
|
|
1835
|
+
try {
|
|
1836
|
+
localLogger.info(`[${ADDON_NAME}] Deleting site: ${id}`);
|
|
1837
|
+
const site = siteData.getSite(id);
|
|
1838
|
+
if (!site) {
|
|
1839
|
+
return {
|
|
1840
|
+
success: false,
|
|
1841
|
+
error: `Site not found: ${id}`,
|
|
1842
|
+
siteId: id,
|
|
1843
|
+
};
|
|
1844
|
+
}
|
|
1845
|
+
await deleteSiteService.deleteSite({
|
|
1846
|
+
site,
|
|
1847
|
+
trashFiles,
|
|
1848
|
+
updateHosts,
|
|
1849
|
+
});
|
|
1850
|
+
localLogger.info(`[${ADDON_NAME}] Successfully deleted site: ${site.name}`);
|
|
1851
|
+
return {
|
|
1852
|
+
success: true,
|
|
1853
|
+
error: null,
|
|
1854
|
+
siteId: id,
|
|
1855
|
+
};
|
|
1856
|
+
}
|
|
1857
|
+
catch (error) {
|
|
1858
|
+
localLogger.error(`[${ADDON_NAME}] Failed to delete site:`, error);
|
|
1859
|
+
return {
|
|
1860
|
+
success: false,
|
|
1861
|
+
error: error.message || 'Unknown error',
|
|
1862
|
+
siteId: id,
|
|
1863
|
+
};
|
|
1864
|
+
}
|
|
1865
|
+
},
|
|
1866
|
+
deleteSites: async (_parent, args) => {
|
|
1867
|
+
const { ids, trashFiles = true } = args;
|
|
1868
|
+
try {
|
|
1869
|
+
localLogger.info(`[${ADDON_NAME}] Deleting ${ids.length} sites`);
|
|
1870
|
+
await deleteSiteService.deleteSites({
|
|
1871
|
+
siteIds: ids,
|
|
1872
|
+
trashFiles,
|
|
1873
|
+
updateHosts: true,
|
|
1874
|
+
});
|
|
1875
|
+
localLogger.info(`[${ADDON_NAME}] Successfully deleted ${ids.length} sites`);
|
|
1876
|
+
return {
|
|
1877
|
+
success: true,
|
|
1878
|
+
error: null,
|
|
1879
|
+
siteId: ids.join(','),
|
|
1880
|
+
};
|
|
1881
|
+
}
|
|
1882
|
+
catch (error) {
|
|
1883
|
+
localLogger.error(`[${ADDON_NAME}] Failed to delete sites:`, error);
|
|
1884
|
+
return {
|
|
1885
|
+
success: false,
|
|
1886
|
+
error: error.message || 'Unknown error',
|
|
1887
|
+
siteId: ids.join(','),
|
|
1888
|
+
};
|
|
1889
|
+
}
|
|
1890
|
+
},
|
|
1891
|
+
openSite: async (_parent, args) => {
|
|
1892
|
+
const { siteId, path = '/' } = args.input;
|
|
1893
|
+
try {
|
|
1894
|
+
const site = siteData.getSite(siteId);
|
|
1895
|
+
if (!site) {
|
|
1896
|
+
return {
|
|
1897
|
+
success: false,
|
|
1898
|
+
error: `Site not found: ${siteId}`,
|
|
1899
|
+
url: null,
|
|
1900
|
+
};
|
|
1901
|
+
}
|
|
1902
|
+
// Check if site is running
|
|
1903
|
+
const status = await siteProcessManager.getSiteStatus(site);
|
|
1904
|
+
if (status !== 'running') {
|
|
1905
|
+
return {
|
|
1906
|
+
success: false,
|
|
1907
|
+
error: `Site "${site.name}" must be running to open in browser. Start it first.`,
|
|
1908
|
+
url: null,
|
|
1909
|
+
};
|
|
1910
|
+
}
|
|
1911
|
+
const protocol = site.isStarred ? 'https' : 'http';
|
|
1912
|
+
const url = `${protocol}://${site.domain}${path}`;
|
|
1913
|
+
localLogger.info(`[${ADDON_NAME}] Opening site in browser: ${url}`);
|
|
1914
|
+
if (browserManager) {
|
|
1915
|
+
await browserManager.openInBrowser(url);
|
|
1916
|
+
}
|
|
1917
|
+
else {
|
|
1918
|
+
// Fallback to shell.openExternal
|
|
1919
|
+
const { shell } = require('electron');
|
|
1920
|
+
await shell.openExternal(url);
|
|
1921
|
+
}
|
|
1922
|
+
return {
|
|
1923
|
+
success: true,
|
|
1924
|
+
error: null,
|
|
1925
|
+
url,
|
|
1926
|
+
};
|
|
1927
|
+
}
|
|
1928
|
+
catch (error) {
|
|
1929
|
+
localLogger.error(`[${ADDON_NAME}] Failed to open site:`, error);
|
|
1930
|
+
return {
|
|
1931
|
+
success: false,
|
|
1932
|
+
error: error.message || 'Unknown error',
|
|
1933
|
+
url: null,
|
|
1934
|
+
};
|
|
1935
|
+
}
|
|
1936
|
+
},
|
|
1937
|
+
cloneSite: async (_parent, args) => {
|
|
1938
|
+
const { siteId, newName } = args.input;
|
|
1939
|
+
try {
|
|
1940
|
+
const site = siteData.getSite(siteId);
|
|
1941
|
+
if (!site) {
|
|
1942
|
+
return {
|
|
1943
|
+
success: false,
|
|
1944
|
+
error: `Site not found: ${siteId}`,
|
|
1945
|
+
newSiteId: null,
|
|
1946
|
+
newSiteName: null,
|
|
1947
|
+
newSiteDomain: null,
|
|
1948
|
+
};
|
|
1949
|
+
}
|
|
1950
|
+
// Check if site is running - needed for database cloning
|
|
1951
|
+
const status = await siteProcessManager.getSiteStatus(site);
|
|
1952
|
+
if (status !== 'running') {
|
|
1953
|
+
return {
|
|
1954
|
+
success: false,
|
|
1955
|
+
error: `Site "${site.name}" must be running to clone. Start it first.`,
|
|
1956
|
+
newSiteId: null,
|
|
1957
|
+
newSiteName: null,
|
|
1958
|
+
newSiteDomain: null,
|
|
1959
|
+
};
|
|
1960
|
+
}
|
|
1961
|
+
localLogger.info(`[${ADDON_NAME}] Cloning site ${site.name} to ${newName}`);
|
|
1962
|
+
const newSite = await cloneSiteService.cloneSite({
|
|
1963
|
+
site,
|
|
1964
|
+
newSiteName: newName,
|
|
1965
|
+
});
|
|
1966
|
+
localLogger.info(`[${ADDON_NAME}] Successfully cloned site: ${newSite.name} (${newSite.id})`);
|
|
1967
|
+
return {
|
|
1968
|
+
success: true,
|
|
1969
|
+
error: null,
|
|
1970
|
+
newSiteId: newSite.id,
|
|
1971
|
+
newSiteName: newSite.name,
|
|
1972
|
+
newSiteDomain: newSite.domain,
|
|
1973
|
+
};
|
|
1974
|
+
}
|
|
1975
|
+
catch (error) {
|
|
1976
|
+
localLogger.error(`[${ADDON_NAME}] Failed to clone site:`, error);
|
|
1977
|
+
return {
|
|
1978
|
+
success: false,
|
|
1979
|
+
error: error.message || 'Unknown error',
|
|
1980
|
+
newSiteId: null,
|
|
1981
|
+
newSiteName: null,
|
|
1982
|
+
newSiteDomain: null,
|
|
1983
|
+
};
|
|
1984
|
+
}
|
|
1985
|
+
},
|
|
1986
|
+
exportSite: async (_parent, args) => {
|
|
1987
|
+
const { siteId, outputPath } = args.input;
|
|
1988
|
+
const os = require('os');
|
|
1989
|
+
const path = require('path');
|
|
1990
|
+
try {
|
|
1991
|
+
const site = siteData.getSite(siteId);
|
|
1992
|
+
if (!site) {
|
|
1993
|
+
return {
|
|
1994
|
+
success: false,
|
|
1995
|
+
error: `Site not found: ${siteId}`,
|
|
1996
|
+
exportPath: null,
|
|
1997
|
+
};
|
|
1998
|
+
}
|
|
1999
|
+
// Check if site is running - needed for database export
|
|
2000
|
+
const status = await siteProcessManager.getSiteStatus(site);
|
|
2001
|
+
if (status !== 'running') {
|
|
2002
|
+
return {
|
|
2003
|
+
success: false,
|
|
2004
|
+
error: `Site "${site.name}" must be running to export. Start it first.`,
|
|
2005
|
+
exportPath: null,
|
|
2006
|
+
};
|
|
2007
|
+
}
|
|
2008
|
+
// Default to Downloads folder
|
|
2009
|
+
const outputDir = outputPath || path.join(os.homedir(), 'Downloads');
|
|
2010
|
+
const timestamp = new Date().toISOString().replace(/[:.]/g, '-').slice(0, 19);
|
|
2011
|
+
const fileName = `${site.name}-${timestamp}.zip`;
|
|
2012
|
+
const fullPath = path.join(outputDir, fileName);
|
|
2013
|
+
localLogger.info(`[${ADDON_NAME}] Exporting site ${site.name} to ${fullPath}`);
|
|
2014
|
+
// Use default export filter (excludes archive files)
|
|
2015
|
+
const defaultExportFilter = '*.zip, *.tar.gz, *.bz2, *.tgz';
|
|
2016
|
+
await exportSiteService.exportSite({
|
|
2017
|
+
site,
|
|
2018
|
+
outputPath: fullPath,
|
|
2019
|
+
filter: defaultExportFilter,
|
|
2020
|
+
});
|
|
2021
|
+
localLogger.info(`[${ADDON_NAME}] Successfully exported site to: ${fullPath}`);
|
|
2022
|
+
return {
|
|
2023
|
+
success: true,
|
|
2024
|
+
error: null,
|
|
2025
|
+
exportPath: fullPath,
|
|
2026
|
+
};
|
|
2027
|
+
}
|
|
2028
|
+
catch (error) {
|
|
2029
|
+
localLogger.error(`[${ADDON_NAME}] Failed to export site:`, error);
|
|
2030
|
+
return {
|
|
2031
|
+
success: false,
|
|
2032
|
+
error: error.message || 'Unknown error',
|
|
2033
|
+
exportPath: null,
|
|
2034
|
+
};
|
|
2035
|
+
}
|
|
2036
|
+
},
|
|
2037
|
+
saveBlueprint: async (_parent, args) => {
|
|
2038
|
+
const { siteId, name } = args.input;
|
|
2039
|
+
try {
|
|
2040
|
+
const site = siteData.getSite(siteId);
|
|
2041
|
+
if (!site) {
|
|
2042
|
+
return {
|
|
2043
|
+
success: false,
|
|
2044
|
+
error: `Site not found: ${siteId}`,
|
|
2045
|
+
blueprintName: null,
|
|
2046
|
+
};
|
|
2047
|
+
}
|
|
2048
|
+
// Check if site is running - needed for database export
|
|
2049
|
+
const status = await siteProcessManager.getSiteStatus(site);
|
|
2050
|
+
if (status !== 'running') {
|
|
2051
|
+
return {
|
|
2052
|
+
success: false,
|
|
2053
|
+
error: `Site "${site.name}" must be running to save as blueprint. Start it first.`,
|
|
2054
|
+
blueprintName: null,
|
|
2055
|
+
};
|
|
2056
|
+
}
|
|
2057
|
+
localLogger.info(`[${ADDON_NAME}] Saving site ${site.name} as blueprint: ${name}`);
|
|
2058
|
+
// Use default export filter (excludes archive files)
|
|
2059
|
+
const defaultFilter = '*.zip, *.tar.gz, *.bz2, *.tgz';
|
|
2060
|
+
await blueprintsService.saveBlueprint({
|
|
2061
|
+
name,
|
|
2062
|
+
siteId,
|
|
2063
|
+
filter: defaultFilter,
|
|
2064
|
+
});
|
|
2065
|
+
localLogger.info(`[${ADDON_NAME}] Successfully saved blueprint: ${name}`);
|
|
2066
|
+
return {
|
|
2067
|
+
success: true,
|
|
2068
|
+
error: null,
|
|
2069
|
+
blueprintName: name,
|
|
2070
|
+
};
|
|
2071
|
+
}
|
|
2072
|
+
catch (error) {
|
|
2073
|
+
localLogger.error(`[${ADDON_NAME}] Failed to save blueprint:`, error);
|
|
2074
|
+
return {
|
|
2075
|
+
success: false,
|
|
2076
|
+
error: error.message || 'Unknown error',
|
|
2077
|
+
blueprintName: null,
|
|
2078
|
+
};
|
|
2079
|
+
}
|
|
2080
|
+
},
|
|
2081
|
+
// Phase 8: WordPress Development Tools
|
|
2082
|
+
exportDatabase: async (_parent, args) => {
|
|
2083
|
+
const { siteId, outputPath } = args.input;
|
|
2084
|
+
const os = require('os');
|
|
2085
|
+
const pathModule = require('path');
|
|
2086
|
+
try {
|
|
2087
|
+
const site = siteData.getSite(siteId);
|
|
2088
|
+
if (!site) {
|
|
2089
|
+
return {
|
|
2090
|
+
success: false,
|
|
2091
|
+
error: `Site not found: ${siteId}`,
|
|
2092
|
+
outputPath: null,
|
|
2093
|
+
};
|
|
2094
|
+
}
|
|
2095
|
+
// Check if site is running - database must be accessible
|
|
2096
|
+
const status = await siteProcessManager.getSiteStatus(site);
|
|
2097
|
+
if (status !== 'running') {
|
|
2098
|
+
return {
|
|
2099
|
+
success: false,
|
|
2100
|
+
error: `Site "${site.name}" must be running to export database. Start it first.`,
|
|
2101
|
+
outputPath: null,
|
|
2102
|
+
};
|
|
2103
|
+
}
|
|
2104
|
+
// Default to Downloads folder with site name
|
|
2105
|
+
const defaultPath = pathModule.join(os.homedir(), 'Downloads', `${site.name.replace(/[^a-z0-9]/gi, '-')}.sql`);
|
|
2106
|
+
const finalPath = outputPath || defaultPath;
|
|
2107
|
+
localLogger.info(`[${ADDON_NAME}] Exporting database for ${site.name} to ${finalPath}`);
|
|
2108
|
+
// Use siteDatabase.dump() which properly sets up MySQL environment
|
|
2109
|
+
if (!siteDatabase) {
|
|
2110
|
+
return {
|
|
2111
|
+
success: false,
|
|
2112
|
+
error: 'Database service not available',
|
|
2113
|
+
outputPath: null,
|
|
2114
|
+
};
|
|
2115
|
+
}
|
|
2116
|
+
await siteDatabase.dump(site, finalPath);
|
|
2117
|
+
localLogger.info(`[${ADDON_NAME}] Successfully exported database to: ${finalPath}`);
|
|
2118
|
+
return {
|
|
2119
|
+
success: true,
|
|
2120
|
+
error: null,
|
|
2121
|
+
outputPath: finalPath,
|
|
2122
|
+
};
|
|
2123
|
+
}
|
|
2124
|
+
catch (error) {
|
|
2125
|
+
localLogger.error(`[${ADDON_NAME}] Failed to export database:`, error);
|
|
2126
|
+
return {
|
|
2127
|
+
success: false,
|
|
2128
|
+
error: error.message || 'Unknown error',
|
|
2129
|
+
outputPath: null,
|
|
2130
|
+
};
|
|
2131
|
+
}
|
|
2132
|
+
},
|
|
2133
|
+
importDatabase: async (_parent, args) => {
|
|
2134
|
+
const { siteId, sqlPath } = args.input;
|
|
2135
|
+
const fs = require('fs');
|
|
2136
|
+
try {
|
|
2137
|
+
const site = siteData.getSite(siteId);
|
|
2138
|
+
if (!site) {
|
|
2139
|
+
return {
|
|
2140
|
+
success: false,
|
|
2141
|
+
error: `Site not found: ${siteId}`,
|
|
2142
|
+
};
|
|
2143
|
+
}
|
|
2144
|
+
if (!fs.existsSync(sqlPath)) {
|
|
2145
|
+
return {
|
|
2146
|
+
success: false,
|
|
2147
|
+
error: `SQL file not found: ${sqlPath}`,
|
|
2148
|
+
};
|
|
2149
|
+
}
|
|
2150
|
+
// Check if site is running - database must be accessible
|
|
2151
|
+
const status = await siteProcessManager.getSiteStatus(site);
|
|
2152
|
+
if (status !== 'running') {
|
|
2153
|
+
return {
|
|
2154
|
+
success: false,
|
|
2155
|
+
error: `Site "${site.name}" must be running to import database. Start it first.`,
|
|
2156
|
+
};
|
|
2157
|
+
}
|
|
2158
|
+
localLogger.info(`[${ADDON_NAME}] Importing database for ${site.name} from ${sqlPath}`);
|
|
2159
|
+
// Use importSQLFile service which properly sets up MySQL environment
|
|
2160
|
+
if (!importSQLFileService) {
|
|
2161
|
+
return {
|
|
2162
|
+
success: false,
|
|
2163
|
+
error: 'Import SQL file service not available',
|
|
2164
|
+
};
|
|
2165
|
+
}
|
|
2166
|
+
await importSQLFileService(site, sqlPath);
|
|
2167
|
+
localLogger.info(`[${ADDON_NAME}] Successfully imported database from: ${sqlPath}`);
|
|
2168
|
+
return {
|
|
2169
|
+
success: true,
|
|
2170
|
+
error: null,
|
|
2171
|
+
};
|
|
2172
|
+
}
|
|
2173
|
+
catch (error) {
|
|
2174
|
+
localLogger.error(`[${ADDON_NAME}] Failed to import database:`, error);
|
|
2175
|
+
return {
|
|
2176
|
+
success: false,
|
|
2177
|
+
error: error.message || 'Unknown error',
|
|
2178
|
+
};
|
|
2179
|
+
}
|
|
2180
|
+
},
|
|
2181
|
+
openAdminer: async (_parent, args) => {
|
|
2182
|
+
const { siteId } = args.input;
|
|
2183
|
+
try {
|
|
2184
|
+
const site = siteData.getSite(siteId);
|
|
2185
|
+
if (!site) {
|
|
2186
|
+
return {
|
|
2187
|
+
success: false,
|
|
2188
|
+
error: `Site not found: ${siteId}`,
|
|
2189
|
+
};
|
|
2190
|
+
}
|
|
2191
|
+
// Check if site is running - database must be accessible
|
|
2192
|
+
const status = await siteProcessManager.getSiteStatus(site);
|
|
2193
|
+
if (status !== 'running') {
|
|
2194
|
+
return {
|
|
2195
|
+
success: false,
|
|
2196
|
+
error: `Site "${site.name}" must be running to open Adminer. Start it first.`,
|
|
2197
|
+
};
|
|
2198
|
+
}
|
|
2199
|
+
localLogger.info(`[${ADDON_NAME}] Opening Adminer for ${site.name}`);
|
|
2200
|
+
if (adminer) {
|
|
2201
|
+
await adminer.open(site);
|
|
2202
|
+
}
|
|
2203
|
+
else {
|
|
2204
|
+
return {
|
|
2205
|
+
success: false,
|
|
2206
|
+
error: 'Adminer service not available',
|
|
2207
|
+
};
|
|
2208
|
+
}
|
|
2209
|
+
return {
|
|
2210
|
+
success: true,
|
|
2211
|
+
error: null,
|
|
2212
|
+
};
|
|
2213
|
+
}
|
|
2214
|
+
catch (error) {
|
|
2215
|
+
localLogger.error(`[${ADDON_NAME}] Failed to open Adminer:`, error);
|
|
2216
|
+
return {
|
|
2217
|
+
success: false,
|
|
2218
|
+
error: error.message || 'Unknown error',
|
|
2219
|
+
};
|
|
2220
|
+
}
|
|
2221
|
+
},
|
|
2222
|
+
trustSsl: async (_parent, args) => {
|
|
2223
|
+
const { siteId } = args.input;
|
|
2224
|
+
try {
|
|
2225
|
+
const site = siteData.getSite(siteId);
|
|
2226
|
+
if (!site) {
|
|
2227
|
+
return {
|
|
2228
|
+
success: false,
|
|
2229
|
+
error: `Site not found: ${siteId}`,
|
|
2230
|
+
};
|
|
2231
|
+
}
|
|
2232
|
+
localLogger.info(`[${ADDON_NAME}] Trusting SSL for ${site.name}`);
|
|
2233
|
+
if (x509Cert) {
|
|
2234
|
+
await x509Cert.trustCert(site);
|
|
2235
|
+
}
|
|
2236
|
+
else {
|
|
2237
|
+
return {
|
|
2238
|
+
success: false,
|
|
2239
|
+
error: 'X509 certificate service not available',
|
|
2240
|
+
};
|
|
2241
|
+
}
|
|
2242
|
+
return {
|
|
2243
|
+
success: true,
|
|
2244
|
+
error: null,
|
|
2245
|
+
};
|
|
2246
|
+
}
|
|
2247
|
+
catch (error) {
|
|
2248
|
+
localLogger.error(`[${ADDON_NAME}] Failed to trust SSL:`, error);
|
|
2249
|
+
return {
|
|
2250
|
+
success: false,
|
|
2251
|
+
error: error.message || 'Unknown error',
|
|
2252
|
+
};
|
|
2253
|
+
}
|
|
2254
|
+
},
|
|
2255
|
+
mcpRenameSite: async (_parent, args) => {
|
|
2256
|
+
const { siteId, newName } = args.input;
|
|
2257
|
+
try {
|
|
2258
|
+
const site = siteData.getSite(siteId);
|
|
2259
|
+
if (!site) {
|
|
2260
|
+
return {
|
|
2261
|
+
success: false,
|
|
2262
|
+
error: `Site not found: ${siteId}`,
|
|
2263
|
+
newName: null,
|
|
2264
|
+
};
|
|
2265
|
+
}
|
|
2266
|
+
localLogger.info(`[${ADDON_NAME}] Renaming ${site.name} to ${newName}`);
|
|
2267
|
+
// Update site name via siteData
|
|
2268
|
+
site.name = newName;
|
|
2269
|
+
await siteData.updateSite(siteId, { name: newName });
|
|
2270
|
+
localLogger.info(`[${ADDON_NAME}] Successfully renamed site to: ${newName}`);
|
|
2271
|
+
return {
|
|
2272
|
+
success: true,
|
|
2273
|
+
error: null,
|
|
2274
|
+
newName,
|
|
2275
|
+
};
|
|
2276
|
+
}
|
|
2277
|
+
catch (error) {
|
|
2278
|
+
localLogger.error(`[${ADDON_NAME}] Failed to rename site:`, error);
|
|
2279
|
+
return {
|
|
2280
|
+
success: false,
|
|
2281
|
+
error: error.message || 'Unknown error',
|
|
2282
|
+
newName: null,
|
|
2283
|
+
};
|
|
2284
|
+
}
|
|
2285
|
+
},
|
|
2286
|
+
changePhpVersion: async (_parent, args) => {
|
|
2287
|
+
const { siteId, phpVersion } = args.input;
|
|
2288
|
+
try {
|
|
2289
|
+
const site = siteData.getSite(siteId);
|
|
2290
|
+
if (!site) {
|
|
2291
|
+
return {
|
|
2292
|
+
success: false,
|
|
2293
|
+
error: `Site not found: ${siteId}`,
|
|
2294
|
+
phpVersion: null,
|
|
2295
|
+
};
|
|
2296
|
+
}
|
|
2297
|
+
localLogger.info(`[${ADDON_NAME}] Changing PHP version for ${site.name} to ${phpVersion}`);
|
|
2298
|
+
if (siteProvisioner) {
|
|
2299
|
+
await siteProvisioner.swapService(site, 'php', phpVersion);
|
|
2300
|
+
}
|
|
2301
|
+
else {
|
|
2302
|
+
return {
|
|
2303
|
+
success: false,
|
|
2304
|
+
error: 'Site provisioner service not available',
|
|
2305
|
+
phpVersion: null,
|
|
2306
|
+
};
|
|
2307
|
+
}
|
|
2308
|
+
localLogger.info(`[${ADDON_NAME}] Successfully changed PHP version to: ${phpVersion}`);
|
|
2309
|
+
return {
|
|
2310
|
+
success: true,
|
|
2311
|
+
error: null,
|
|
2312
|
+
phpVersion,
|
|
2313
|
+
};
|
|
2314
|
+
}
|
|
2315
|
+
catch (error) {
|
|
2316
|
+
localLogger.error(`[${ADDON_NAME}] Failed to change PHP version:`, error);
|
|
2317
|
+
return {
|
|
2318
|
+
success: false,
|
|
2319
|
+
error: error.message || 'Unknown error',
|
|
2320
|
+
phpVersion: null,
|
|
2321
|
+
};
|
|
2322
|
+
}
|
|
2323
|
+
},
|
|
2324
|
+
importSite: async (_parent, args) => {
|
|
2325
|
+
const { zipPath, siteName } = args.input;
|
|
2326
|
+
const fs = require('fs');
|
|
2327
|
+
try {
|
|
2328
|
+
if (!fs.existsSync(zipPath)) {
|
|
2329
|
+
return {
|
|
2330
|
+
success: false,
|
|
2331
|
+
error: `Zip file not found: ${zipPath}`,
|
|
2332
|
+
siteId: null,
|
|
2333
|
+
siteName: null,
|
|
2334
|
+
};
|
|
2335
|
+
}
|
|
2336
|
+
localLogger.info(`[${ADDON_NAME}] Importing site from ${zipPath}`);
|
|
2337
|
+
if (!importSiteService) {
|
|
2338
|
+
return {
|
|
2339
|
+
success: false,
|
|
2340
|
+
error: 'Import site service not available',
|
|
2341
|
+
siteId: null,
|
|
2342
|
+
siteName: null,
|
|
2343
|
+
};
|
|
2344
|
+
}
|
|
2345
|
+
const result = await importSiteService.run({
|
|
2346
|
+
zipPath,
|
|
2347
|
+
siteName: siteName || undefined,
|
|
2348
|
+
});
|
|
2349
|
+
localLogger.info(`[${ADDON_NAME}] Successfully imported site: ${result.name}`);
|
|
2350
|
+
return {
|
|
2351
|
+
success: true,
|
|
2352
|
+
error: null,
|
|
2353
|
+
siteId: result.id,
|
|
2354
|
+
siteName: result.name,
|
|
2355
|
+
};
|
|
2356
|
+
}
|
|
2357
|
+
catch (error) {
|
|
2358
|
+
localLogger.error(`[${ADDON_NAME}] Failed to import site:`, error);
|
|
2359
|
+
return {
|
|
2360
|
+
success: false,
|
|
2361
|
+
error: error.message || 'Unknown error',
|
|
2362
|
+
siteId: null,
|
|
2363
|
+
siteName: null,
|
|
2364
|
+
};
|
|
2365
|
+
}
|
|
2366
|
+
},
|
|
2367
|
+
// Phase 9: Site Configuration & Dev Tools
|
|
2368
|
+
toggleXdebug: async (_parent, args) => {
|
|
2369
|
+
const { siteId, enabled } = args.input;
|
|
2370
|
+
try {
|
|
2371
|
+
const site = siteData.getSite(siteId);
|
|
2372
|
+
if (!site) {
|
|
2373
|
+
return {
|
|
2374
|
+
success: false,
|
|
2375
|
+
error: `Site not found: ${siteId}`,
|
|
2376
|
+
enabled: null,
|
|
2377
|
+
};
|
|
2378
|
+
}
|
|
2379
|
+
localLogger.info(`[${ADDON_NAME}] ${enabled ? 'Enabling' : 'Disabling'} Xdebug for ${site.name}`);
|
|
2380
|
+
// Update the site's xdebugEnabled property
|
|
2381
|
+
await siteData.updateSite(siteId, { xdebugEnabled: enabled });
|
|
2382
|
+
// Restart the site if it's running to apply the change
|
|
2383
|
+
const status = await siteProcessManager.getSiteStatus(site);
|
|
2384
|
+
if (status === 'running') {
|
|
2385
|
+
localLogger.info(`[${ADDON_NAME}] Restarting site to apply Xdebug change`);
|
|
2386
|
+
await siteProcessManager.restart(site);
|
|
2387
|
+
}
|
|
2388
|
+
localLogger.info(`[${ADDON_NAME}] Successfully ${enabled ? 'enabled' : 'disabled'} Xdebug`);
|
|
2389
|
+
return {
|
|
2390
|
+
success: true,
|
|
2391
|
+
error: null,
|
|
2392
|
+
enabled,
|
|
2393
|
+
};
|
|
2394
|
+
}
|
|
2395
|
+
catch (error) {
|
|
2396
|
+
localLogger.error(`[${ADDON_NAME}] Failed to toggle Xdebug:`, error);
|
|
2397
|
+
return {
|
|
2398
|
+
success: false,
|
|
2399
|
+
error: error.message || 'Unknown error',
|
|
2400
|
+
enabled: null,
|
|
2401
|
+
};
|
|
2402
|
+
}
|
|
2403
|
+
},
|
|
2404
|
+
getSiteLogs: async (_parent, args) => {
|
|
2405
|
+
const { siteId, logType = 'php', lines = 100 } = args.input;
|
|
2406
|
+
const fs = require('fs');
|
|
2407
|
+
const fsPromises = fs.promises;
|
|
2408
|
+
const pathModule = require('path');
|
|
2409
|
+
// Helper for async file existence check
|
|
2410
|
+
const fileExists = async (filePath) => {
|
|
2411
|
+
try {
|
|
2412
|
+
await fsPromises.access(filePath);
|
|
2413
|
+
return true;
|
|
2414
|
+
}
|
|
2415
|
+
catch {
|
|
2416
|
+
return false;
|
|
2417
|
+
}
|
|
2418
|
+
};
|
|
2419
|
+
// Helper to read last N lines of a file
|
|
2420
|
+
const readLastLines = async (filePath, numLines) => {
|
|
2421
|
+
try {
|
|
2422
|
+
const content = await fsPromises.readFile(filePath, 'utf-8');
|
|
2423
|
+
const logLines = content.split('\n');
|
|
2424
|
+
return logLines.slice(-numLines).join('\n') || '(empty)';
|
|
2425
|
+
}
|
|
2426
|
+
catch {
|
|
2427
|
+
return '';
|
|
2428
|
+
}
|
|
2429
|
+
};
|
|
2430
|
+
try {
|
|
2431
|
+
const site = siteData.getSite(siteId);
|
|
2432
|
+
if (!site) {
|
|
2433
|
+
return {
|
|
2434
|
+
success: false,
|
|
2435
|
+
error: `Site not found: ${siteId}`,
|
|
2436
|
+
logs: [],
|
|
2437
|
+
};
|
|
2438
|
+
}
|
|
2439
|
+
localLogger.info(`[${ADDON_NAME}] Getting ${logType} logs for ${site.name}`);
|
|
2440
|
+
const logs = [];
|
|
2441
|
+
const logsDir = pathModule.join(site.path, 'logs');
|
|
2442
|
+
const logFiles = {
|
|
2443
|
+
php: ['php', 'php-fpm'],
|
|
2444
|
+
nginx: ['nginx'],
|
|
2445
|
+
mysql: ['mysql'],
|
|
2446
|
+
all: ['php', 'php-fpm', 'nginx', 'mysql'],
|
|
2447
|
+
};
|
|
2448
|
+
const targetLogs = logFiles[logType] || logFiles.php;
|
|
2449
|
+
for (const logName of targetLogs) {
|
|
2450
|
+
// Check for error and access logs
|
|
2451
|
+
for (const suffix of ['error.log', 'access.log', '.log']) {
|
|
2452
|
+
const logPath = pathModule.join(logsDir, `${logName}${suffix === '.log' ? '' : '/'}${suffix}`);
|
|
2453
|
+
const altLogPath = pathModule.join(logsDir, `${logName}${suffix}`);
|
|
2454
|
+
let finalPath = null;
|
|
2455
|
+
if (await fileExists(logPath)) {
|
|
2456
|
+
finalPath = logPath;
|
|
2457
|
+
}
|
|
2458
|
+
else if (await fileExists(altLogPath)) {
|
|
2459
|
+
finalPath = altLogPath;
|
|
2460
|
+
}
|
|
2461
|
+
if (finalPath) {
|
|
2462
|
+
const content = await readLastLines(finalPath, lines);
|
|
2463
|
+
if (content) {
|
|
2464
|
+
logs.push({
|
|
2465
|
+
type: logName,
|
|
2466
|
+
content,
|
|
2467
|
+
path: finalPath,
|
|
2468
|
+
});
|
|
2469
|
+
}
|
|
2470
|
+
}
|
|
2471
|
+
}
|
|
2472
|
+
}
|
|
2473
|
+
if (logs.length === 0) {
|
|
2474
|
+
// Try to find any log files
|
|
2475
|
+
if (await fileExists(logsDir)) {
|
|
2476
|
+
const entries = await fsPromises.readdir(logsDir, { withFileTypes: true });
|
|
2477
|
+
for (const entry of entries) {
|
|
2478
|
+
if (entry.isDirectory()) {
|
|
2479
|
+
const subDir = pathModule.join(logsDir, entry.name);
|
|
2480
|
+
const subEntries = await fsPromises.readdir(subDir);
|
|
2481
|
+
for (const subFile of subEntries) {
|
|
2482
|
+
if (subFile.endsWith('.log')) {
|
|
2483
|
+
const logPath = pathModule.join(subDir, subFile);
|
|
2484
|
+
const content = await readLastLines(logPath, lines);
|
|
2485
|
+
if (content) {
|
|
2486
|
+
logs.push({
|
|
2487
|
+
type: entry.name,
|
|
2488
|
+
content,
|
|
2489
|
+
path: logPath,
|
|
2490
|
+
});
|
|
2491
|
+
}
|
|
2492
|
+
}
|
|
2493
|
+
}
|
|
2494
|
+
}
|
|
2495
|
+
else if (entry.name.endsWith('.log')) {
|
|
2496
|
+
const logPath = pathModule.join(logsDir, entry.name);
|
|
2497
|
+
const content = await readLastLines(logPath, lines);
|
|
2498
|
+
if (content) {
|
|
2499
|
+
logs.push({
|
|
2500
|
+
type: entry.name.replace('.log', ''),
|
|
2501
|
+
content,
|
|
2502
|
+
path: logPath,
|
|
2503
|
+
});
|
|
2504
|
+
}
|
|
2505
|
+
}
|
|
2506
|
+
}
|
|
2507
|
+
}
|
|
2508
|
+
}
|
|
2509
|
+
return {
|
|
2510
|
+
success: true,
|
|
2511
|
+
error: null,
|
|
2512
|
+
logs,
|
|
2513
|
+
};
|
|
2514
|
+
}
|
|
2515
|
+
catch (error) {
|
|
2516
|
+
localLogger.error(`[${ADDON_NAME}] Failed to get logs:`, error);
|
|
2517
|
+
return {
|
|
2518
|
+
success: false,
|
|
2519
|
+
error: error.message || 'Unknown error',
|
|
2520
|
+
logs: [],
|
|
2521
|
+
};
|
|
2522
|
+
}
|
|
2523
|
+
},
|
|
2524
|
+
// Phase 10: Cloud Backup Mutations
|
|
2525
|
+
createBackup: async (_parent, args) => {
|
|
2526
|
+
const { siteId, provider, note } = args;
|
|
2527
|
+
try {
|
|
2528
|
+
localLogger.info(`[${ADDON_NAME}] Creating backup for site ${siteId} to ${provider}`);
|
|
2529
|
+
// Get site
|
|
2530
|
+
const site = siteData.getSite(siteId);
|
|
2531
|
+
if (!site) {
|
|
2532
|
+
return {
|
|
2533
|
+
success: false,
|
|
2534
|
+
snapshotId: null,
|
|
2535
|
+
timestamp: null,
|
|
2536
|
+
message: null,
|
|
2537
|
+
error: `Site not found: ${siteId}`,
|
|
2538
|
+
};
|
|
2539
|
+
}
|
|
2540
|
+
// Get providers from Cloud Backups addon
|
|
2541
|
+
const providers = await getBackupProviders();
|
|
2542
|
+
if (providers.length === 0) {
|
|
2543
|
+
return {
|
|
2544
|
+
success: false,
|
|
2545
|
+
snapshotId: null,
|
|
2546
|
+
timestamp: null,
|
|
2547
|
+
message: null,
|
|
2548
|
+
error: 'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub.',
|
|
2549
|
+
};
|
|
2550
|
+
}
|
|
2551
|
+
// Find the matching provider (map 'googleDrive' to 'google' for the addon)
|
|
2552
|
+
const providerMap = { googleDrive: 'google', dropbox: 'dropbox' };
|
|
2553
|
+
const providerId = providerMap[provider] || provider;
|
|
2554
|
+
const matchedProvider = providers.find((p) => p.id === providerId);
|
|
2555
|
+
if (!matchedProvider) {
|
|
2556
|
+
return {
|
|
2557
|
+
success: false,
|
|
2558
|
+
snapshotId: null,
|
|
2559
|
+
timestamp: null,
|
|
2560
|
+
message: null,
|
|
2561
|
+
error: `Provider '${provider}' not configured. Available: ${providers.map((p) => p.name).join(', ')}`,
|
|
2562
|
+
};
|
|
2563
|
+
}
|
|
2564
|
+
// Map the Hub provider ID to rclone backend name
|
|
2565
|
+
// The addon uses 'google' in enabled-providers but expects 'drive' for backup operations
|
|
2566
|
+
const backupProviderMap = { google: 'drive', dropbox: 'dropbox' };
|
|
2567
|
+
const backupProviderId = backupProviderMap[matchedProvider.id] || matchedProvider.id;
|
|
2568
|
+
localLogger.info(`[${ADDON_NAME}] Using backup provider ID: ${backupProviderId} (from ${matchedProvider.id})`);
|
|
2569
|
+
// Create backup via IPC (use long timeout for backup operations)
|
|
2570
|
+
const description = note || 'Backup created via MCP';
|
|
2571
|
+
const result = await invokeBackupIPC('backups:backup-site', BACKUP_IPC_TIMEOUT, siteId, backupProviderId, description);
|
|
2572
|
+
localLogger.info(`[${ADDON_NAME}] Backup IPC result: ${JSON.stringify(result)}`);
|
|
2573
|
+
// Check for top-level IPC error
|
|
2574
|
+
if (result.error) {
|
|
2575
|
+
return {
|
|
2576
|
+
success: false,
|
|
2577
|
+
snapshotId: null,
|
|
2578
|
+
timestamp: null,
|
|
2579
|
+
message: null,
|
|
2580
|
+
error: result.error.message || 'Backup creation failed',
|
|
2581
|
+
};
|
|
2582
|
+
}
|
|
2583
|
+
// Unwrap nested result structure - the actual result is at result.result
|
|
2584
|
+
const backupResult = result.result;
|
|
2585
|
+
// Check if the backup result contains an error (nested at result.result.error)
|
|
2586
|
+
if (backupResult?.error) {
|
|
2587
|
+
const errorMsg = backupResult.error.message || backupResult.error.original?.message || 'Backup failed';
|
|
2588
|
+
return {
|
|
2589
|
+
success: false,
|
|
2590
|
+
snapshotId: null,
|
|
2591
|
+
timestamp: null,
|
|
2592
|
+
message: null,
|
|
2593
|
+
error: errorMsg,
|
|
2594
|
+
};
|
|
2595
|
+
}
|
|
2596
|
+
// Try to extract snapshot ID (may be nested in result.result.result)
|
|
2597
|
+
let snapshotId = backupResult?.snapshotId || backupResult?.id;
|
|
2598
|
+
if (!snapshotId && backupResult?.result) {
|
|
2599
|
+
snapshotId = backupResult.result.snapshotId || backupResult.result.id;
|
|
2600
|
+
}
|
|
2601
|
+
// If no error was returned, the backup succeeded even if no snapshot ID is provided
|
|
2602
|
+
// The addon doesn't always return the snapshot ID in its IPC response
|
|
2603
|
+
return {
|
|
2604
|
+
success: true,
|
|
2605
|
+
snapshotId: snapshotId || null,
|
|
2606
|
+
timestamp: new Date().toISOString(),
|
|
2607
|
+
message: `Backup created successfully to ${matchedProvider.name}`,
|
|
2608
|
+
error: null,
|
|
2609
|
+
};
|
|
2610
|
+
}
|
|
2611
|
+
catch (error) {
|
|
2612
|
+
localLogger.error(`[${ADDON_NAME}] Failed to create backup:`, error);
|
|
2613
|
+
return {
|
|
2614
|
+
success: false,
|
|
2615
|
+
snapshotId: null,
|
|
2616
|
+
timestamp: null,
|
|
2617
|
+
message: null,
|
|
2618
|
+
error: error.message || 'Unknown error',
|
|
2619
|
+
};
|
|
2620
|
+
}
|
|
2621
|
+
},
|
|
2622
|
+
restoreBackup: async (_parent, args) => {
|
|
2623
|
+
const { siteId, provider, snapshotId, confirm = false } = args;
|
|
2624
|
+
try {
|
|
2625
|
+
localLogger.info(`[${ADDON_NAME}] Restoring backup ${snapshotId} for site ${siteId}`);
|
|
2626
|
+
// Check confirmation
|
|
2627
|
+
if (!confirm) {
|
|
2628
|
+
return {
|
|
2629
|
+
success: false,
|
|
2630
|
+
message: null,
|
|
2631
|
+
error: 'Restore requires confirm=true to prevent accidental data loss. Current site files and database will be overwritten.',
|
|
2632
|
+
};
|
|
2633
|
+
}
|
|
2634
|
+
// Get site
|
|
2635
|
+
const site = siteData.getSite(siteId);
|
|
2636
|
+
if (!site) {
|
|
2637
|
+
return {
|
|
2638
|
+
success: false,
|
|
2639
|
+
message: null,
|
|
2640
|
+
error: `Site not found: ${siteId}`,
|
|
2641
|
+
};
|
|
2642
|
+
}
|
|
2643
|
+
// Get providers from Cloud Backups addon
|
|
2644
|
+
const providers = await getBackupProviders();
|
|
2645
|
+
if (providers.length === 0) {
|
|
2646
|
+
return {
|
|
2647
|
+
success: false,
|
|
2648
|
+
message: null,
|
|
2649
|
+
error: 'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub.',
|
|
2650
|
+
};
|
|
2651
|
+
}
|
|
2652
|
+
// Find the matching provider (map 'googleDrive' to 'google' for the addon)
|
|
2653
|
+
const providerMap = { googleDrive: 'google', dropbox: 'dropbox' };
|
|
2654
|
+
const providerId = providerMap[provider] || provider;
|
|
2655
|
+
const matchedProvider = providers.find((p) => p.id === providerId);
|
|
2656
|
+
if (!matchedProvider) {
|
|
2657
|
+
return {
|
|
2658
|
+
success: false,
|
|
2659
|
+
message: null,
|
|
2660
|
+
error: `Provider '${provider}' not configured. Available: ${providers.map((p) => p.name).join(', ')}`,
|
|
2661
|
+
};
|
|
2662
|
+
}
|
|
2663
|
+
// Map the Hub provider ID to rclone backend name
|
|
2664
|
+
const backupProviderMap = { google: 'drive', dropbox: 'dropbox' };
|
|
2665
|
+
const backupProviderId = backupProviderMap[matchedProvider.id] || matchedProvider.id;
|
|
2666
|
+
// Restore backup via IPC (use long timeout for restore operations)
|
|
2667
|
+
const result = await invokeBackupIPC('backups:restore-backup', BACKUP_IPC_TIMEOUT, siteId, backupProviderId, snapshotId);
|
|
2668
|
+
localLogger.info(`[${ADDON_NAME}] Restore result: ${JSON.stringify(result)}`);
|
|
2669
|
+
// Check for errors - can be at result.error or result.result.error (IPC async pattern)
|
|
2670
|
+
const ipcError = result.error || result.result?.error;
|
|
2671
|
+
if (ipcError) {
|
|
2672
|
+
const errorMessage = typeof ipcError === 'string' ? ipcError : ipcError.message || 'Restore failed';
|
|
2673
|
+
return {
|
|
2674
|
+
success: false,
|
|
2675
|
+
message: null,
|
|
2676
|
+
error: errorMessage,
|
|
2677
|
+
};
|
|
2678
|
+
}
|
|
2679
|
+
return {
|
|
2680
|
+
success: true,
|
|
2681
|
+
message: `Site restored from backup ${snapshotId}`,
|
|
2682
|
+
error: null,
|
|
2683
|
+
};
|
|
2684
|
+
}
|
|
2685
|
+
catch (error) {
|
|
2686
|
+
localLogger.error(`[${ADDON_NAME}] Failed to restore backup:`, error);
|
|
2687
|
+
return {
|
|
2688
|
+
success: false,
|
|
2689
|
+
message: null,
|
|
2690
|
+
error: error.message || 'Unknown error',
|
|
2691
|
+
};
|
|
2692
|
+
}
|
|
2693
|
+
},
|
|
2694
|
+
deleteBackup: async (_parent, args) => {
|
|
2695
|
+
const { siteId, provider, snapshotId, confirm = false } = args;
|
|
2696
|
+
try {
|
|
2697
|
+
localLogger.info(`[${ADDON_NAME}] Deleting backup ${snapshotId} for site ${siteId}`);
|
|
2698
|
+
// Check confirmation
|
|
2699
|
+
if (!confirm) {
|
|
2700
|
+
return {
|
|
2701
|
+
success: false,
|
|
2702
|
+
deletedSnapshotId: null,
|
|
2703
|
+
message: null,
|
|
2704
|
+
error: 'Delete requires confirm=true to prevent accidental deletion.',
|
|
2705
|
+
};
|
|
2706
|
+
}
|
|
2707
|
+
// Get site
|
|
2708
|
+
const site = siteData.getSite(siteId);
|
|
2709
|
+
if (!site) {
|
|
2710
|
+
return {
|
|
2711
|
+
success: false,
|
|
2712
|
+
deletedSnapshotId: null,
|
|
2713
|
+
message: null,
|
|
2714
|
+
error: `Site not found: ${siteId}`,
|
|
2715
|
+
};
|
|
2716
|
+
}
|
|
2717
|
+
// Get providers from Cloud Backups addon
|
|
2718
|
+
const providers = await getBackupProviders();
|
|
2719
|
+
if (providers.length === 0) {
|
|
2720
|
+
return {
|
|
2721
|
+
success: false,
|
|
2722
|
+
deletedSnapshotId: null,
|
|
2723
|
+
message: null,
|
|
2724
|
+
error: 'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub.',
|
|
2725
|
+
};
|
|
2726
|
+
}
|
|
2727
|
+
// Find the matching provider (map 'googleDrive' to 'google' for the addon)
|
|
2728
|
+
const providerMap = { googleDrive: 'google', dropbox: 'dropbox' };
|
|
2729
|
+
const providerId = providerMap[provider] || provider;
|
|
2730
|
+
const matchedProvider = providers.find((p) => p.id === providerId);
|
|
2731
|
+
if (!matchedProvider) {
|
|
2732
|
+
return {
|
|
2733
|
+
success: false,
|
|
2734
|
+
deletedSnapshotId: null,
|
|
2735
|
+
message: null,
|
|
2736
|
+
error: `Provider '${provider}' not configured. Available: ${providers.map((p) => p.name).join(', ')}`,
|
|
2737
|
+
};
|
|
2738
|
+
}
|
|
2739
|
+
// Map the Hub provider ID to rclone backend name
|
|
2740
|
+
const backupProviderMap = { google: 'drive', dropbox: 'dropbox' };
|
|
2741
|
+
const backupProviderId = backupProviderMap[matchedProvider.id] || matchedProvider.id;
|
|
2742
|
+
// Try to delete backup via IPC (may not be supported by the addon)
|
|
2743
|
+
const result = await invokeBackupIPC('backups:delete-backup', DEFAULT_IPC_TIMEOUT, siteId, backupProviderId, snapshotId);
|
|
2744
|
+
if (result.error) {
|
|
2745
|
+
// If the IPC channel doesn't exist or isn't supported, provide helpful message
|
|
2746
|
+
if (result.error.message?.includes('timed out')) {
|
|
2747
|
+
return {
|
|
2748
|
+
success: false,
|
|
2749
|
+
deletedSnapshotId: null,
|
|
2750
|
+
message: null,
|
|
2751
|
+
error: 'Delete backup operation is not available via MCP. Please delete backups through the Local UI.',
|
|
2752
|
+
};
|
|
2753
|
+
}
|
|
2754
|
+
return {
|
|
2755
|
+
success: false,
|
|
2756
|
+
deletedSnapshotId: null,
|
|
2757
|
+
message: null,
|
|
2758
|
+
error: result.error.message || 'Delete failed',
|
|
2759
|
+
};
|
|
2760
|
+
}
|
|
2761
|
+
return {
|
|
2762
|
+
success: true,
|
|
2763
|
+
deletedSnapshotId: snapshotId,
|
|
2764
|
+
message: 'Backup deleted',
|
|
2765
|
+
error: null,
|
|
2766
|
+
};
|
|
2767
|
+
}
|
|
2768
|
+
catch (error) {
|
|
2769
|
+
localLogger.error(`[${ADDON_NAME}] Failed to delete backup:`, error);
|
|
2770
|
+
return {
|
|
2771
|
+
success: false,
|
|
2772
|
+
deletedSnapshotId: null,
|
|
2773
|
+
message: null,
|
|
2774
|
+
error: error.message || 'Unknown error',
|
|
2775
|
+
};
|
|
2776
|
+
}
|
|
2777
|
+
},
|
|
2778
|
+
downloadBackup: async (_parent, args) => {
|
|
2779
|
+
const { siteId, provider, snapshotId } = args;
|
|
2780
|
+
try {
|
|
2781
|
+
localLogger.info(`[${ADDON_NAME}] Downloading backup ${snapshotId} for site ${siteId}`);
|
|
2782
|
+
// Get site
|
|
2783
|
+
const site = siteData.getSite(siteId);
|
|
2784
|
+
if (!site) {
|
|
2785
|
+
return {
|
|
2786
|
+
success: false,
|
|
2787
|
+
filePath: null,
|
|
2788
|
+
message: null,
|
|
2789
|
+
error: `Site not found: ${siteId}`,
|
|
2790
|
+
};
|
|
2791
|
+
}
|
|
2792
|
+
// Get providers from Cloud Backups addon
|
|
2793
|
+
const providers = await getBackupProviders();
|
|
2794
|
+
if (providers.length === 0) {
|
|
2795
|
+
return {
|
|
2796
|
+
success: false,
|
|
2797
|
+
filePath: null,
|
|
2798
|
+
message: null,
|
|
2799
|
+
error: 'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub.',
|
|
2800
|
+
};
|
|
2801
|
+
}
|
|
2802
|
+
// Find the matching provider (map 'googleDrive' to 'google' for the addon)
|
|
2803
|
+
const providerMap = { googleDrive: 'google', dropbox: 'dropbox' };
|
|
2804
|
+
const providerId = providerMap[provider] || provider;
|
|
2805
|
+
const matchedProvider = providers.find((p) => p.id === providerId);
|
|
2806
|
+
if (!matchedProvider) {
|
|
2807
|
+
return {
|
|
2808
|
+
success: false,
|
|
2809
|
+
filePath: null,
|
|
2810
|
+
message: null,
|
|
2811
|
+
error: `Provider '${provider}' not configured. Available: ${providers.map((p) => p.name).join(', ')}`,
|
|
2812
|
+
};
|
|
2813
|
+
}
|
|
2814
|
+
// Map the Hub provider ID to rclone backend name
|
|
2815
|
+
const backupProviderMap = { google: 'drive', dropbox: 'dropbox' };
|
|
2816
|
+
const backupProviderId = backupProviderMap[matchedProvider.id] || matchedProvider.id;
|
|
2817
|
+
// Try to download backup via IPC (use long timeout for downloads)
|
|
2818
|
+
const result = await invokeBackupIPC('backups:download-backup', BACKUP_IPC_TIMEOUT, siteId, backupProviderId, snapshotId);
|
|
2819
|
+
if (result.error) {
|
|
2820
|
+
// If the IPC channel doesn't exist or isn't supported, provide helpful message
|
|
2821
|
+
if (result.error.message?.includes('timed out')) {
|
|
2822
|
+
return {
|
|
2823
|
+
success: false,
|
|
2824
|
+
filePath: null,
|
|
2825
|
+
message: null,
|
|
2826
|
+
error: 'Download backup operation is not available via MCP. Please download backups through the Local UI.',
|
|
2827
|
+
};
|
|
2828
|
+
}
|
|
2829
|
+
return {
|
|
2830
|
+
success: false,
|
|
2831
|
+
filePath: null,
|
|
2832
|
+
message: null,
|
|
2833
|
+
error: result.error.message || 'Download failed',
|
|
2834
|
+
};
|
|
2835
|
+
}
|
|
2836
|
+
return {
|
|
2837
|
+
success: true,
|
|
2838
|
+
filePath: result.result?.filePath || null,
|
|
2839
|
+
message: 'Backup downloaded to Downloads folder',
|
|
2840
|
+
error: null,
|
|
2841
|
+
};
|
|
2842
|
+
}
|
|
2843
|
+
catch (error) {
|
|
2844
|
+
localLogger.error(`[${ADDON_NAME}] Failed to download backup:`, error);
|
|
2845
|
+
return {
|
|
2846
|
+
success: false,
|
|
2847
|
+
filePath: null,
|
|
2848
|
+
message: null,
|
|
2849
|
+
error: error.message || 'Unknown error',
|
|
2850
|
+
};
|
|
2851
|
+
}
|
|
2852
|
+
},
|
|
2853
|
+
editBackupNote: async (_parent, args) => {
|
|
2854
|
+
const { siteId, provider, snapshotId, note } = args;
|
|
2855
|
+
try {
|
|
2856
|
+
localLogger.info(`[${ADDON_NAME}] Editing backup note for ${snapshotId}`);
|
|
2857
|
+
// Get site
|
|
2858
|
+
const site = siteData.getSite(siteId);
|
|
2859
|
+
if (!site) {
|
|
2860
|
+
return {
|
|
2861
|
+
success: false,
|
|
2862
|
+
snapshotId: null,
|
|
2863
|
+
note: null,
|
|
2864
|
+
error: `Site not found: ${siteId}`,
|
|
2865
|
+
};
|
|
2866
|
+
}
|
|
2867
|
+
// Get providers from Cloud Backups addon
|
|
2868
|
+
const providers = await getBackupProviders();
|
|
2869
|
+
if (providers.length === 0) {
|
|
2870
|
+
return {
|
|
2871
|
+
success: false,
|
|
2872
|
+
snapshotId: null,
|
|
2873
|
+
note: null,
|
|
2874
|
+
error: 'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub.',
|
|
2875
|
+
};
|
|
2876
|
+
}
|
|
2877
|
+
// Find the matching provider (map 'googleDrive' to 'google' for the addon)
|
|
2878
|
+
const providerMap = { googleDrive: 'google', dropbox: 'dropbox' };
|
|
2879
|
+
const providerId = providerMap[provider] || provider;
|
|
2880
|
+
const matchedProvider = providers.find((p) => p.id === providerId);
|
|
2881
|
+
if (!matchedProvider) {
|
|
2882
|
+
return {
|
|
2883
|
+
success: false,
|
|
2884
|
+
snapshotId: null,
|
|
2885
|
+
note: null,
|
|
2886
|
+
error: `Provider '${provider}' not configured. Available: ${providers.map((p) => p.name).join(', ')}`,
|
|
2887
|
+
};
|
|
2888
|
+
}
|
|
2889
|
+
// Map the Hub provider ID to rclone backend name
|
|
2890
|
+
const backupProviderMap = { google: 'drive', dropbox: 'dropbox' };
|
|
2891
|
+
const backupProviderId = backupProviderMap[matchedProvider.id] || matchedProvider.id;
|
|
2892
|
+
// Try to edit backup note via IPC (quick metadata operation)
|
|
2893
|
+
const result = await invokeBackupIPC('backups:edit-note', DEFAULT_IPC_TIMEOUT, siteId, backupProviderId, snapshotId, note);
|
|
2894
|
+
if (result.error) {
|
|
2895
|
+
// If the IPC channel doesn't exist or isn't supported, provide helpful message
|
|
2896
|
+
if (result.error.message?.includes('timed out')) {
|
|
2897
|
+
return {
|
|
2898
|
+
success: false,
|
|
2899
|
+
snapshotId: null,
|
|
2900
|
+
note: null,
|
|
2901
|
+
error: 'Edit backup note operation is not available via MCP. Please edit backup notes through the Local UI.',
|
|
2902
|
+
};
|
|
2903
|
+
}
|
|
2904
|
+
return {
|
|
2905
|
+
success: false,
|
|
2906
|
+
snapshotId: null,
|
|
2907
|
+
note: null,
|
|
2908
|
+
error: result.error.message || 'Edit note failed',
|
|
2909
|
+
};
|
|
2910
|
+
}
|
|
2911
|
+
return {
|
|
2912
|
+
success: true,
|
|
2913
|
+
snapshotId,
|
|
2914
|
+
note,
|
|
2915
|
+
error: null,
|
|
2916
|
+
};
|
|
2917
|
+
}
|
|
2918
|
+
catch (error) {
|
|
2919
|
+
localLogger.error(`[${ADDON_NAME}] Failed to edit backup note:`, error);
|
|
2920
|
+
return {
|
|
2921
|
+
success: false,
|
|
2922
|
+
snapshotId: null,
|
|
2923
|
+
note: null,
|
|
2924
|
+
error: error.message || 'Unknown error',
|
|
2925
|
+
};
|
|
2926
|
+
}
|
|
2927
|
+
},
|
|
2928
|
+
// Phase 11: WP Engine Connect
|
|
2929
|
+
wpeAuthenticate: async () => {
|
|
2930
|
+
try {
|
|
2931
|
+
localLogger.info(`[${ADDON_NAME}] Initiating WP Engine authentication`);
|
|
2932
|
+
if (!wpeOAuthService) {
|
|
2933
|
+
return {
|
|
2934
|
+
success: false,
|
|
2935
|
+
email: null,
|
|
2936
|
+
message: null,
|
|
2937
|
+
error: 'WP Engine OAuth service not available',
|
|
2938
|
+
};
|
|
2939
|
+
}
|
|
2940
|
+
// Trigger OAuth flow - this will open browser for user consent
|
|
2941
|
+
// authenticate() returns OAuthTokens on success
|
|
2942
|
+
const tokens = await wpeOAuthService.authenticate();
|
|
2943
|
+
if (tokens && tokens.accessToken) {
|
|
2944
|
+
// Try to get user email from CAPI
|
|
2945
|
+
let email = null;
|
|
2946
|
+
if (capiService) {
|
|
2947
|
+
try {
|
|
2948
|
+
const currentUser = await capiService.getCurrentUser();
|
|
2949
|
+
email = currentUser?.email || null;
|
|
2950
|
+
}
|
|
2951
|
+
catch {
|
|
2952
|
+
// User info not available
|
|
2953
|
+
}
|
|
2954
|
+
}
|
|
2955
|
+
localLogger.info(`[${ADDON_NAME}] Successfully authenticated with WPE${email ? ` as ${email}` : ''}`);
|
|
2956
|
+
return {
|
|
2957
|
+
success: true,
|
|
2958
|
+
email,
|
|
2959
|
+
message: 'Successfully authenticated with WP Engine',
|
|
2960
|
+
error: null,
|
|
2961
|
+
};
|
|
2962
|
+
}
|
|
2963
|
+
return {
|
|
2964
|
+
success: true,
|
|
2965
|
+
email: null,
|
|
2966
|
+
message: 'Authentication initiated. Please complete the login in your browser.',
|
|
2967
|
+
error: null,
|
|
2968
|
+
};
|
|
2969
|
+
}
|
|
2970
|
+
catch (error) {
|
|
2971
|
+
localLogger.error(`[${ADDON_NAME}] WPE authentication failed:`, error);
|
|
2972
|
+
return {
|
|
2973
|
+
success: false,
|
|
2974
|
+
email: null,
|
|
2975
|
+
message: null,
|
|
2976
|
+
error: error.message || 'Authentication failed',
|
|
2977
|
+
};
|
|
2978
|
+
}
|
|
2979
|
+
},
|
|
2980
|
+
wpeLogout: async () => {
|
|
2981
|
+
try {
|
|
2982
|
+
localLogger.info(`[${ADDON_NAME}] Logging out from WP Engine`);
|
|
2983
|
+
if (!wpeOAuthService) {
|
|
2984
|
+
return {
|
|
2985
|
+
success: false,
|
|
2986
|
+
message: null,
|
|
2987
|
+
error: 'WP Engine OAuth service not available',
|
|
2988
|
+
};
|
|
2989
|
+
}
|
|
2990
|
+
// clearTokens() is the logout method
|
|
2991
|
+
await wpeOAuthService.clearTokens();
|
|
2992
|
+
localLogger.info(`[${ADDON_NAME}] Successfully logged out from WPE`);
|
|
2993
|
+
return {
|
|
2994
|
+
success: true,
|
|
2995
|
+
message: 'Logged out from WP Engine',
|
|
2996
|
+
error: null,
|
|
2997
|
+
};
|
|
2998
|
+
}
|
|
2999
|
+
catch (error) {
|
|
3000
|
+
localLogger.error(`[${ADDON_NAME}] WPE logout failed:`, error);
|
|
3001
|
+
return {
|
|
3002
|
+
success: false,
|
|
3003
|
+
message: null,
|
|
3004
|
+
error: error.message || 'Logout failed',
|
|
3005
|
+
};
|
|
3006
|
+
}
|
|
3007
|
+
},
|
|
3008
|
+
// Phase 11c: Push to WP Engine
|
|
3009
|
+
pushToWpe: async (_parent, args) => {
|
|
3010
|
+
const { localSiteId, remoteInstallId, includeSql = false, confirm = false } = args;
|
|
3011
|
+
try {
|
|
3012
|
+
localLogger.info(`[${ADDON_NAME}] Push to WPE: site=${localSiteId}, remote=${remoteInstallId}, includeSql=${includeSql}`);
|
|
3013
|
+
// Require confirmation for push operations
|
|
3014
|
+
if (!confirm) {
|
|
3015
|
+
return {
|
|
3016
|
+
success: false,
|
|
3017
|
+
message: null,
|
|
3018
|
+
error: 'Push requires confirm=true to prevent accidental overwrites. Set confirm=true to proceed.',
|
|
3019
|
+
};
|
|
3020
|
+
}
|
|
3021
|
+
// Verify site exists
|
|
3022
|
+
const site = siteData.getSite(localSiteId);
|
|
3023
|
+
if (!site) {
|
|
3024
|
+
return {
|
|
3025
|
+
success: false,
|
|
3026
|
+
message: null,
|
|
3027
|
+
error: `Site not found: ${localSiteId}`,
|
|
3028
|
+
};
|
|
3029
|
+
}
|
|
3030
|
+
// Check WPE connection exists
|
|
3031
|
+
const wpeConnection = site.hostConnections?.find((c) => c.hostId === 'wpe');
|
|
3032
|
+
if (!wpeConnection) {
|
|
3033
|
+
return {
|
|
3034
|
+
success: false,
|
|
3035
|
+
message: null,
|
|
3036
|
+
error: 'Site is not linked to WP Engine. Use Connect in Local to link the site first.',
|
|
3037
|
+
};
|
|
3038
|
+
}
|
|
3039
|
+
// Check push service availability
|
|
3040
|
+
if (!wpePushService || typeof wpePushService.push !== 'function') {
|
|
3041
|
+
return {
|
|
3042
|
+
success: false,
|
|
3043
|
+
message: null,
|
|
3044
|
+
error: 'WPE Push service not available',
|
|
3045
|
+
};
|
|
3046
|
+
}
|
|
3047
|
+
// Get install details from CAPI to get required parameters
|
|
3048
|
+
let installName = remoteInstallId;
|
|
3049
|
+
let primaryDomain = '';
|
|
3050
|
+
let installId = '';
|
|
3051
|
+
if (capiService && typeof capiService.getInstallList === 'function') {
|
|
3052
|
+
const installs = await capiService.getInstallList();
|
|
3053
|
+
const matchingInstall = installs?.find((i) => i.site?.id === wpeConnection.remoteSiteId &&
|
|
3054
|
+
(!wpeConnection.remoteSiteEnv || i.environment === wpeConnection.remoteSiteEnv));
|
|
3055
|
+
if (matchingInstall) {
|
|
3056
|
+
installName = matchingInstall.name;
|
|
3057
|
+
primaryDomain =
|
|
3058
|
+
matchingInstall.primary_domain ||
|
|
3059
|
+
matchingInstall.cname ||
|
|
3060
|
+
`${matchingInstall.name}.wpengine.com`;
|
|
3061
|
+
installId = matchingInstall.id;
|
|
3062
|
+
}
|
|
3063
|
+
}
|
|
3064
|
+
if (!primaryDomain) {
|
|
3065
|
+
return {
|
|
3066
|
+
success: false,
|
|
3067
|
+
message: null,
|
|
3068
|
+
error: 'Could not determine WP Engine install details. Please ensure you are authenticated.',
|
|
3069
|
+
};
|
|
3070
|
+
}
|
|
3071
|
+
// Start the push operation (async - returns immediately)
|
|
3072
|
+
wpePushService
|
|
3073
|
+
.push({
|
|
3074
|
+
includeSql,
|
|
3075
|
+
wpengineInstallName: installName,
|
|
3076
|
+
wpengineInstallId: installId,
|
|
3077
|
+
wpengineSiteId: wpeConnection.remoteSiteId,
|
|
3078
|
+
wpenginePrimaryDomain: primaryDomain,
|
|
3079
|
+
localSiteId: site.id,
|
|
3080
|
+
environment: wpeConnection.remoteSiteEnv,
|
|
3081
|
+
isMagicSync: false,
|
|
3082
|
+
})
|
|
3083
|
+
.catch((err) => {
|
|
3084
|
+
localLogger.error(`[${ADDON_NAME}] Push failed:`, err);
|
|
3085
|
+
});
|
|
3086
|
+
return {
|
|
3087
|
+
success: true,
|
|
3088
|
+
message: `Push started to ${installName}. Check Local UI for progress.`,
|
|
3089
|
+
error: null,
|
|
3090
|
+
};
|
|
3091
|
+
}
|
|
3092
|
+
catch (error) {
|
|
3093
|
+
localLogger.error(`[${ADDON_NAME}] Failed to start push:`, error);
|
|
3094
|
+
return {
|
|
3095
|
+
success: false,
|
|
3096
|
+
message: null,
|
|
3097
|
+
error: error.message || 'Failed to start push',
|
|
3098
|
+
};
|
|
3099
|
+
}
|
|
3100
|
+
},
|
|
3101
|
+
// Phase 11c: Pull from WP Engine
|
|
3102
|
+
pullFromWpe: async (_parent, args) => {
|
|
3103
|
+
const { localSiteId, remoteInstallId, includeSql = false } = args;
|
|
3104
|
+
try {
|
|
3105
|
+
localLogger.info(`[${ADDON_NAME}] Pull from WPE: site=${localSiteId}, remote=${remoteInstallId}, includeSql=${includeSql}`);
|
|
3106
|
+
// Verify site exists
|
|
3107
|
+
const site = siteData.getSite(localSiteId);
|
|
3108
|
+
if (!site) {
|
|
3109
|
+
return {
|
|
3110
|
+
success: false,
|
|
3111
|
+
message: null,
|
|
3112
|
+
error: `Site not found: ${localSiteId}`,
|
|
3113
|
+
};
|
|
3114
|
+
}
|
|
3115
|
+
// Check WPE connection exists
|
|
3116
|
+
const wpeConnection = site.hostConnections?.find((c) => c.hostId === 'wpe');
|
|
3117
|
+
if (!wpeConnection) {
|
|
3118
|
+
return {
|
|
3119
|
+
success: false,
|
|
3120
|
+
message: null,
|
|
3121
|
+
error: 'Site is not linked to WP Engine. Use Connect in Local to link the site first.',
|
|
3122
|
+
};
|
|
3123
|
+
}
|
|
3124
|
+
// Check pull service availability
|
|
3125
|
+
if (!wpePullService || typeof wpePullService.pull !== 'function') {
|
|
3126
|
+
return {
|
|
3127
|
+
success: false,
|
|
3128
|
+
message: null,
|
|
3129
|
+
error: 'WPE Pull service not available',
|
|
3130
|
+
};
|
|
3131
|
+
}
|
|
3132
|
+
// Get install details from CAPI
|
|
3133
|
+
let installName = remoteInstallId;
|
|
3134
|
+
let primaryDomain = '';
|
|
3135
|
+
let installId = '';
|
|
3136
|
+
if (capiService && typeof capiService.getInstallList === 'function') {
|
|
3137
|
+
const installs = await capiService.getInstallList();
|
|
3138
|
+
const matchingInstall = installs?.find((i) => i.site?.id === wpeConnection.remoteSiteId &&
|
|
3139
|
+
(!wpeConnection.remoteSiteEnv || i.environment === wpeConnection.remoteSiteEnv));
|
|
3140
|
+
if (matchingInstall) {
|
|
3141
|
+
installName = matchingInstall.name;
|
|
3142
|
+
primaryDomain =
|
|
3143
|
+
matchingInstall.primary_domain ||
|
|
3144
|
+
matchingInstall.cname ||
|
|
3145
|
+
`${matchingInstall.name}.wpengine.com`;
|
|
3146
|
+
installId = matchingInstall.id;
|
|
3147
|
+
}
|
|
3148
|
+
}
|
|
3149
|
+
if (!primaryDomain) {
|
|
3150
|
+
return {
|
|
3151
|
+
success: false,
|
|
3152
|
+
message: null,
|
|
3153
|
+
error: 'Could not determine WP Engine install details. Please ensure you are authenticated.',
|
|
3154
|
+
};
|
|
3155
|
+
}
|
|
3156
|
+
// Start the pull operation (async - returns immediately)
|
|
3157
|
+
wpePullService
|
|
3158
|
+
.pull({
|
|
3159
|
+
includeSql,
|
|
3160
|
+
wpengineInstallName: installName,
|
|
3161
|
+
wpengineInstallId: installId,
|
|
3162
|
+
wpengineSiteId: wpeConnection.remoteSiteId,
|
|
3163
|
+
wpenginePrimaryDomain: primaryDomain,
|
|
3164
|
+
localSiteId: site.id,
|
|
3165
|
+
environment: wpeConnection.remoteSiteEnv,
|
|
3166
|
+
isMagicSync: false,
|
|
3167
|
+
})
|
|
3168
|
+
.catch((err) => {
|
|
3169
|
+
localLogger.error(`[${ADDON_NAME}] Pull failed:`, err);
|
|
3170
|
+
});
|
|
3171
|
+
return {
|
|
3172
|
+
success: true,
|
|
3173
|
+
message: `Pull started from ${installName}. Check Local UI for progress.`,
|
|
3174
|
+
error: null,
|
|
3175
|
+
};
|
|
3176
|
+
}
|
|
3177
|
+
catch (error) {
|
|
3178
|
+
localLogger.error(`[${ADDON_NAME}] Failed to start pull:`, error);
|
|
3179
|
+
return {
|
|
3180
|
+
success: false,
|
|
3181
|
+
message: null,
|
|
3182
|
+
error: error.message || 'Failed to start pull',
|
|
3183
|
+
};
|
|
3184
|
+
}
|
|
3185
|
+
},
|
|
3186
|
+
},
|
|
3187
|
+
};
|
|
3188
|
+
}
|
|
3189
|
+
/**
|
|
3190
|
+
* Start the MCP server
|
|
3191
|
+
*/
|
|
3192
|
+
async function startMcpServer(services, logger) {
|
|
3193
|
+
if (mcpServer) {
|
|
3194
|
+
logger.warn(`[${ADDON_NAME}] MCP server already running`);
|
|
3195
|
+
return;
|
|
3196
|
+
}
|
|
3197
|
+
try {
|
|
3198
|
+
mcpServer = new McpServer_1.McpServer({ port: constants_1.MCP_SERVER.DEFAULT_PORT }, services, logger);
|
|
3199
|
+
await mcpServer.start();
|
|
3200
|
+
const info = mcpServer.getConnectionInfo();
|
|
3201
|
+
logger.info(`[${ADDON_NAME}] MCP server started on port ${info.port}`);
|
|
3202
|
+
logger.info(`[${ADDON_NAME}] MCP connection info saved to: ~/Library/Application Support/Local/mcp-connection-info.json`);
|
|
3203
|
+
logger.info(`[${ADDON_NAME}] Available tools: ${info.tools.join(', ')}`);
|
|
3204
|
+
}
|
|
3205
|
+
catch (error) {
|
|
3206
|
+
logger.error(`[${ADDON_NAME}] Failed to start MCP server:`, error);
|
|
3207
|
+
}
|
|
3208
|
+
}
|
|
3209
|
+
/**
|
|
3210
|
+
* Stop the MCP server
|
|
3211
|
+
*/
|
|
3212
|
+
async function stopMcpServer(logger) {
|
|
3213
|
+
if (mcpServer) {
|
|
3214
|
+
await mcpServer.stop();
|
|
3215
|
+
mcpServer = null;
|
|
3216
|
+
logger.info(`[${ADDON_NAME}] MCP server stopped`);
|
|
3217
|
+
}
|
|
3218
|
+
}
|
|
3219
|
+
/**
|
|
3220
|
+
* Register IPC handlers for renderer communication
|
|
3221
|
+
*/
|
|
3222
|
+
function registerIpcHandlers(services, logger) {
|
|
3223
|
+
// Get MCP server status
|
|
3224
|
+
electron_1.ipcMain.handle('mcp:getStatus', async () => {
|
|
3225
|
+
if (!mcpServer) {
|
|
3226
|
+
return { running: false, port: 0, uptime: 0 };
|
|
3227
|
+
}
|
|
3228
|
+
return mcpServer.getStatus();
|
|
3229
|
+
});
|
|
3230
|
+
// Get connection info
|
|
3231
|
+
electron_1.ipcMain.handle('mcp:getConnectionInfo', async () => {
|
|
3232
|
+
if (!mcpServer) {
|
|
3233
|
+
return null;
|
|
3234
|
+
}
|
|
3235
|
+
return mcpServer.getConnectionInfo();
|
|
3236
|
+
});
|
|
3237
|
+
// Start MCP server
|
|
3238
|
+
electron_1.ipcMain.handle('mcp:start', async () => {
|
|
3239
|
+
try {
|
|
3240
|
+
await startMcpServer(services, logger);
|
|
3241
|
+
return { success: true };
|
|
3242
|
+
}
|
|
3243
|
+
catch (error) {
|
|
3244
|
+
return { success: false, error: error.message };
|
|
3245
|
+
}
|
|
3246
|
+
});
|
|
3247
|
+
// Stop MCP server
|
|
3248
|
+
electron_1.ipcMain.handle('mcp:stop', async () => {
|
|
3249
|
+
try {
|
|
3250
|
+
await stopMcpServer(logger);
|
|
3251
|
+
return { success: true };
|
|
3252
|
+
}
|
|
3253
|
+
catch (error) {
|
|
3254
|
+
return { success: false, error: error.message };
|
|
3255
|
+
}
|
|
3256
|
+
});
|
|
3257
|
+
// Restart MCP server
|
|
3258
|
+
electron_1.ipcMain.handle('mcp:restart', async () => {
|
|
3259
|
+
try {
|
|
3260
|
+
await stopMcpServer(logger);
|
|
3261
|
+
await startMcpServer(services, logger);
|
|
3262
|
+
return { success: true };
|
|
3263
|
+
}
|
|
3264
|
+
catch (error) {
|
|
3265
|
+
return { success: false, error: error.message };
|
|
3266
|
+
}
|
|
3267
|
+
});
|
|
3268
|
+
// Regenerate auth token
|
|
3269
|
+
electron_1.ipcMain.handle('mcp:regenerateToken', async () => {
|
|
3270
|
+
if (!mcpServer) {
|
|
3271
|
+
return { success: false, error: 'MCP server not running' };
|
|
3272
|
+
}
|
|
3273
|
+
try {
|
|
3274
|
+
const newToken = await mcpServer.regenerateToken();
|
|
3275
|
+
return { success: true, token: newToken };
|
|
3276
|
+
}
|
|
3277
|
+
catch (error) {
|
|
3278
|
+
return { success: false, error: error.message };
|
|
3279
|
+
}
|
|
3280
|
+
});
|
|
3281
|
+
logger.info(`[${ADDON_NAME}] Registered IPC handlers: mcp:getStatus, mcp:getConnectionInfo, mcp:start, mcp:stop, mcp:restart, mcp:regenerateToken`);
|
|
3282
|
+
}
|
|
3283
|
+
/**
|
|
3284
|
+
* Main addon initialization function
|
|
3285
|
+
*/
|
|
3286
|
+
function default_1(_context) {
|
|
3287
|
+
const services = LocalMain.getServiceContainer().cradle;
|
|
3288
|
+
const { localLogger, graphql } = services;
|
|
3289
|
+
try {
|
|
3290
|
+
localLogger.info(`[${ADDON_NAME}] Initializing...`);
|
|
3291
|
+
// Register GraphQL extensions (for local-cli and MCP)
|
|
3292
|
+
const resolvers = createResolvers(services);
|
|
3293
|
+
graphql.registerGraphQLService('mcp-server', typeDefs, resolvers);
|
|
3294
|
+
localLogger.info(`[${ADDON_NAME}] Registered GraphQL: 29 tools (Phase 1-11b)`);
|
|
3295
|
+
// Start MCP server (for AI tools)
|
|
3296
|
+
const localServices = {
|
|
3297
|
+
siteData: services.siteData,
|
|
3298
|
+
siteProcessManager: services.siteProcessManager,
|
|
3299
|
+
wpCli: services.wpCli,
|
|
3300
|
+
deleteSite: services.deleteSite,
|
|
3301
|
+
addSite: services.addSite,
|
|
3302
|
+
localLogger: services.localLogger,
|
|
3303
|
+
adminer: services.adminer,
|
|
3304
|
+
x509Cert: services.x509Cert,
|
|
3305
|
+
siteProvisioner: services.siteProvisioner,
|
|
3306
|
+
importSite: services.importSite,
|
|
3307
|
+
lightningServices: services.lightningServices,
|
|
3308
|
+
// Phase 11: WP Engine Connect
|
|
3309
|
+
wpeOAuth: services.wpeOAuth,
|
|
3310
|
+
capi: services.capi,
|
|
3311
|
+
};
|
|
3312
|
+
// MCP server disabled - CLI-only mode
|
|
3313
|
+
// To enable MCP server for AI tool integration, uncomment the following:
|
|
3314
|
+
// startMcpServer(localServices, localLogger);
|
|
3315
|
+
// registerIpcHandlers(localServices, localLogger);
|
|
3316
|
+
localLogger.info(`[${ADDON_NAME}] Successfully initialized (CLI-only mode)`);
|
|
3317
|
+
}
|
|
3318
|
+
catch (error) {
|
|
3319
|
+
localLogger.error(`[${ADDON_NAME}] Failed to initialize:`, error);
|
|
3320
|
+
}
|
|
3321
|
+
}
|
|
3322
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"index.js","sourceRoot":"","sources":["../../src/main/index.ts"],"names":[],"mappings":";AAAA;;;;;;GAMG;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;AAwsHH,4BAuCC;AA7uHD,mEAAqD;AACrD,uCAAmC;AACnC,8DAA8B;AAC9B,+CAA4C;AAC5C,mDAAiD;AAGjD,MAAM,UAAU,GAAG,YAAY,CAAC;AAEhC,IAAI,SAAS,GAAqB,IAAI,CAAC;AAEvC;;GAEG;AACH,MAAM,QAAQ,GAAG,IAAA,qBAAG,EAAA;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;;CA4vBnB,CAAC;AAEF;;GAEG;AACH,SAAS,eAAe,CAAC,QAAa;IACpC,MAAM,EACJ,UAAU,EAAE,iBAAiB,EAC7B,QAAQ,EACR,WAAW,EACX,KAAK,EACL,kBAAkB,EAClB,OAAO,EAAE,cAAc,EACvB,SAAS,EAAE,gBAAgB,EAC3B,UAAU,EAAE,iBAAiB,EAC7B,UAAU,EAAE,iBAAiB,EAC7B,cAAc,EACd,OAAO,EACP,QAAQ,EACR,eAAe,EACf,UAAU,EAAE,iBAAiB,EAC7B,iBAAiB,EACjB,YAAY,EACZ,aAAa,EAAE,oBAAoB;IACnC,8BAA8B;IAC9B,QAAQ,EAAE,eAAe,EACzB,IAAI,EAAE,WAAW;IACjB,2BAA2B;IAC3B,OAAO,EAAE,cAAc,EACvB,OAAO,EAAE,cAAc,EACvB,cAAc,EAAE,qBAAqB,EACrC,cAAc,EAAE,qBAAqB;IACrC,uFAAuF;IACvF,gEAAgE;MACjE,GAAG,QAAQ,CAAC;IAEb,wDAAwD;IACxD,mDAAmD;IACnD,4DAA4D;IAC5D,MAAM,kBAAkB,GAAG,MAAM,CAAC,CAAC,mCAAmC;IACtE,MAAM,mBAAmB,GAAG,KAAK,CAAC,CAAC,kCAAkC;IAErE,MAAM,eAAe,GAAG,KAAK,EAC3B,OAAe,EACf,YAAoB,kBAAkB,EACtC,GAAG,IAAW,EACA,EAAE;QAChB,OAAO,IAAI,OAAO,CAAC,CAAC,OAAO,EAAE,MAAM,EAAE,EAAE;YACrC,MAAM,SAAS,GAAG,IAAI,CAAC,GAAG,EAAE,CAAC;YAC7B,MAAM,MAAM,GAAG,IAAI,CAAC,MAAM,EAAE,CAAC,QAAQ,CAAC,EAAE,CAAC,CAAC,MAAM,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC;YACvD,MAAM,mBAAmB,GAAG,GAAG,OAAO,YAAY,SAAS,IAAI,MAAM,EAAE,CAAC;YACxE,MAAM,iBAAiB,GAAG,GAAG,OAAO,UAAU,SAAS,IAAI,MAAM,EAAE,CAAC;YAEpE,MAAM,cAAc,GAAG,IAAI,CAAC,KAAK,CAAC,SAAS,GAAG,IAAI,CAAC,CAAC;YACpD,MAAM,OAAO,GAAG,UAAU,CAAC,GAAG,EAAE;gBAC9B,kBAAO,CAAC,kBAAkB,CAAC,mBAAmB,CAAC,CAAC;gBAChD,kBAAO,CAAC,kBAAkB,CAAC,iBAAiB,CAAC,CAAC;gBAC9C,MAAM,CAAC,IAAI,KAAK,CAAC,eAAe,OAAO,oBAAoB,cAAc,UAAU,CAAC,CAAC,CAAC;YACxF,CAAC,EAAE,SAAS,CAAC,CAAC;YAEd,kBAAO,CAAC,IAAI,CAAC,mBAAmB,EAAE,CAAC,MAAW,EAAE,MAAW,EAAE,EAAE;gBAC7D,YAAY,CAAC,OAAO,CAAC,CAAC;gBACtB,kBAAO,CAAC,kBAAkB,CAAC,iBAAiB,CAAC,CAAC;gBAC9C,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,sBAAsB,OAAO,EAAE,CAAC,CAAC;gBAChE,OAAO,CAAC,EAAE,MAAM,EAAE,CAAC,CAAC;YACtB,CAAC,CAAC,CAAC;YAEH,kBAAO,CAAC,IAAI,CAAC,iBAAiB,EAAE,CAAC,MAAW,EAAE,KAAU,EAAE,EAAE;gBAC1D,YAAY,CAAC,OAAO,CAAC,CAAC;gBACtB,kBAAO,CAAC,kBAAkB,CAAC,mBAAmB,CAAC,CAAC;gBAChD,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,oBAAoB,OAAO,KAAK,KAAK,EAAE,OAAO,EAAE,CAAC,CAAC;gBAClF,OAAO,CAAC,EAAE,KAAK,EAAE,CAAC,CAAC;YACrB,CAAC,CAAC,CAAC;YAEH,MAAM,SAAS,GAAG;gBAChB,KAAK,EAAE,CAAC,YAAoB,EAAE,IAAS,EAAE,EAAE;oBACzC,kBAAO,CAAC,IAAI,CAAC,YAAY,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;gBACzC,CAAC;gBACD,MAAM,EAAE;oBACN,IAAI,EAAE,CAAC,YAAoB,EAAE,IAAS,EAAE,EAAE;wBACxC,kBAAO,CAAC,IAAI,CAAC,YAAY,EAAE,IAAI,EAAE,IAAI,CAAC,CAAC;oBACzC,CAAC;iBACF;aACF,CAAC;YAEF,MAAM,aAAa,GAAG,EAAE,mBAAmB,EAAE,iBAAiB,EAAE,CAAC;YACjE,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,0BAA0B,OAAO,EAAE,CAAC,CAAC;YACpE,kBAAO,CAAC,IAAI,CAAC,OAAO,EAAE,SAAS,EAAE,aAAa,EAAE,GAAG,IAAI,CAAC,CAAC;QAC3D,CAAC,CAAC,CAAC;IACL,CAAC,CAAC;IAEF,8DAA8D;IAC9D,MAAM,kBAAkB,GAAG,KAAK,IAAkD,EAAE;QAClF,IAAI,CAAC;YACH,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,2BAA2B,EAAE,mBAAmB,CAAC,CAAC;YACvF,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,qBAAqB,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;YAE9E,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;gBACjB,WAAW,CAAC,KAAK,CACf,IAAI,UAAU,qCAAqC,MAAM,CAAC,KAAK,CAAC,OAAO,EAAE,CAC1E,CAAC;gBACF,OAAO,EAAE,CAAC;YACZ,CAAC;YAED,yEAAyE;YACzE,oDAAoD;YACpD,IAAI,SAAS,GAAQ,MAAM,CAAC,MAAM,CAAC;YAEnC,kCAAkC;YAClC,IAAI,SAAS,IAAI,OAAO,SAAS,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;gBAC5E,IAAI,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC;oBACpC,SAAS,GAAG,SAAS,CAAC,MAAM,CAAC;gBAC/B,CAAC;qBAAM,IAAI,SAAS,CAAC,MAAM,IAAI,OAAO,SAAS,CAAC,MAAM,KAAK,QAAQ,EAAE,CAAC;oBACpE,sBAAsB;oBACtB,SAAS,GAAG,SAAS,CAAC,MAAM,CAAC;gBAC/B,CAAC;YACH,CAAC;YAED,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,0BAA0B,IAAI,CAAC,SAAS,CAAC,SAAS,CAAC,EAAE,CAAC,CAAC;YAEtF,IAAI,KAAK,CAAC,OAAO,CAAC,SAAS,CAAC,EAAE,CAAC;gBAC7B,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,SAAS,SAAS,CAAC,MAAM,mBAAmB,CAAC,CAAC;gBAC7E,OAAO,SAAS,CAAC;YACnB,CAAC;YAED,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,mDAAmD,OAAO,SAAS,EAAE,CACpF,CAAC;YACF,OAAO,EAAE,CAAC;QACZ,CAAC;QAAC,OAAO,KAAU,EAAE,CAAC;YACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,qCAAqC,KAAK,CAAC,OAAO,EAAE,CAAC,CAAC;YACtF,OAAO,EAAE,CAAC;QACZ,CAAC;IACH,CAAC,CAAC;IAEF,gCAAgC;IAChC,MAAM,YAAY,GAAG,KAAK,EACxB,OAAY,EACZ,IAAgG,EAChG,EAAE;QACF,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,MAAM,EAAE,WAAW,GAAG,IAAI,EAAE,UAAU,GAAG,IAAI,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;QAEnF,IAAI,CAAC;YACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,wBAAwB,MAAM,CAAC,IAAI,CAAC,GAAG,CAAC,EAAE,CAAC,CAAC;YAE3E,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;YACtC,IAAI,CAAC,IAAI,EAAE,CAAC;gBACV,OAAO;oBACL,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,IAAI;oBACZ,KAAK,EAAE,mBAAmB,MAAM,EAAE;iBACnC,CAAC;YACJ,CAAC;YAED,MAAM,MAAM,GAAG,MAAM,kBAAkB,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;YAC5D,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;gBACzB,OAAO;oBACL,OAAO,EAAE,KAAK;oBACd,MAAM,EAAE,IAAI;oBACZ,KAAK,EAAE,SAAS,IAAI,CAAC,IAAI,0DAA0D,IAAI,CAAC,IAAI,EAAE;iBAC/F,CAAC;YACJ,CAAC;YAED,MAAM,MAAM,GAAG,MAAM,KAAK,CAAC,GAAG,CAAC,IAAI,EAAE,MAAM,EAAE;gBAC3C,WAAW;gBACX,UAAU;gBACV,YAAY,EAAE,KAAK;aACpB,CAAC,CAAC;YAEH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,iCAAiC,CAAC,CAAC;YAElE,OAAO;gBACL,OAAO,EAAE,IAAI;gBACb,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,IAAI,EAAE;gBAC5B,KAAK,EAAE,IAAI;aACZ,CAAC;QACJ,CAAC;QAAC,OAAO,KAAU,EAAE,CAAC;YACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,kBAAkB,EAAE,KAAK,CAAC,CAAC;YAC3D,OAAO;gBACL,OAAO,EAAE,KAAK;gBACd,MAAM,EAAE,IAAI;gBACZ,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;aACxC,CAAC;QACJ,CAAC;IACH,CAAC,CAAC;IAEF,OAAO;QACL,KAAK,EAAE;YACL,UAAU,EAAE,YAAY;YAExB,UAAU,EAAE,KAAK,IAAI,EAAE;gBACrB,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,uBAAuB,CAAC,CAAC;oBAExD,MAAM,cAAc,GAAG,MAAM,iBAAiB,CAAC,aAAa,EAAE,CAAC;oBAE/D,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,UAAU,EAAE,cAAc,CAAC,GAAG,CAAC,CAAC,EAAO,EAAE,EAAE,CAAC,CAAC;4BAC3C,IAAI,EAAE,EAAE,CAAC,IAAI;4BACb,YAAY,EAAE,EAAE,CAAC,YAAY;4BAC7B,4DAA4D;4BAC5D,UAAU,EACR,OAAO,EAAE,CAAC,UAAU,KAAK,QAAQ;gCAC/B,CAAC,CAAC,EAAE,CAAC,UAAU,EAAE,IAAI,IAAI,EAAE,CAAC,UAAU,EAAE,OAAO;gCAC/C,CAAC,CAAC,EAAE,CAAC,UAAU;4BACnB,SAAS,EACP,OAAO,EAAE,CAAC,SAAS,KAAK,QAAQ;gCAC9B,CAAC,CAAC,EAAE,CAAC,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,SAAS,EAAE,IAAI;gCAC1C,CAAC,CAAC,EAAE,CAAC,SAAS;4BAClB,QAAQ,EACN,OAAO,EAAE,CAAC,QAAQ,KAAK,QAAQ;gCAC7B,CAAC,CAAC,EAAE,CAAC,QAAQ,EAAE,IAAI,IAAI,EAAE,CAAC,QAAQ,EAAE,IAAI;gCACxC,CAAC,CAAC,EAAE,CAAC,QAAQ;yBAClB,CAAC,CAAC;qBACJ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,+BAA+B,EAAE,KAAK,CAAC,CAAC;oBACxE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,UAAU,EAAE,EAAE;qBACf,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,YAAY,EAAE,KAAK,EAAE,OAAY,EAAE,IAAuB,EAAE,EAAE;gBAC5D,MAAM,EAAE,IAAI,GAAG,KAAK,EAAE,GAAG,IAAI,CAAC;gBAE9B,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,6BAA6B,IAAI,GAAG,CAAC,CAAC;oBAErE,IAAI,CAAC,iBAAiB,EAAE,CAAC;wBACvB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,kCAAkC;4BACzC,QAAQ,EAAE,EAAE;yBACb,CAAC;oBACJ,CAAC;oBAED,MAAM,OAAO,GAA2B;wBACtC,GAAG,EAAE,KAAK;wBACV,QAAQ,EAAE,OAAO;wBACjB,SAAS,EAAE,OAAO;qBACnB,CAAC;oBAEF,MAAM,UAAU,GAAG,IAAI,KAAK,KAAK,CAAC,CAAC,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;oBAC9D,MAAM,kBAAkB,GAAG,iBAAiB,CAAC,qBAAqB,CAAC,UAAU,CAAC,CAAC;oBAE/E,MAAM,WAAW,GAA2D,EAAE,CAAC;oBAE/E,KAAK,MAAM,CAAC,IAAI,EAAE,QAAQ,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,kBAAkB,CAAC,EAAE,CAAC;wBAClE,KAAK,MAAM,CAAC,OAAO,EAAE,IAAI,CAAC,IAAI,MAAM,CAAC,OAAO,CAAC,QAA+B,CAAC,EAAE,CAAC;4BAC9E,WAAW,CAAC,IAAI,CAAC;gCACf,IAAI;gCACJ,IAAI,EAAE,IAAI,EAAE,IAAI,IAAI,IAAI;gCACxB,OAAO;6BACR,CAAC,CAAC;wBACL,CAAC;oBACH,CAAC;oBAED,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,QAAQ,EAAE,WAAW;qBACtB,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,4BAA4B,EAAE,KAAK,CAAC,CAAC;oBACrE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,QAAQ,EAAE,EAAE;qBACb,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,8BAA8B;YAC9B,SAAS,EAAE,KAAK,IAAI,EAAE;gBACpB,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,4CAA4C,CAAC,CAAC;oBAE7E,IAAI,CAAC,eAAe,EAAE,CAAC;wBACrB,OAAO;4BACL,aAAa,EAAE,KAAK;4BACpB,KAAK,EAAE,IAAI;4BACX,SAAS,EAAE,IAAI;4BACf,WAAW,EAAE,IAAI;4BACjB,WAAW,EAAE,IAAI;4BACjB,KAAK,EAAE,uCAAuC;yBAC/C,CAAC;oBACJ,CAAC;oBAED,mEAAmE;oBACnE,MAAM,WAAW,GAAG,MAAM,eAAe,CAAC,cAAc,EAAE,CAAC;oBAE3D,IAAI,CAAC,WAAW,EAAE,CAAC;wBACjB,OAAO;4BACL,aAAa,EAAE,KAAK;4BACpB,KAAK,EAAE,IAAI;4BACX,SAAS,EAAE,IAAI;4BACf,WAAW,EAAE,IAAI;4BACjB,WAAW,EAAE,IAAI;4BACjB,KAAK,EAAE,IAAI;yBACZ,CAAC;oBACJ,CAAC;oBAED,8CAA8C;oBAC9C,IAAI,KAAK,GAAG,IAAI,CAAC;oBACjB,IAAI,WAAW,EAAE,CAAC;wBAChB,IAAI,CAAC;4BACH,MAAM,WAAW,GAAG,MAAM,WAAW,CAAC,cAAc,EAAE,CAAC;4BACvD,KAAK,GAAG,WAAW,EAAE,KAAK,IAAI,IAAI,CAAC;wBACrC,CAAC;wBAAC,MAAM,CAAC;4BACP,mDAAmD;wBACrD,CAAC;oBACH,CAAC;oBAED,OAAO;wBACL,aAAa,EAAE,IAAI;wBACnB,KAAK;wBACL,SAAS,EAAE,IAAI;wBACf,WAAW,EAAE,IAAI;wBACjB,WAAW,EAAE,IAAI;wBACjB,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,+BAA+B,EAAE,KAAK,CAAC,CAAC;oBACxE,OAAO;wBACL,aAAa,EAAE,KAAK;wBACpB,KAAK,EAAE,IAAI;wBACX,SAAS,EAAE,IAAI;wBACf,WAAW,EAAE,IAAI;wBACjB,WAAW,EAAE,IAAI;wBACjB,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,YAAY,EAAE,KAAK,EAAE,OAAY,EAAE,IAA4B,EAAE,EAAE;gBACjE,MAAM,EAAE,SAAS,EAAE,GAAG,IAAI,CAAC;gBAE3B,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,4BAA4B,SAAS,CAAC,CAAC,CAAC,gBAAgB,SAAS,EAAE,CAAC,CAAC,CAAC,EAAE,EAAE,CACzF,CAAC;oBAEF,IAAI,CAAC,eAAe,EAAE,CAAC;wBACrB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,uCAAuC;4BAC9C,KAAK,EAAE,EAAE;4BACT,KAAK,EAAE,CAAC;yBACT,CAAC;oBACJ,CAAC;oBAED,uDAAuD;oBACvD,MAAM,WAAW,GAAG,MAAM,eAAe,CAAC,cAAc,EAAE,CAAC;oBAC3D,IAAI,CAAC,WAAW,EAAE,CAAC;wBACjB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,+DAA+D;4BACtE,KAAK,EAAE,EAAE;4BACT,KAAK,EAAE,CAAC;yBACT,CAAC;oBACJ,CAAC;oBAED,IAAI,CAAC,WAAW,EAAE,CAAC;wBACjB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,sCAAsC;4BAC7C,KAAK,EAAE,EAAE;4BACT,KAAK,EAAE,CAAC;yBACT,CAAC;oBACJ,CAAC;oBAED,8CAA8C;oBAC9C,MAAM,QAAQ,GAAG,MAAM,WAAW,CAAC,cAAc,EAAE,CAAC;oBAEpD,IAAI,CAAC,QAAQ,EAAE,CAAC;wBACd,OAAO;4BACL,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,IAAI;4BACX,KAAK,EAAE,EAAE;4BACT,KAAK,EAAE,CAAC;yBACT,CAAC;oBACJ,CAAC;oBAED,MAAM,KAAK,GAAG,QAAQ,CAAC,GAAG,CAAC,CAAC,OAAY,EAAE,EAAE,CAAC,CAAC;wBAC5C,EAAE,EAAE,OAAO,CAAC,EAAE;wBACd,IAAI,EAAE,OAAO,CAAC,IAAI;wBAClB,WAAW,EAAE,OAAO,CAAC,WAAW,IAAI,YAAY;wBAChD,UAAU,EAAE,OAAO,CAAC,UAAU,IAAI,IAAI;wBACtC,aAAa,EAAE,OAAO,CAAC,aAAa,IAAI,OAAO,CAAC,KAAK,IAAI,IAAI;wBAC7D,SAAS,EAAE,OAAO,CAAC,SAAS,IAAI,SAAS,IAAI,IAAI;wBACjD,WAAW,EAAE,OAAO,CAAC,WAAW,IAAI,IAAI;wBACxC,QAAQ,EAAE,GAAG,OAAO,CAAC,IAAI,mBAAmB;wBAC5C,QAAQ,EAAE,OAAO,CAAC,IAAI;qBACvB,CAAC,CAAC,CAAC;oBAEJ,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,KAAK;wBACL,KAAK,EAAE,KAAK,CAAC,MAAM;qBACpB,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,6BAA6B,EAAE,KAAK,CAAC,CAAC;oBACtE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,KAAK,EAAE,EAAE;wBACT,KAAK,EAAE,CAAC;qBACT,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,0BAA0B;YAC1B,UAAU,EAAE,KAAK,EAAE,OAAY,EAAE,IAAwB,EAAE,EAAE;gBAC3D,MAAM,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC;gBAExB,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,qCAAqC,MAAM,EAAE,CAAC,CAAC;oBAE9E,yBAAyB;oBACzB,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,MAAM,EAAE,KAAK;4BACb,QAAQ,EAAE,IAAI;4BACd,WAAW,EAAE,EAAE;4BACf,eAAe,EAAE,CAAC;4BAClB,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,mBAAmB,MAAM,EAAE;yBACnC,CAAC;oBACJ,CAAC;oBAED,gCAAgC;oBAChC,MAAM,eAAe,GAAG,IAAI,CAAC,eAAe,IAAI,EAAE,CAAC;oBACnD,MAAM,cAAc,GAAG,eAAe,CAAC,MAAM,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,KAAK,CAAC,CAAC;oBAE9E,IAAI,cAAc,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;wBAChC,OAAO;4BACL,MAAM,EAAE,KAAK;4BACb,QAAQ,EAAE,IAAI,CAAC,IAAI;4BACnB,WAAW,EAAE,EAAE;4BACf,eAAe,EAAE,CAAC;4BAClB,OAAO,EACL,gGAAgG;4BAClG,KAAK,EAAE,IAAI;yBACZ,CAAC;oBACJ,CAAC;oBAED,0EAA0E;oBAC1E,MAAM,WAAW,GAAG,MAAM,OAAO,CAAC,GAAG,CACnC,cAAc,CAAC,GAAG,CAAC,KAAK,EAAE,CAAM,EAAE,EAAE;wBAClC,IAAI,WAAW,GAAG,CAAC,CAAC,YAAY,CAAC,CAAC,kBAAkB;wBACpD,IAAI,SAAS,GAAG,IAAI,CAAC;wBACrB,IAAI,aAAa,GAAG,IAAI,CAAC;wBAEzB,8DAA8D;wBAC9D,qEAAqE;wBACrE,IAAI,WAAW,IAAI,OAAO,WAAW,CAAC,cAAc,KAAK,UAAU,EAAE,CAAC;4BACpE,IAAI,CAAC;gCACH,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,sCAAsC,CAAC,CAAC,YAAY,SAAS,CAAC,CAAC,aAAa,EAAE,CAC7F,CAAC;gCACF,MAAM,QAAQ,GAAG,MAAM,WAAW,CAAC,cAAc,EAAE,CAAC;gCACpD,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,SAAS,QAAQ,EAAE,MAAM,IAAI,CAAC,qBAAqB,CAClE,CAAC;gCACF,IAAI,QAAQ,IAAI,QAAQ,CAAC,MAAM,GAAG,CAAC,EAAE,CAAC;oCACpC,4CAA4C;oCAC5C,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,+BAA+B,IAAI,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC,CAAC,EAAE,IAAI,EAAE,CAAC,CAAC,EAAE,CACpF,CAAC;oCAEF,qEAAqE;oCACrE,0CAA0C;oCAC1C,MAAM,eAAe,GAAG,QAAQ,CAAC,IAAI,CACnC,CAAC,CAAM,EAAE,EAAE,CACT,CAAC,CAAC,IAAI,EAAE,EAAE,KAAK,CAAC,CAAC,YAAY;wCAC7B,CAAC,CAAC,CAAC,CAAC,aAAa,IAAI,CAAC,CAAC,WAAW,KAAK,CAAC,CAAC,aAAa,CAAC,CAC1D,CAAC;oCAEF,IAAI,eAAe,EAAE,CAAC;wCACpB,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,kBAAkB,eAAe,CAAC,IAAI,EAAE,CAAC,CAAC;wCACzE,WAAW,GAAG,eAAe,CAAC,IAAI,CAAC;wCACnC,SAAS,GAAG,oCAAoC,eAAe,CAAC,IAAI,EAAE,CAAC;wCACvE,aAAa;4CACX,eAAe,CAAC,cAAc,IAAI,eAAe,CAAC,KAAK,IAAI,IAAI,CAAC;oCACpE,CAAC;yCAAM,CAAC;wCACN,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,2CAA2C,CAAC,CAAC,YAAY,EAAE,CAC1E,CAAC;oCACJ,CAAC;gCACH,CAAC;4BACH,CAAC;4BAAC,OAAO,CAAM,EAAE,CAAC;gCAChB,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,0CAA0C,CAAC,CAAC,OAAO,EAAE,CACpE,CAAC;4BACJ,CAAC;wBACH,CAAC;6BAAM,CAAC;4BACN,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,+CAA+C,CAAC,CAAC;wBAClF,CAAC;wBAED,OAAO;4BACL,eAAe,EAAE,CAAC,CAAC,YAAY;4BAC/B,WAAW;4BACX,WAAW,EAAE,CAAC,CAAC,aAAa,IAAI,IAAI;4BACpC,SAAS,EAAE,CAAC,CAAC,SAAS,IAAI,IAAI;4BAC9B,SAAS;4BACT,aAAa;yBACd,CAAC;oBACJ,CAAC,CAAC,CACH,CAAC;oBAEF,2DAA2D;oBAC3D,MAAM,YAAY,GAAG;wBACnB,OAAO,EAAE,IAAI;wBACb,OAAO,EAAE,IAAI;wBACb,SAAS,EAAE,CAAC,WAAW,EAAE,cAAc,EAAE,eAAe,CAAC;wBACzD,kBAAkB,EAAE,IAAI;wBACxB,qBAAqB,EAAE,IAAI;qBAC5B,CAAC;oBAEF,OAAO;wBACL,MAAM,EAAE,IAAI;wBACZ,QAAQ,EAAE,IAAI,CAAC,IAAI;wBACnB,WAAW;wBACX,eAAe,EAAE,WAAW,CAAC,MAAM;wBACnC,YAAY;wBACZ,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,2BAA2B,EAAE,KAAK,CAAC,CAAC;oBACpE,OAAO;wBACL,MAAM,EAAE,KAAK;wBACb,QAAQ,EAAE,IAAI;wBACd,WAAW,EAAE,EAAE;wBACf,eAAe,EAAE,CAAC;wBAClB,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,0BAA0B;YAC1B,YAAY,EAAE,KAAK,IAAI,EAAE;gBACvB,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,0BAA0B,CAAC,CAAC;oBAE3D,iDAAiD;oBACjD,MAAM,SAAS,GAAG,MAAM,kBAAkB,EAAE,CAAC;oBAC7C,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,SAAS,SAAS,CAAC,MAAM,mBAAmB,CAAC,CAAC;oBAE7E,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;wBAC3B,OAAO;4BACL,SAAS,EAAE,KAAK;4BAChB,cAAc,EAAE,KAAK;4BACrB,OAAO,EAAE,IAAI;4BACb,WAAW,EAAE,IAAI;4BACjB,OAAO,EACL,6HAA6H;4BAC/H,KAAK,EAAE,IAAI;yBACZ,CAAC;oBACJ,CAAC;oBAED,2CAA2C;oBAC3C,MAAM,eAAe,GAAG,SAAS,CAAC,IAAI,CACpC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,SAAS,IAAI,CAAC,CAAC,IAAI,EAAE,WAAW,EAAE,CAAC,QAAQ,CAAC,SAAS,CAAC,CACrE,CAAC;oBACT,MAAM,cAAc,GAAG,SAAS,CAAC,IAAI,CACnC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,QAAQ,IAAI,CAAC,CAAC,IAAI,EAAE,WAAW,EAAE,CAAC,QAAQ,CAAC,QAAQ,CAAC,CACnE,CAAC;oBAET,MAAM,aAAa,GAAG,eAAe;wBACnC,CAAC,CAAC;4BACE,aAAa,EAAE,IAAI;4BACnB,SAAS,EAAE,eAAe,CAAC,EAAE;4BAC7B,KAAK,EAAE,eAAe,CAAC,KAAK,IAAI,IAAI;yBACrC;wBACH,CAAC,CAAC;4BACE,aAAa,EAAE,KAAK;4BACpB,SAAS,EAAE,IAAqB;4BAChC,KAAK,EAAE,IAAqB;yBAC7B,CAAC;oBAEN,MAAM,iBAAiB,GAAG,cAAc;wBACtC,CAAC,CAAC;4BACE,aAAa,EAAE,IAAI;4BACnB,SAAS,EAAE,cAAc,CAAC,EAAE;4BAC5B,KAAK,EAAE,cAAc,CAAC,KAAK,IAAI,IAAI;yBACpC;wBACH,CAAC,CAAC;4BACE,aAAa,EAAE,KAAK;4BACpB,SAAS,EAAE,IAAqB;4BAChC,KAAK,EAAE,IAAqB;yBAC7B,CAAC;oBAEN,MAAM,WAAW,GAAG,SAAS,CAAC,MAAM,GAAG,CAAC,CAAC;oBAEzC,OAAO;wBACL,SAAS,EAAE,WAAW;wBACtB,cAAc,EAAE,IAAI;wBACpB,OAAO,EAAE,aAAa;wBACtB,WAAW,EAAE,iBAAiB;wBAC9B,OAAO,EAAE,WAAW;4BAClB,CAAC,CAAC,IAAI;4BACN,CAAC,CAAC,6FAA6F;wBACjG,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,kCAAkC,EAAE,KAAK,CAAC,CAAC;oBAC3E,OAAO;wBACL,SAAS,EAAE,KAAK;wBAChB,cAAc,EAAE,KAAK;wBACrB,OAAO,EAAE,IAAI;wBACb,WAAW,EAAE,IAAI;wBACjB,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,WAAW,EAAE,KAAK,EAAE,OAAY,EAAE,IAA0C,EAAE,EAAE;gBAC9E,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,GAAG,IAAI,CAAC;gBAElC,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,8BAA8B,MAAM,SAAS,QAAQ,EAAE,CAAC,CAAC;oBAExF,WAAW;oBACX,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI;4BACd,QAAQ;4BACR,OAAO,EAAE,EAAE;4BACX,KAAK,EAAE,CAAC;4BACR,KAAK,EAAE,mBAAmB,MAAM,EAAE;yBACnC,CAAC;oBACJ,CAAC;oBAED,yCAAyC;oBACzC,MAAM,SAAS,GAAG,MAAM,kBAAkB,EAAE,CAAC;oBAC7C,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;wBAC3B,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI,CAAC,IAAI;4BACnB,QAAQ;4BACR,OAAO,EAAE,EAAE;4BACX,KAAK,EAAE,CAAC;4BACR,KAAK,EACH,sFAAsF;yBACzF,CAAC;oBACJ,CAAC;oBAED,2EAA2E;oBAC3E,MAAM,WAAW,GAA2B,EAAE,WAAW,EAAE,QAAQ,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;oBAC1F,MAAM,UAAU,GAAG,WAAW,CAAC,QAAQ,CAAC,IAAI,QAAQ,CAAC;oBACrD,MAAM,eAAe,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,UAAU,CAAC,CAAC;oBAExE,IAAI,CAAC,eAAe,EAAE,CAAC;wBACrB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI,CAAC,IAAI;4BACnB,QAAQ;4BACR,OAAO,EAAE,EAAE;4BACX,KAAK,EAAE,CAAC;4BACR,KAAK,EAAE,aAAa,QAAQ,gCAAgC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;yBAC3G,CAAC;oBACJ,CAAC;oBAED,2EAA2E;oBAC3E,yFAAyF;oBACzF,oDAAoD;oBACpD,MAAM,MAAM,GAAG,MAAM,eAAe,CAClC,4BAA4B,EAC5B,mBAAmB,EACnB,MAAM,EACN,eAAe,CAAC,EAAE,EAClB,CAAC,CACF,CAAC;oBACF,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,oCAAoC,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAC3E,CAAC;oBAEF,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;wBACjB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI,CAAC,IAAI;4BACnB,QAAQ;4BACR,OAAO,EAAE,EAAE;4BACX,KAAK,EAAE,CAAC;4BACR,KAAK,EAAE,MAAM,CAAC,KAAK,CAAC,OAAO,IAAI,wBAAwB;yBACxD,CAAC;oBACJ,CAAC;oBAED,wDAAwD;oBACxD,IAAI,WAAW,GAAG,MAAM,CAAC,MAAM,CAAC;oBAChC,IAAI,WAAW,IAAI,OAAO,WAAW,KAAK,QAAQ,IAAI,CAAC,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC,EAAE,CAAC;wBAClF,6CAA6C;wBAC7C,IAAI,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC,MAAM,CAAC,EAAE,CAAC;4BACtC,WAAW,GAAG,WAAW,CAAC,MAAM,CAAC;wBACnC,CAAC;6BAAM,IAAI,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC,SAAS,CAAC,EAAE,CAAC;4BAChD,WAAW,GAAG,WAAW,CAAC,SAAS,CAAC;wBACtC,CAAC;6BAAM,IAAI,WAAW,CAAC,MAAM,IAAI,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC,MAAM,CAAC,SAAS,CAAC,EAAE,CAAC;4BAC7E,WAAW,GAAG,WAAW,CAAC,MAAM,CAAC,SAAS,CAAC;wBAC7C,CAAC;oBACH,CAAC;oBAED,MAAM,OAAO,GAAG,KAAK,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC,CAAC,CAAC,WAAW,CAAC,CAAC,CAAC,EAAE,CAAC;oBAC9D,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,eAAe,OAAO,CAAC,MAAM,UAAU,CAAC,CAAC;oBAExE,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,QAAQ,EAAE,IAAI,CAAC,IAAI;wBACnB,QAAQ;wBACR,OAAO,EAAE,OAAO,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC;4BAChC,mFAAmF;4BACnF,kDAAkD;4BAClD,UAAU,EAAE,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,UAAU,IAAI,CAAC,CAAC,QAAQ;4BAChD,SAAS,EAAE,CAAC,CAAC,SAAS,IAAI,CAAC,CAAC,SAAS,IAAI,CAAC,CAAC,SAAS,IAAI,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,OAAO;4BAC3E,IAAI,EACF,CAAC,CAAC,YAAY,EAAE,WAAW,IAAI,CAAC,CAAC,IAAI,IAAI,CAAC,CAAC,WAAW,IAAI,CAAC,CAAC,IAAI,EAAE,WAAW,IAAI,EAAE;4BACrF,UAAU,EAAE,CAAC,CAAC,YAAY,EAAE,IAAI;gCAC9B,CAAC,CAAC,GAAG,CAAC,CAAC,YAAY,CAAC,IAAI,QAAQ;gCAChC,CAAC,CAAC,CAAC,CAAC,UAAU,IAAI,IAAI,CAAC,MAAM;4BAC/B,QAAQ,EAAE,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,YAAY,EAAE,QAAQ,IAAI,CAAC,CAAC,QAAQ,IAAI,EAAE,CAAC;yBACvE,CAAC,CAAC;wBACH,KAAK,EAAE,OAAO,CAAC,MAAM;wBACrB,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,2BAA2B,EAAE,KAAK,CAAC,CAAC;oBACpE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,QAAQ,EAAE,IAAI;wBACd,QAAQ;wBACR,OAAO,EAAE,EAAE;wBACX,KAAK,EAAE,CAAC;wBACR,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,0BAA0B;YAC1B,cAAc,EAAE,KAAK,EAAE,OAAY,EAAE,IAAwC,EAAE,EAAE;gBAC/E,MAAM,EAAE,MAAM,EAAE,KAAK,GAAG,EAAE,EAAE,GAAG,IAAI,CAAC;gBAEpC,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,mCAAmC,MAAM,EAAE,CAAC,CAAC;oBAE5E,+BAA+B;oBAC/B,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI;4BACd,MAAM,EAAE,EAAE;4BACV,KAAK,EAAE,CAAC;4BACR,KAAK,EAAE,mBAAmB,MAAM,EAAE;yBACnC,CAAC;oBACJ,CAAC;oBAED,+CAA+C;oBAC/C,IAAI,CAAC,qBAAqB,IAAI,OAAO,qBAAqB,CAAC,SAAS,KAAK,UAAU,EAAE,CAAC;wBACpF,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI,CAAC,IAAI;4BACnB,MAAM,EAAE,EAAE;4BACV,KAAK,EAAE,CAAC;4BACR,KAAK,EAAE,oCAAoC;yBAC5C,CAAC;oBACJ,CAAC;oBAED,MAAM,MAAM,GAAG,qBAAqB,CAAC,SAAS,CAAC,MAAM,CAAC,CAAC;oBACvD,MAAM,aAAa,GAAG,MAAM,CAAC,KAAK,CAAC,CAAC,EAAE,KAAK,CAAC,CAAC;oBAE7C,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,QAAQ,EAAE,IAAI,CAAC,IAAI;wBACnB,MAAM,EAAE,aAAa,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC;4BACrC,iBAAiB,EAAE,CAAC,CAAC,iBAAiB,IAAI,IAAI;4BAC9C,SAAS,EAAE,CAAC,CAAC,SAAS;4BACtB,WAAW,EAAE,CAAC,CAAC,WAAW;4BAC1B,SAAS,EAAE,CAAC,CAAC,SAAS;4BACtB,MAAM,EAAE,CAAC,CAAC,MAAM,IAAI,IAAI;yBACzB,CAAC,CAAC;wBACH,KAAK,EAAE,aAAa,CAAC,MAAM;wBAC3B,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,+BAA+B,EAAE,KAAK,CAAC,CAAC;oBACxE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,QAAQ,EAAE,IAAI;wBACd,MAAM,EAAE,EAAE;wBACV,KAAK,EAAE,CAAC;wBACR,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,8DAA8D;YAC9D,cAAc,EAAE,KAAK,EAAE,OAAY,EAAE,IAA4C,EAAE,EAAE;gBACnF,MAAM,EAAE,MAAM,EAAE,SAAS,GAAG,MAAM,EAAE,GAAG,IAAI,CAAC;gBAE5C,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,8BAA8B,MAAM,eAAe,SAAS,EAAE,CAC7E,CAAC;oBAEF,qBAAqB;oBACrB,IAAI,SAAS,KAAK,MAAM,IAAI,SAAS,KAAK,MAAM,EAAE,CAAC;wBACjD,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI;4BACd,SAAS;4BACT,KAAK,EAAE,EAAE;4BACT,QAAQ,EAAE,EAAE;4BACZ,OAAO,EAAE,EAAE;4BACX,YAAY,EAAE,CAAC;4BACf,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,8CAA8C;yBACtD,CAAC;oBACJ,CAAC;oBAED,WAAW;oBACX,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI;4BACd,SAAS;4BACT,KAAK,EAAE,EAAE;4BACT,QAAQ,EAAE,EAAE;4BACZ,OAAO,EAAE,EAAE;4BACX,YAAY,EAAE,CAAC;4BACf,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,mBAAmB,MAAM,EAAE;yBACnC,CAAC;oBACJ,CAAC;oBAED,uBAAuB;oBACvB,MAAM,aAAa,GAAG,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,KAAK,CAAC,CAAC;oBACjF,IAAI,CAAC,aAAa,EAAE,CAAC;wBACnB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI,CAAC,IAAI;4BACnB,SAAS;4BACT,KAAK,EAAE,EAAE;4BACT,QAAQ,EAAE,EAAE;4BACZ,OAAO,EAAE,EAAE;4BACX,YAAY,EAAE,CAAC;4BACf,OAAO,EAAE,IAAI;4BACb,KAAK,EACH,+EAA+E;yBAClF,CAAC;oBACJ,CAAC;oBAED,6BAA6B;oBAC7B,IACE,CAAC,qBAAqB;wBACtB,OAAO,qBAAqB,CAAC,iBAAiB,KAAK,UAAU,EAC7D,CAAC;wBACD,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI,CAAC,IAAI;4BACnB,SAAS;4BACT,KAAK,EAAE,EAAE;4BACT,QAAQ,EAAE,EAAE;4BACZ,OAAO,EAAE,EAAE;4BACX,YAAY,EAAE,CAAC;4BACf,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,mCAAmC;yBAC3C,CAAC;oBACJ,CAAC;oBAED,gCAAgC;oBAChC,IAAI,WAAW,GAAG,aAAa,CAAC,YAAY,CAAC;oBAC7C,IAAI,aAAa,GAAG,EAAE,CAAC;oBACvB,IAAI,SAAS,GAAG,EAAE,CAAC;oBAEnB,IAAI,WAAW,IAAI,OAAO,WAAW,CAAC,cAAc,KAAK,UAAU,EAAE,CAAC;wBACpE,MAAM,QAAQ,GAAG,MAAM,WAAW,CAAC,cAAc,EAAE,CAAC;wBACpD,MAAM,eAAe,GAAG,QAAQ,EAAE,IAAI,CACpC,CAAC,CAAM,EAAE,EAAE,CACT,CAAC,CAAC,IAAI,EAAE,EAAE,KAAK,aAAa,CAAC,YAAY;4BACzC,CAAC,CAAC,aAAa,CAAC,aAAa,IAAI,CAAC,CAAC,WAAW,KAAK,aAAa,CAAC,aAAa,CAAC,CAClF,CAAC;wBACF,IAAI,eAAe,EAAE,CAAC;4BACpB,WAAW,GAAG,eAAe,CAAC,IAAI,CAAC;4BACnC,aAAa;gCACX,eAAe,CAAC,cAAc;oCAC9B,eAAe,CAAC,KAAK;oCACrB,GAAG,eAAe,CAAC,IAAI,eAAe,CAAC;4BACzC,SAAS,GAAG,eAAe,CAAC,EAAE,CAAC;wBACjC,CAAC;oBACH,CAAC;oBAED,IAAI,CAAC,aAAa,EAAE,CAAC;wBACnB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI,CAAC,IAAI;4BACnB,SAAS;4BACT,KAAK,EAAE,EAAE;4BACT,QAAQ,EAAE,EAAE;4BACZ,OAAO,EAAE,EAAE;4BACX,YAAY,EAAE,CAAC;4BACf,OAAO,EAAE,IAAI;4BACb,KAAK,EACH,qFAAqF;yBACxF,CAAC;oBACJ,CAAC;oBAED,oDAAoD;oBACpD,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,mCAAmC,WAAW,EAAE,CAAC,CAAC;oBACjF,MAAM,aAAa,GAAG,MAAM,qBAAqB,CAAC,iBAAiB,CAAC;wBAClE,WAAW,EAAE;4BACX,mBAAmB,EAAE,WAAW;4BAChC,iBAAiB,EAAE,SAAS;4BAC5B,cAAc,EAAE,aAAa,CAAC,YAAY;4BAC1C,qBAAqB,EAAE,aAAa;4BACpC,WAAW,EAAE,IAAI,CAAC,EAAE;yBACrB;wBACD,SAAS,EAAE,SAA4B;wBACvC,cAAc,EAAE,KAAK;qBACtB,CAAC,CAAC;oBAEH,qBAAqB;oBACrB,MAAM,KAAK,GAAG,aAAa;yBACxB,MAAM,CACL,CAAC,CAAM,EAAE,EAAE,CACT,CAAC,CAAC,WAAW,KAAK,QAAQ;wBAC1B,CAAC,CAAC,WAAW,KAAK,QAAQ;wBAC1B,CAAC,CAAC,WAAW,KAAK,UAAU,CAC/B;yBACA,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC;wBAChB,IAAI,EAAE,CAAC,CAAC,IAAI;wBACZ,WAAW,EAAE,CAAC,CAAC,WAAW;wBAC1B,IAAI,EAAE,CAAC,CAAC,IAAI;wBACZ,IAAI,EAAE,CAAC,CAAC,IAAI;qBACb,CAAC,CAAC,CAAC;oBAEN,MAAM,QAAQ,GAAG,aAAa;yBAC3B,MAAM,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,KAAK,QAAQ,CAAC;yBAC9C,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC;wBAChB,IAAI,EAAE,CAAC,CAAC,IAAI;wBACZ,WAAW,EAAE,CAAC,CAAC,WAAW;wBAC1B,IAAI,EAAE,CAAC,CAAC,IAAI;wBACZ,IAAI,EAAE,CAAC,CAAC,IAAI;qBACb,CAAC,CAAC,CAAC;oBAEN,MAAM,OAAO,GAAG,aAAa;yBAC1B,MAAM,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,WAAW,KAAK,QAAQ,CAAC;yBAC9C,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC;wBAChB,IAAI,EAAE,CAAC,CAAC,IAAI;wBACZ,WAAW,EAAE,CAAC,CAAC,WAAW;wBAC1B,IAAI,EAAE,CAAC,CAAC,IAAI;wBACZ,IAAI,EAAE,CAAC,CAAC,IAAI;qBACb,CAAC,CAAC,CAAC;oBAEN,MAAM,YAAY,GAAG,KAAK,CAAC,MAAM,GAAG,QAAQ,CAAC,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;oBACrE,MAAM,cAAc,GAAG,SAAS,KAAK,MAAM,CAAC,CAAC,CAAC,aAAa,CAAC,CAAC,CAAC,aAAa,CAAC;oBAE5E,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,QAAQ,EAAE,IAAI,CAAC,IAAI;wBACnB,SAAS;wBACT,KAAK;wBACL,QAAQ;wBACR,OAAO;wBACP,YAAY;wBACZ,OAAO,EACL,YAAY,GAAG,CAAC;4BACd,CAAC,CAAC,GAAG,YAAY,qBAAqB,cAAc,MAAM,KAAK,CAAC,MAAM,WAAW,QAAQ,CAAC,MAAM,cAAc,OAAO,CAAC,MAAM,UAAU;4BACtI,CAAC,CAAC,wBAAwB,cAAc,GAAG;wBAC/C,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,+BAA+B,EAAE,KAAK,CAAC,CAAC;oBACxE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,QAAQ,EAAE,IAAI;wBACd,SAAS;wBACT,KAAK,EAAE,EAAE;wBACT,QAAQ,EAAE,EAAE;wBACZ,OAAO,EAAE,EAAE;wBACX,YAAY,EAAE,CAAC;wBACf,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;SACF;QACD,QAAQ,EAAE;YACR,KAAK,EAAE,YAAY;YAEnB,UAAU,EAAE,KAAK,EACf,OAAY,EACZ,IAWC,EACD,EAAE;gBACF,+BAA+B;gBAC/B,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,kCAAkC,IAAI,CAAC,SAAS,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;gBAEzF,MAAM,EACJ,IAAI,EACJ,UAAU,EACV,SAAS,GAAG,OAAO,EACnB,QAAQ,GAAG,OAAO,EAClB,eAAe,GAAG,OAAO,EACzB,eAAe,GAAG,UAAU,EAC5B,YAAY,GAAG,kBAAkB,EACjC,SAAS,GACV,GAAG,IAAI,CAAC,KAAK,CAAC;gBAEf,iCAAiC;gBACjC,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,0BAA0B,IAAI,gBAAgB,SAAS,uBAAuB,OAAO,SAAS,EAAE,CAC/G,CAAC;gBAEF,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,oBAAoB,IAAI,GAAG,SAAS,CAAC,CAAC,CAAC,oBAAoB,SAAS,EAAE,CAAC,CAAC,CAAC,EAAE,EAAE,CAC5F,CAAC;oBAEF,qCAAqC;oBACrC,MAAM,QAAQ,GAAG,IAAI;yBAClB,WAAW,EAAE;yBACb,OAAO,CAAC,aAAa,EAAE,GAAG,CAAC;yBAC3B,OAAO,CAAC,QAAQ,EAAE,EAAE,CAAC,CAAC;oBACzB,MAAM,UAAU,GAAG,GAAG,QAAQ,QAAQ,CAAC;oBAEvC,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;oBACzB,MAAM,IAAI,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC;oBAC7B,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;oBACzB,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,aAAa,EAAE,QAAQ,CAAC,CAAC;oBAElE,4EAA4E;oBAC5E,IAAI,SAAS,EAAE,CAAC;wBACd,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,mCAAmC,SAAS,EAAE,CAAC,CAAC;wBAE/E,yCAAyC;wBACzC,MAAM,EAAE,GAAG,EAAE,GAAG,OAAO,CAAC,UAAU,CAAC,CAAC;wBACpC,MAAM,YAAY,GAAG,GAAG,CAAC,OAAO,CAAC,UAAU,CAAC,CAAC;wBAC7C,MAAM,gBAAgB,GAAG,IAAI,CAAC,IAAI,CAAC,YAAY,EAAE,YAAY,EAAE,GAAG,SAAS,MAAM,CAAC,CAAC;wBAEnF,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,+BAA+B,gBAAgB,EAAE,CAAC,CAAC;wBAElF,0BAA0B;wBAC1B,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,gBAAgB,CAAC,EAAE,CAAC;4BACrC,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,6BAA6B,gBAAgB,EAAE,CAAC,CAAC;4BACjF,OAAO;gCACL,OAAO,EAAE,KAAK;gCACd,KAAK,EAAE,wBAAwB,SAAS,oDAAoD;gCAC5F,MAAM,EAAE,IAAI;gCACZ,QAAQ,EAAE,IAAI;gCACd,UAAU,EAAE,IAAI;6BACjB,CAAC;wBACJ,CAAC;wBAED,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,yBAAyB,gBAAgB,EAAE,CAAC,CAAC;wBAE5E,kEAAkE;wBAClE,IAAI,aAAkB,CAAC;wBACvB,IAAI,CAAC;4BACH,MAAM,SAAS,GAAG,OAAO,CAAC,iBAAiB,CAAC,CAAC;4BAC7C,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,uCAAuC,CAAC,CAAC;4BAExE,MAAM,GAAG,GAAG,IAAI,SAAS,CAAC,KAAK,CAAC,EAAE,IAAI,EAAE,gBAAgB,EAAE,CAAC,CAAC;4BAC5D,MAAM,OAAO,GAAG,MAAM,GAAG,CAAC,OAAO,EAAE,CAAC;4BACpC,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,gCAAgC,MAAM,CAAC,IAAI,CAAC,OAAO,CAAC,CAAC,MAAM,EAAE,CAC5E,CAAC;4BAEF,MAAM,QAAQ,GAAG,OAAO,CAAC,iBAAiB,CAAC;gCACzC,CAAC,CAAC,iBAAiB;gCACnB,CAAC,CAAC,sBAAsB,CAAC;4BAC3B,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,4BAA4B,QAAQ,EAAE,CAAC,CAAC;4BAEvE,MAAM,IAAI,GAAG,MAAM,GAAG,CAAC,SAAS,CAAC,QAAQ,CAAC,CAAC;4BAC3C,aAAa,GAAG,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,CAAC,CAAC;4BAClD,MAAM,GAAG,CAAC,KAAK,EAAE,CAAC;4BAClB,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,+BAA+B,EAC7C,IAAI,CAAC,SAAS,CAAC,aAAa,CAAC,CAAC,SAAS,CAAC,CAAC,EAAE,GAAG,CAAC,CAChD,CAAC;wBACJ,CAAC;wBAAC,OAAO,QAAa,EAAE,CAAC;4BACvB,WAAW,CAAC,KAAK,CACf,IAAI,UAAU,mCAAmC,QAAQ,CAAC,OAAO,EAAE,EACnE,QAAQ,CACT,CAAC;4BACF,OAAO;gCACL,OAAO,EAAE,KAAK;gCACd,KAAK,EAAE,sCAAsC,QAAQ,CAAC,OAAO,EAAE;gCAC/D,MAAM,EAAE,IAAI;gCACZ,QAAQ,EAAE,IAAI;gCACd,UAAU,EAAE,IAAI;6BACjB,CAAC;wBACJ,CAAC;wBAED,wBAAwB;wBACxB,MAAM,cAAc,GAAQ;4BAC1B,QAAQ,EAAE,IAAI;4BACd,UAAU,EAAE,UAAU;4BACtB,QAAQ,EAAE,QAAQ;4BAClB,GAAG,EAAE,gBAAgB;4BACrB,UAAU,EAAE;gCACV,IAAI,EAAE,iBAAiB;gCACvB,OAAO,EAAE,aAAa;6BACvB;4BACD,WAAW,EAAE,aAAa,CAAC,WAAW,IAAI,UAAU;4BACpD,SAAS,EAAE,SAAS;yBACrB,CAAC;wBAEF,oDAAoD;wBACpD,IAAI,aAAa,CAAC,QAAQ,EAAE,CAAC;4BAC3B,sBAAsB;4BACtB,MAAM,UAAU,GAAG,MAAM,CAAC,MAAM,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC,IAAI,CAC3D,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,KAAK,CACtB,CAAC;4BACT,IAAI,UAAU,EAAE,CAAC;gCACf,cAAc,CAAC,UAAU,GAAG,UAAU,CAAC,OAAO,CAAC;4BACjD,CAAC;4BAED,mBAAmB;4BACnB,MAAM,SAAS,GAAG,MAAM,CAAC,MAAM,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC,IAAI,CAC1D,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,UAAU,IAAI,CAAC,CAAC,IAAI,KAAK,IAAI,CAC9C,CAAC;4BACT,IAAI,SAAS,EAAE,CAAC;gCACd,cAAc,CAAC,QAAQ,GAAG,GAAG,SAAS,CAAC,IAAI,IAAI,SAAS,CAAC,OAAO,EAAE,CAAC;4BACrE,CAAC;4BAED,qBAAqB;4BACrB,MAAM,UAAU,GAAG,MAAM,CAAC,MAAM,CAAC,aAAa,CAAC,QAAQ,CAAC,CAAC,IAAI,CAC3D,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,KAAK,MAAM,IAAI,CAAC,CAAC,IAAI,KAAK,KAAK,CAC3C,CAAC;4BACT,IAAI,UAAU,EAAE,CAAC;gCACf,cAAc,CAAC,SAAS,GAAG,GAAG,UAAU,CAAC,IAAI,IAAI,UAAU,CAAC,OAAO,EAAE,CAAC;4BACxE,CAAC;wBACH,CAAC;6BAAM,IAAI,aAAa,CAAC,UAAU,EAAE,CAAC;4BACpC,cAAc,CAAC,UAAU,GAAG,aAAa,CAAC,UAAU,CAAC;wBACvD,CAAC;wBAED,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,6BAA6B,EAC3C,IAAI,CAAC,SAAS,CAAC,cAAc,CAAC,CAAC,SAAS,CAAC,CAAC,EAAE,GAAG,CAAC,CACjD,CAAC;wBAEF,IAAI,CAAC,iBAAiB,EAAE,CAAC;4BACvB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,uCAAuC,CAAC,CAAC;4BACzE,OAAO;gCACL,OAAO,EAAE,KAAK;gCACd,KAAK,EAAE,8BAA8B;gCACrC,MAAM,EAAE,IAAI;gCACZ,QAAQ,EAAE,IAAI;gCACd,UAAU,EAAE,IAAI;6BACjB,CAAC;wBACJ,CAAC;wBAED,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,sCAAsC,CAAC,CAAC;wBAEvE,qDAAqD;wBACrD,MAAM,YAAY,GAAG,MAAM,iBAAiB,CAAC,GAAG,CAAC,cAAc,CAAC,CAAC;wBAEjE,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,kBAAkB,EAChC,IAAI,CAAC,SAAS,CAAC,YAAY,IAAI,MAAM,CAAC,CAAC,SAAS,CAAC,CAAC,EAAE,GAAG,CAAC,CACzD,CAAC;wBAEF,IAAI,YAAY,IAAI,YAAY,CAAC,EAAE,EAAE,CAAC;4BACpC,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,+CAA+C,IAAI,KAAK,YAAY,CAAC,EAAE,GAAG,CACzF,CAAC;4BACF,OAAO;gCACL,OAAO,EAAE,IAAI;gCACb,KAAK,EAAE,IAAI;gCACX,MAAM,EAAE,YAAY,CAAC,EAAE;gCACvB,QAAQ,EAAE,IAAI;gCACd,UAAU,EAAE,UAAU;6BACvB,CAAC;wBACJ,CAAC;6BAAM,CAAC;4BACN,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,wCAAwC,CAAC,CAAC;4BACzE,OAAO;gCACL,OAAO,EAAE,IAAI;gCACb,KAAK,EAAE,IAAI;gCACX,MAAM,EAAE,IAAI;gCACZ,QAAQ,EAAE,IAAI;gCACd,UAAU,EAAE,UAAU;6BACvB,CAAC;wBACJ,CAAC;oBACH,CAAC;oBAED,qCAAqC;oBACrC,MAAM,WAAW,GAAQ;wBACvB,QAAQ,EAAE,IAAI;wBACd,UAAU,EAAE,UAAU;wBACtB,QAAQ,EAAE,QAAQ;wBAClB,SAAS,EAAE,SAAS;wBACpB,QAAQ,EAAE,QAAQ;qBACnB,CAAC;oBAEF,IAAI,UAAU,EAAE,CAAC;wBACf,WAAW,CAAC,UAAU,GAAG,UAAU,CAAC;oBACtC,CAAC;oBAED,MAAM,aAAa,GAAG;wBACpB,aAAa,EAAE,eAAe;wBAC9B,aAAa,EAAE,eAAe;wBAC9B,UAAU,EAAE,YAAY;qBACzB,CAAC;oBAEF,MAAM,IAAI,GAAG,MAAM,cAAc,CAAC,OAAO,CAAC;wBACxC,WAAW;wBACX,aAAa;wBACb,QAAQ,EAAE,KAAK;qBAChB,CAAC,CAAC;oBAEH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,gCAAgC,IAAI,KAAK,IAAI,CAAC,EAAE,GAAG,CAAC,CAAC;oBAEpF,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,MAAM,EAAE,IAAI,CAAC,EAAE;wBACf,QAAQ,EAAE,IAAI;wBACd,UAAU,EAAE,UAAU;qBACvB,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,0BAA0B,EAAE,KAAK,CAAC,CAAC;oBACnE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,MAAM,EAAE,IAAI;wBACZ,QAAQ,EAAE,IAAI;wBACd,UAAU,EAAE,IAAI;qBACjB,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,UAAU,EAAE,KAAK,EACf,OAAY,EACZ,IAA4E,EAC5E,EAAE;gBACF,MAAM,EAAE,EAAE,EAAE,UAAU,GAAG,IAAI,EAAE,WAAW,GAAG,IAAI,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBAEjE,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,oBAAoB,EAAE,EAAE,CAAC,CAAC;oBAEzD,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,EAAE,CAAC,CAAC;oBAClC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mBAAmB,EAAE,EAAE;4BAC9B,MAAM,EAAE,EAAE;yBACX,CAAC;oBACJ,CAAC;oBAED,MAAM,iBAAiB,CAAC,UAAU,CAAC;wBACjC,IAAI;wBACJ,UAAU;wBACV,WAAW;qBACZ,CAAC,CAAC;oBAEH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,gCAAgC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;oBAE5E,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,MAAM,EAAE,EAAE;qBACX,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,0BAA0B,EAAE,KAAK,CAAC,CAAC;oBACnE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,MAAM,EAAE,EAAE;qBACX,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,WAAW,EAAE,KAAK,EAAE,OAAY,EAAE,IAA6C,EAAE,EAAE;gBACjF,MAAM,EAAE,GAAG,EAAE,UAAU,GAAG,IAAI,EAAE,GAAG,IAAI,CAAC;gBAExC,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,cAAc,GAAG,CAAC,MAAM,QAAQ,CAAC,CAAC;oBAEjE,MAAM,iBAAiB,CAAC,WAAW,CAAC;wBAClC,OAAO,EAAE,GAAG;wBACZ,UAAU;wBACV,WAAW,EAAE,IAAI;qBAClB,CAAC,CAAC;oBAEH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,0BAA0B,GAAG,CAAC,MAAM,QAAQ,CAAC,CAAC;oBAE7E,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,MAAM,EAAE,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC;qBACtB,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,2BAA2B,EAAE,KAAK,CAAC,CAAC;oBACpE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,MAAM,EAAE,GAAG,CAAC,IAAI,CAAC,GAAG,CAAC;qBACtB,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,QAAQ,EAAE,KAAK,EAAE,OAAY,EAAE,IAAkD,EAAE,EAAE;gBACnF,MAAM,EAAE,MAAM,EAAE,IAAI,GAAG,GAAG,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBAE1C,IAAI,CAAC;oBACH,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mBAAmB,MAAM,EAAE;4BAClC,GAAG,EAAE,IAAI;yBACV,CAAC;oBACJ,CAAC;oBAED,2BAA2B;oBAC3B,MAAM,MAAM,GAAG,MAAM,kBAAkB,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;oBAC5D,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;wBACzB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,SAAS,IAAI,CAAC,IAAI,uDAAuD;4BAChF,GAAG,EAAE,IAAI;yBACV,CAAC;oBACJ,CAAC;oBAED,MAAM,QAAQ,GAAG,IAAI,CAAC,SAAS,CAAC,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,MAAM,CAAC;oBACnD,MAAM,GAAG,GAAG,GAAG,QAAQ,MAAM,IAAI,CAAC,MAAM,GAAG,IAAI,EAAE,CAAC;oBAElD,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,8BAA8B,GAAG,EAAE,CAAC,CAAC;oBAEpE,IAAI,cAAc,EAAE,CAAC;wBACnB,MAAM,cAAc,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC;oBAC1C,CAAC;yBAAM,CAAC;wBACN,iCAAiC;wBACjC,MAAM,EAAE,KAAK,EAAE,GAAG,OAAO,CAAC,UAAU,CAAC,CAAC;wBACtC,MAAM,KAAK,CAAC,YAAY,CAAC,GAAG,CAAC,CAAC;oBAChC,CAAC;oBAED,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,GAAG;qBACJ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,wBAAwB,EAAE,KAAK,CAAC,CAAC;oBACjE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,GAAG,EAAE,IAAI;qBACV,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,SAAS,EAAE,KAAK,EAAE,OAAY,EAAE,IAAoD,EAAE,EAAE;gBACtF,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBAEvC,IAAI,CAAC;oBACH,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mBAAmB,MAAM,EAAE;4BAClC,SAAS,EAAE,IAAI;4BACf,WAAW,EAAE,IAAI;4BACjB,aAAa,EAAE,IAAI;yBACpB,CAAC;oBACJ,CAAC;oBAED,yDAAyD;oBACzD,MAAM,MAAM,GAAG,MAAM,kBAAkB,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;oBAC5D,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;wBACzB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,SAAS,IAAI,CAAC,IAAI,6CAA6C;4BACtE,SAAS,EAAE,IAAI;4BACf,WAAW,EAAE,IAAI;4BACjB,aAAa,EAAE,IAAI;yBACpB,CAAC;oBACJ,CAAC;oBAED,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,kBAAkB,IAAI,CAAC,IAAI,OAAO,OAAO,EAAE,CAAC,CAAC;oBAE5E,MAAM,OAAO,GAAG,MAAM,gBAAgB,CAAC,SAAS,CAAC;wBAC/C,IAAI;wBACJ,WAAW,EAAE,OAAO;qBACrB,CAAC,CAAC;oBAEH,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,+BAA+B,OAAO,CAAC,IAAI,KAAK,OAAO,CAAC,EAAE,GAAG,CAC5E,CAAC;oBAEF,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,SAAS,EAAE,OAAO,CAAC,EAAE;wBACrB,WAAW,EAAE,OAAO,CAAC,IAAI;wBACzB,aAAa,EAAE,OAAO,CAAC,MAAM;qBAC9B,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,yBAAyB,EAAE,KAAK,CAAC,CAAC;oBAClE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,SAAS,EAAE,IAAI;wBACf,WAAW,EAAE,IAAI;wBACjB,aAAa,EAAE,IAAI;qBACpB,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,UAAU,EAAE,KAAK,EACf,OAAY,EACZ,IAAwD,EACxD,EAAE;gBACF,MAAM,EAAE,MAAM,EAAE,UAAU,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBAC1C,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;gBACzB,MAAM,IAAI,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC;gBAE7B,IAAI,CAAC;oBACH,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mBAAmB,MAAM,EAAE;4BAClC,UAAU,EAAE,IAAI;yBACjB,CAAC;oBACJ,CAAC;oBAED,wDAAwD;oBACxD,MAAM,MAAM,GAAG,MAAM,kBAAkB,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;oBAC5D,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;wBACzB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,SAAS,IAAI,CAAC,IAAI,8CAA8C;4BACvE,UAAU,EAAE,IAAI;yBACjB,CAAC;oBACJ,CAAC;oBAED,8BAA8B;oBAC9B,MAAM,SAAS,GAAG,UAAU,IAAI,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,OAAO,EAAE,EAAE,WAAW,CAAC,CAAC;oBACrE,MAAM,SAAS,GAAG,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE,CAAC,OAAO,CAAC,OAAO,EAAE,GAAG,CAAC,CAAC,KAAK,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC;oBAC9E,MAAM,QAAQ,GAAG,GAAG,IAAI,CAAC,IAAI,IAAI,SAAS,MAAM,CAAC;oBACjD,MAAM,QAAQ,GAAG,IAAI,CAAC,IAAI,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;oBAEhD,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,oBAAoB,IAAI,CAAC,IAAI,OAAO,QAAQ,EAAE,CAAC,CAAC;oBAE/E,qDAAqD;oBACrD,MAAM,mBAAmB,GAAG,+BAA+B,CAAC;oBAE5D,MAAM,iBAAiB,CAAC,UAAU,CAAC;wBACjC,IAAI;wBACJ,UAAU,EAAE,QAAQ;wBACpB,MAAM,EAAE,mBAAmB;qBAC5B,CAAC,CAAC;oBAEH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,oCAAoC,QAAQ,EAAE,CAAC,CAAC;oBAE/E,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,UAAU,EAAE,QAAQ;qBACrB,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,0BAA0B,EAAE,KAAK,CAAC,CAAC;oBACnE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,UAAU,EAAE,IAAI;qBACjB,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,aAAa,EAAE,KAAK,EAAE,OAAY,EAAE,IAAiD,EAAE,EAAE;gBACvF,MAAM,EAAE,MAAM,EAAE,IAAI,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBAEpC,IAAI,CAAC;oBACH,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mBAAmB,MAAM,EAAE;4BAClC,aAAa,EAAE,IAAI;yBACpB,CAAC;oBACJ,CAAC;oBAED,wDAAwD;oBACxD,MAAM,MAAM,GAAG,MAAM,kBAAkB,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;oBAC5D,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;wBACzB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,SAAS,IAAI,CAAC,IAAI,yDAAyD;4BAClF,aAAa,EAAE,IAAI;yBACpB,CAAC;oBACJ,CAAC;oBAED,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,iBAAiB,IAAI,CAAC,IAAI,kBAAkB,IAAI,EAAE,CAAC,CAAC;oBAEnF,qDAAqD;oBACrD,MAAM,aAAa,GAAG,+BAA+B,CAAC;oBAEtD,MAAM,iBAAiB,CAAC,aAAa,CAAC;wBACpC,IAAI;wBACJ,MAAM;wBACN,MAAM,EAAE,aAAa;qBACtB,CAAC,CAAC;oBAEH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,mCAAmC,IAAI,EAAE,CAAC,CAAC;oBAE1E,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,aAAa,EAAE,IAAI;qBACpB,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,6BAA6B,EAAE,KAAK,CAAC,CAAC;oBACtE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,aAAa,EAAE,IAAI;qBACpB,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,uCAAuC;YACvC,cAAc,EAAE,KAAK,EACnB,OAAY,EACZ,IAAwD,EACxD,EAAE;gBACF,MAAM,EAAE,MAAM,EAAE,UAAU,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBAC1C,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;gBACzB,MAAM,UAAU,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC;gBAEnC,IAAI,CAAC;oBACH,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mBAAmB,MAAM,EAAE;4BAClC,UAAU,EAAE,IAAI;yBACjB,CAAC;oBACJ,CAAC;oBAED,yDAAyD;oBACzD,MAAM,MAAM,GAAG,MAAM,kBAAkB,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;oBAC5D,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;wBACzB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,SAAS,IAAI,CAAC,IAAI,uDAAuD;4BAChF,UAAU,EAAE,IAAI;yBACjB,CAAC;oBACJ,CAAC;oBAED,6CAA6C;oBAC7C,MAAM,WAAW,GAAG,UAAU,CAAC,IAAI,CACjC,EAAE,CAAC,OAAO,EAAE,EACZ,WAAW,EACX,GAAG,IAAI,CAAC,IAAI,CAAC,OAAO,CAAC,aAAa,EAAE,GAAG,CAAC,MAAM,CAC/C,CAAC;oBACF,MAAM,SAAS,GAAG,UAAU,IAAI,WAAW,CAAC;oBAE5C,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,4BAA4B,IAAI,CAAC,IAAI,OAAO,SAAS,EAAE,CAAC,CAAC;oBAExF,mEAAmE;oBACnE,IAAI,CAAC,YAAY,EAAE,CAAC;wBAClB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,gCAAgC;4BACvC,UAAU,EAAE,IAAI;yBACjB,CAAC;oBACJ,CAAC;oBAED,MAAM,YAAY,CAAC,IAAI,CAAC,IAAI,EAAE,SAAS,CAAC,CAAC;oBAEzC,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,wCAAwC,SAAS,EAAE,CAAC,CAAC;oBAEpF,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,UAAU,EAAE,SAAS;qBACtB,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,8BAA8B,EAAE,KAAK,CAAC,CAAC;oBACvE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,UAAU,EAAE,IAAI;qBACjB,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,cAAc,EAAE,KAAK,EACnB,OAAY,EACZ,IAAoD,EACpD,EAAE;gBACF,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBACvC,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;gBAEzB,IAAI,CAAC;oBACH,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mBAAmB,MAAM,EAAE;yBACnC,CAAC;oBACJ,CAAC;oBAED,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;wBAC5B,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,uBAAuB,OAAO,EAAE;yBACxC,CAAC;oBACJ,CAAC;oBAED,yDAAyD;oBACzD,MAAM,MAAM,GAAG,MAAM,kBAAkB,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;oBAC5D,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;wBACzB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,SAAS,IAAI,CAAC,IAAI,uDAAuD;yBACjF,CAAC;oBACJ,CAAC;oBAED,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,4BAA4B,IAAI,CAAC,IAAI,SAAS,OAAO,EAAE,CAAC,CAAC;oBAExF,qEAAqE;oBACrE,IAAI,CAAC,oBAAoB,EAAE,CAAC;wBAC1B,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,uCAAuC;yBAC/C,CAAC;oBACJ,CAAC;oBAED,MAAM,oBAAoB,CAAC,IAAI,EAAE,OAAO,CAAC,CAAC;oBAE1C,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,0CAA0C,OAAO,EAAE,CAAC,CAAC;oBAEpF,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,8BAA8B,EAAE,KAAK,CAAC,CAAC;oBACvE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,WAAW,EAAE,KAAK,EAAE,OAAY,EAAE,IAAmC,EAAE,EAAE;gBACvE,MAAM,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBAE9B,IAAI,CAAC;oBACH,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mBAAmB,MAAM,EAAE;yBACnC,CAAC;oBACJ,CAAC;oBAED,yDAAyD;oBACzD,MAAM,MAAM,GAAG,MAAM,kBAAkB,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;oBAC5D,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;wBACzB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,SAAS,IAAI,CAAC,IAAI,oDAAoD;yBAC9E,CAAC;oBACJ,CAAC;oBAED,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,yBAAyB,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;oBAErE,IAAI,OAAO,EAAE,CAAC;wBACZ,MAAM,OAAO,CAAC,IAAI,CAAC,IAAI,CAAC,CAAC;oBAC3B,CAAC;yBAAM,CAAC;wBACN,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,+BAA+B;yBACvC,CAAC;oBACJ,CAAC;oBAED,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,2BAA2B,EAAE,KAAK,CAAC,CAAC;oBACpE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,QAAQ,EAAE,KAAK,EAAE,OAAY,EAAE,IAAmC,EAAE,EAAE;gBACpE,MAAM,EAAE,MAAM,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBAE9B,IAAI,CAAC;oBACH,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mBAAmB,MAAM,EAAE;yBACnC,CAAC;oBACJ,CAAC;oBAED,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,sBAAsB,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;oBAElE,IAAI,QAAQ,EAAE,CAAC;wBACb,MAAM,QAAQ,CAAC,SAAS,CAAC,IAAI,CAAC,CAAC;oBACjC,CAAC;yBAAM,CAAC;wBACN,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,wCAAwC;yBAChD,CAAC;oBACJ,CAAC;oBAED,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,wBAAwB,EAAE,KAAK,CAAC,CAAC;oBACjE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,aAAa,EAAE,KAAK,EAAE,OAAY,EAAE,IAAoD,EAAE,EAAE;gBAC1F,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBAEvC,IAAI,CAAC;oBACH,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mBAAmB,MAAM,EAAE;4BAClC,OAAO,EAAE,IAAI;yBACd,CAAC;oBACJ,CAAC;oBAED,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,cAAc,IAAI,CAAC,IAAI,OAAO,OAAO,EAAE,CAAC,CAAC;oBAExE,gCAAgC;oBAChC,IAAI,CAAC,IAAI,GAAG,OAAO,CAAC;oBACpB,MAAM,QAAQ,CAAC,UAAU,CAAC,MAAM,EAAE,EAAE,IAAI,EAAE,OAAO,EAAE,CAAC,CAAC;oBAErD,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,mCAAmC,OAAO,EAAE,CAAC,CAAC;oBAE7E,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,OAAO;qBACR,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,0BAA0B,EAAE,KAAK,CAAC,CAAC;oBACnE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,OAAO,EAAE,IAAI;qBACd,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,gBAAgB,EAAE,KAAK,EACrB,OAAY,EACZ,IAAuD,EACvD,EAAE;gBACF,MAAM,EAAE,MAAM,EAAE,UAAU,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBAE1C,IAAI,CAAC;oBACH,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mBAAmB,MAAM,EAAE;4BAClC,UAAU,EAAE,IAAI;yBACjB,CAAC;oBACJ,CAAC;oBAED,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,8BAA8B,IAAI,CAAC,IAAI,OAAO,UAAU,EAAE,CACzE,CAAC;oBAEF,IAAI,eAAe,EAAE,CAAC;wBACpB,MAAM,eAAe,CAAC,WAAW,CAAC,IAAI,EAAE,KAAK,EAAE,UAAU,CAAC,CAAC;oBAC7D,CAAC;yBAAM,CAAC;wBACN,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,wCAAwC;4BAC/C,UAAU,EAAE,IAAI;yBACjB,CAAC;oBACJ,CAAC;oBAED,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,0CAA0C,UAAU,EAAE,CAAC,CAAC;oBAEvF,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,UAAU;qBACX,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,iCAAiC,EAAE,KAAK,CAAC,CAAC;oBAC1E,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,UAAU,EAAE,IAAI;qBACjB,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,UAAU,EAAE,KAAK,EAAE,OAAY,EAAE,IAAuD,EAAE,EAAE;gBAC1F,MAAM,EAAE,OAAO,EAAE,QAAQ,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBACzC,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;gBAEzB,IAAI,CAAC;oBACH,IAAI,CAAC,EAAE,CAAC,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;wBAC5B,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,uBAAuB,OAAO,EAAE;4BACvC,MAAM,EAAE,IAAI;4BACZ,QAAQ,EAAE,IAAI;yBACf,CAAC;oBACJ,CAAC;oBAED,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,yBAAyB,OAAO,EAAE,CAAC,CAAC;oBAEnE,IAAI,CAAC,iBAAiB,EAAE,CAAC;wBACvB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mCAAmC;4BAC1C,MAAM,EAAE,IAAI;4BACZ,QAAQ,EAAE,IAAI;yBACf,CAAC;oBACJ,CAAC;oBAED,MAAM,MAAM,GAAG,MAAM,iBAAiB,CAAC,GAAG,CAAC;wBACzC,OAAO;wBACP,QAAQ,EAAE,QAAQ,IAAI,SAAS;qBAChC,CAAC,CAAC;oBAEH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,iCAAiC,MAAM,CAAC,IAAI,EAAE,CAAC,CAAC;oBAE/E,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,MAAM,EAAE,MAAM,CAAC,EAAE;wBACjB,QAAQ,EAAE,MAAM,CAAC,IAAI;qBACtB,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,0BAA0B,EAAE,KAAK,CAAC,CAAC;oBACnE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,MAAM,EAAE,IAAI;wBACZ,QAAQ,EAAE,IAAI;qBACf,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,0CAA0C;YAC1C,YAAY,EAAE,KAAK,EAAE,OAAY,EAAE,IAAqD,EAAE,EAAE;gBAC1F,MAAM,EAAE,MAAM,EAAE,OAAO,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBAEvC,IAAI,CAAC;oBACH,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mBAAmB,MAAM,EAAE;4BAClC,OAAO,EAAE,IAAI;yBACd,CAAC;oBACJ,CAAC;oBAED,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,KAAK,OAAO,CAAC,CAAC,CAAC,UAAU,CAAC,CAAC,CAAC,WAAW,eAAe,IAAI,CAAC,IAAI,EAAE,CAChF,CAAC;oBAEF,2CAA2C;oBAC3C,MAAM,QAAQ,CAAC,UAAU,CAAC,MAAM,EAAE,EAAE,aAAa,EAAE,OAAO,EAAE,CAAC,CAAC;oBAE9D,uDAAuD;oBACvD,MAAM,MAAM,GAAG,MAAM,kBAAkB,CAAC,aAAa,CAAC,IAAI,CAAC,CAAC;oBAC5D,IAAI,MAAM,KAAK,SAAS,EAAE,CAAC;wBACzB,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,0CAA0C,CAAC,CAAC;wBAC3E,MAAM,kBAAkB,CAAC,OAAO,CAAC,IAAI,CAAC,CAAC;oBACzC,CAAC;oBAED,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,kBAAkB,OAAO,CAAC,CAAC,CAAC,SAAS,CAAC,CAAC,CAAC,UAAU,SAAS,CAC1E,CAAC;oBAEF,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,OAAO;qBACR,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,4BAA4B,EAAE,KAAK,CAAC,CAAC;oBACrE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,OAAO,EAAE,IAAI;qBACd,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,WAAW,EAAE,KAAK,EAChB,OAAY,EACZ,IAAqE,EACrE,EAAE;gBACF,MAAM,EAAE,MAAM,EAAE,OAAO,GAAG,KAAK,EAAE,KAAK,GAAG,GAAG,EAAE,GAAG,IAAI,CAAC,KAAK,CAAC;gBAC5D,MAAM,EAAE,GAAG,OAAO,CAAC,IAAI,CAAC,CAAC;gBACzB,MAAM,UAAU,GAAG,EAAE,CAAC,QAAQ,CAAC;gBAC/B,MAAM,UAAU,GAAG,OAAO,CAAC,MAAM,CAAC,CAAC;gBAEnC,wCAAwC;gBACxC,MAAM,UAAU,GAAG,KAAK,EAAE,QAAgB,EAAoB,EAAE;oBAC9D,IAAI,CAAC;wBACH,MAAM,UAAU,CAAC,MAAM,CAAC,QAAQ,CAAC,CAAC;wBAClC,OAAO,IAAI,CAAC;oBACd,CAAC;oBAAC,MAAM,CAAC;wBACP,OAAO,KAAK,CAAC;oBACf,CAAC;gBACH,CAAC,CAAC;gBAEF,wCAAwC;gBACxC,MAAM,aAAa,GAAG,KAAK,EAAE,QAAgB,EAAE,QAAgB,EAAmB,EAAE;oBAClF,IAAI,CAAC;wBACH,MAAM,OAAO,GAAG,MAAM,UAAU,CAAC,QAAQ,CAAC,QAAQ,EAAE,OAAO,CAAC,CAAC;wBAC7D,MAAM,QAAQ,GAAG,OAAO,CAAC,KAAK,CAAC,IAAI,CAAC,CAAC;wBACrC,OAAO,QAAQ,CAAC,KAAK,CAAC,CAAC,QAAQ,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,SAAS,CAAC;oBAC3D,CAAC;oBAAC,MAAM,CAAC;wBACP,OAAO,EAAE,CAAC;oBACZ,CAAC;gBACH,CAAC,CAAC;gBAEF,IAAI,CAAC;oBACH,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,mBAAmB,MAAM,EAAE;4BAClC,IAAI,EAAE,EAAE;yBACT,CAAC;oBACJ,CAAC;oBAED,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,aAAa,OAAO,aAAa,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;oBAE7E,MAAM,IAAI,GAA2D,EAAE,CAAC;oBACxE,MAAM,OAAO,GAAG,UAAU,CAAC,IAAI,CAAC,IAAI,CAAC,IAAI,EAAE,MAAM,CAAC,CAAC;oBAEnD,MAAM,QAAQ,GAA6B;wBACzC,GAAG,EAAE,CAAC,KAAK,EAAE,SAAS,CAAC;wBACvB,KAAK,EAAE,CAAC,OAAO,CAAC;wBAChB,KAAK,EAAE,CAAC,OAAO,CAAC;wBAChB,GAAG,EAAE,CAAC,KAAK,EAAE,SAAS,EAAE,OAAO,EAAE,OAAO,CAAC;qBAC1C,CAAC;oBAEF,MAAM,UAAU,GAAG,QAAQ,CAAC,OAAO,CAAC,IAAI,QAAQ,CAAC,GAAG,CAAC;oBAErD,KAAK,MAAM,OAAO,IAAI,UAAU,EAAE,CAAC;wBACjC,kCAAkC;wBAClC,KAAK,MAAM,MAAM,IAAI,CAAC,WAAW,EAAE,YAAY,EAAE,MAAM,CAAC,EAAE,CAAC;4BACzD,MAAM,OAAO,GAAG,UAAU,CAAC,IAAI,CAC7B,OAAO,EACP,GAAG,OAAO,GAAG,MAAM,KAAK,MAAM,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,CAAC,GAAG,GAAG,MAAM,EAAE,CACrD,CAAC;4BACF,MAAM,UAAU,GAAG,UAAU,CAAC,IAAI,CAAC,OAAO,EAAE,GAAG,OAAO,GAAG,MAAM,EAAE,CAAC,CAAC;4BAEnE,IAAI,SAAS,GAAkB,IAAI,CAAC;4BACpC,IAAI,MAAM,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;gCAC9B,SAAS,GAAG,OAAO,CAAC;4BACtB,CAAC;iCAAM,IAAI,MAAM,UAAU,CAAC,UAAU,CAAC,EAAE,CAAC;gCACxC,SAAS,GAAG,UAAU,CAAC;4BACzB,CAAC;4BAED,IAAI,SAAS,EAAE,CAAC;gCACd,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,SAAS,EAAE,KAAK,CAAC,CAAC;gCACtD,IAAI,OAAO,EAAE,CAAC;oCACZ,IAAI,CAAC,IAAI,CAAC;wCACR,IAAI,EAAE,OAAO;wCACb,OAAO;wCACP,IAAI,EAAE,SAAS;qCAChB,CAAC,CAAC;gCACL,CAAC;4BACH,CAAC;wBACH,CAAC;oBACH,CAAC;oBAED,IAAI,IAAI,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;wBACtB,4BAA4B;wBAC5B,IAAI,MAAM,UAAU,CAAC,OAAO,CAAC,EAAE,CAAC;4BAC9B,MAAM,OAAO,GAAG,MAAM,UAAU,CAAC,OAAO,CAAC,OAAO,EAAE,EAAE,aAAa,EAAE,IAAI,EAAE,CAAC,CAAC;4BAC3E,KAAK,MAAM,KAAK,IAAI,OAAO,EAAE,CAAC;gCAC5B,IAAI,KAAK,CAAC,WAAW,EAAE,EAAE,CAAC;oCACxB,MAAM,MAAM,GAAG,UAAU,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;oCACpD,MAAM,UAAU,GAAG,MAAM,UAAU,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oCACpD,KAAK,MAAM,OAAO,IAAI,UAAU,EAAE,CAAC;wCACjC,IAAI,OAAO,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;4CAC7B,MAAM,OAAO,GAAG,UAAU,CAAC,IAAI,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;4CACjD,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;4CACpD,IAAI,OAAO,EAAE,CAAC;gDACZ,IAAI,CAAC,IAAI,CAAC;oDACR,IAAI,EAAE,KAAK,CAAC,IAAI;oDAChB,OAAO;oDACP,IAAI,EAAE,OAAO;iDACd,CAAC,CAAC;4CACL,CAAC;wCACH,CAAC;oCACH,CAAC;gCACH,CAAC;qCAAM,IAAI,KAAK,CAAC,IAAI,CAAC,QAAQ,CAAC,MAAM,CAAC,EAAE,CAAC;oCACvC,MAAM,OAAO,GAAG,UAAU,CAAC,IAAI,CAAC,OAAO,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;oCACrD,MAAM,OAAO,GAAG,MAAM,aAAa,CAAC,OAAO,EAAE,KAAK,CAAC,CAAC;oCACpD,IAAI,OAAO,EAAE,CAAC;wCACZ,IAAI,CAAC,IAAI,CAAC;4CACR,IAAI,EAAE,KAAK,CAAC,IAAI,CAAC,OAAO,CAAC,MAAM,EAAE,EAAE,CAAC;4CACpC,OAAO;4CACP,IAAI,EAAE,OAAO;yCACd,CAAC,CAAC;oCACL,CAAC;gCACH,CAAC;4BACH,CAAC;wBACH,CAAC;oBACH,CAAC;oBAED,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,IAAI;qBACL,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,uBAAuB,EAAE,KAAK,CAAC,CAAC;oBAChE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;wBACvC,IAAI,EAAE,EAAE;qBACT,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,mCAAmC;YACnC,YAAY,EAAE,KAAK,EACjB,OAAY,EACZ,IAAyD,EACzD,EAAE;gBACF,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,IAAI,EAAE,GAAG,IAAI,CAAC;gBAExC,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,8BAA8B,MAAM,OAAO,QAAQ,EAAE,CAAC,CAAC;oBAEtF,WAAW;oBACX,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,UAAU,EAAE,IAAI;4BAChB,SAAS,EAAE,IAAI;4BACf,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,mBAAmB,MAAM,EAAE;yBACnC,CAAC;oBACJ,CAAC;oBAED,yCAAyC;oBACzC,MAAM,SAAS,GAAG,MAAM,kBAAkB,EAAE,CAAC;oBAC7C,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;wBAC3B,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,UAAU,EAAE,IAAI;4BAChB,SAAS,EAAE,IAAI;4BACf,OAAO,EAAE,IAAI;4BACb,KAAK,EACH,sFAAsF;yBACzF,CAAC;oBACJ,CAAC;oBAED,2EAA2E;oBAC3E,MAAM,WAAW,GAA2B,EAAE,WAAW,EAAE,QAAQ,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;oBAC1F,MAAM,UAAU,GAAG,WAAW,CAAC,QAAQ,CAAC,IAAI,QAAQ,CAAC;oBACrD,MAAM,eAAe,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,UAAU,CAAC,CAAC;oBAExE,IAAI,CAAC,eAAe,EAAE,CAAC;wBACrB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,UAAU,EAAE,IAAI;4BAChB,SAAS,EAAE,IAAI;4BACf,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,aAAa,QAAQ,gCAAgC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;yBAC3G,CAAC;oBACJ,CAAC;oBAED,iDAAiD;oBACjD,yFAAyF;oBACzF,MAAM,iBAAiB,GAA2B,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;oBAC1F,MAAM,gBAAgB,GAAG,iBAAiB,CAAC,eAAe,CAAC,EAAE,CAAC,IAAI,eAAe,CAAC,EAAE,CAAC;oBACrF,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,+BAA+B,gBAAgB,UAAU,eAAe,CAAC,EAAE,GAAG,CAC7F,CAAC;oBAEF,iEAAiE;oBACjE,MAAM,WAAW,GAAG,IAAI,IAAI,wBAAwB,CAAC;oBACrD,MAAM,MAAM,GAAG,MAAM,eAAe,CAClC,qBAAqB,EACrB,kBAAkB,EAClB,MAAM,EACN,gBAAgB,EAChB,WAAW,CACZ,CAAC;oBACF,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,wBAAwB,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;oBAEjF,gCAAgC;oBAChC,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;wBACjB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,UAAU,EAAE,IAAI;4BAChB,SAAS,EAAE,IAAI;4BACf,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,MAAM,CAAC,KAAK,CAAC,OAAO,IAAI,wBAAwB;yBACxD,CAAC;oBACJ,CAAC;oBAED,yEAAyE;oBACzE,MAAM,YAAY,GAAG,MAAM,CAAC,MAAM,CAAC;oBAEnC,+EAA+E;oBAC/E,IAAI,YAAY,EAAE,KAAK,EAAE,CAAC;wBACxB,MAAM,QAAQ,GACZ,YAAY,CAAC,KAAK,CAAC,OAAO,IAAI,YAAY,CAAC,KAAK,CAAC,QAAQ,EAAE,OAAO,IAAI,eAAe,CAAC;wBACxF,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,UAAU,EAAE,IAAI;4BAChB,SAAS,EAAE,IAAI;4BACf,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,QAAQ;yBAChB,CAAC;oBACJ,CAAC;oBAED,qEAAqE;oBACrE,IAAI,UAAU,GAAG,YAAY,EAAE,UAAU,IAAI,YAAY,EAAE,EAAE,CAAC;oBAC9D,IAAI,CAAC,UAAU,IAAI,YAAY,EAAE,MAAM,EAAE,CAAC;wBACxC,UAAU,GAAG,YAAY,CAAC,MAAM,CAAC,UAAU,IAAI,YAAY,CAAC,MAAM,CAAC,EAAE,CAAC;oBACxE,CAAC;oBAED,oFAAoF;oBACpF,sEAAsE;oBACtE,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,UAAU,EAAE,UAAU,IAAI,IAAI;wBAC9B,SAAS,EAAE,IAAI,IAAI,EAAE,CAAC,WAAW,EAAE;wBACnC,OAAO,EAAE,kCAAkC,eAAe,CAAC,IAAI,EAAE;wBACjE,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,4BAA4B,EAAE,KAAK,CAAC,CAAC;oBACrE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,UAAU,EAAE,IAAI;wBAChB,SAAS,EAAE,IAAI;wBACf,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,aAAa,EAAE,KAAK,EAClB,OAAY,EACZ,IAAiF,EACjF,EAAE;gBACF,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,UAAU,EAAE,OAAO,GAAG,KAAK,EAAE,GAAG,IAAI,CAAC;gBAE/D,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,sBAAsB,UAAU,aAAa,MAAM,EAAE,CAAC,CAAC;oBAEtF,qBAAqB;oBACrB,IAAI,CAAC,OAAO,EAAE,CAAC;wBACb,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EACH,qHAAqH;yBACxH,CAAC;oBACJ,CAAC;oBAED,WAAW;oBACX,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,mBAAmB,MAAM,EAAE;yBACnC,CAAC;oBACJ,CAAC;oBAED,yCAAyC;oBACzC,MAAM,SAAS,GAAG,MAAM,kBAAkB,EAAE,CAAC;oBAC7C,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;wBAC3B,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EACH,sFAAsF;yBACzF,CAAC;oBACJ,CAAC;oBAED,2EAA2E;oBAC3E,MAAM,WAAW,GAA2B,EAAE,WAAW,EAAE,QAAQ,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;oBAC1F,MAAM,UAAU,GAAG,WAAW,CAAC,QAAQ,CAAC,IAAI,QAAQ,CAAC;oBACrD,MAAM,eAAe,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,UAAU,CAAC,CAAC;oBAExE,IAAI,CAAC,eAAe,EAAE,CAAC;wBACrB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,aAAa,QAAQ,gCAAgC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;yBAC3G,CAAC;oBACJ,CAAC;oBAED,iDAAiD;oBACjD,MAAM,iBAAiB,GAA2B,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;oBAC1F,MAAM,gBAAgB,GAAG,iBAAiB,CAAC,eAAe,CAAC,EAAE,CAAC,IAAI,eAAe,CAAC,EAAE,CAAC;oBAErF,mEAAmE;oBACnE,MAAM,MAAM,GAAG,MAAM,eAAe,CAClC,wBAAwB,EACxB,kBAAkB,EAClB,MAAM,EACN,gBAAgB,EAChB,UAAU,CACX,CAAC;oBACF,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,qBAAqB,IAAI,CAAC,SAAS,CAAC,MAAM,CAAC,EAAE,CAAC,CAAC;oBAE9E,uFAAuF;oBACvF,MAAM,QAAQ,GAAG,MAAM,CAAC,KAAK,IAAI,MAAM,CAAC,MAAM,EAAE,KAAK,CAAC;oBACtD,IAAI,QAAQ,EAAE,CAAC;wBACb,MAAM,YAAY,GAChB,OAAO,QAAQ,KAAK,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAAC,CAAC,CAAC,QAAQ,CAAC,OAAO,IAAI,gBAAgB,CAAC;wBACjF,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,YAAY;yBACpB,CAAC;oBACJ,CAAC;oBAED,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,OAAO,EAAE,6BAA6B,UAAU,EAAE;wBAClD,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,6BAA6B,EAAE,KAAK,CAAC,CAAC;oBACtE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,YAAY,EAAE,KAAK,EACjB,OAAY,EACZ,IAAiF,EACjF,EAAE;gBACF,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,UAAU,EAAE,OAAO,GAAG,KAAK,EAAE,GAAG,IAAI,CAAC;gBAE/D,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,qBAAqB,UAAU,aAAa,MAAM,EAAE,CAAC,CAAC;oBAErF,qBAAqB;oBACrB,IAAI,CAAC,OAAO,EAAE,CAAC;wBACb,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,iBAAiB,EAAE,IAAI;4BACvB,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,8DAA8D;yBACtE,CAAC;oBACJ,CAAC;oBAED,WAAW;oBACX,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,iBAAiB,EAAE,IAAI;4BACvB,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,mBAAmB,MAAM,EAAE;yBACnC,CAAC;oBACJ,CAAC;oBAED,yCAAyC;oBACzC,MAAM,SAAS,GAAG,MAAM,kBAAkB,EAAE,CAAC;oBAC7C,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;wBAC3B,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,iBAAiB,EAAE,IAAI;4BACvB,OAAO,EAAE,IAAI;4BACb,KAAK,EACH,sFAAsF;yBACzF,CAAC;oBACJ,CAAC;oBAED,2EAA2E;oBAC3E,MAAM,WAAW,GAA2B,EAAE,WAAW,EAAE,QAAQ,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;oBAC1F,MAAM,UAAU,GAAG,WAAW,CAAC,QAAQ,CAAC,IAAI,QAAQ,CAAC;oBACrD,MAAM,eAAe,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,UAAU,CAAC,CAAC;oBAExE,IAAI,CAAC,eAAe,EAAE,CAAC;wBACrB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,iBAAiB,EAAE,IAAI;4BACvB,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,aAAa,QAAQ,gCAAgC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;yBAC3G,CAAC;oBACJ,CAAC;oBAED,iDAAiD;oBACjD,MAAM,iBAAiB,GAA2B,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;oBAC1F,MAAM,gBAAgB,GAAG,iBAAiB,CAAC,eAAe,CAAC,EAAE,CAAC,IAAI,eAAe,CAAC,EAAE,CAAC;oBAErF,mEAAmE;oBACnE,MAAM,MAAM,GAAG,MAAM,eAAe,CAClC,uBAAuB,EACvB,mBAAmB,EACnB,MAAM,EACN,gBAAgB,EAChB,UAAU,CACX,CAAC;oBAEF,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;wBACjB,+EAA+E;wBAC/E,IAAI,MAAM,CAAC,KAAK,CAAC,OAAO,EAAE,QAAQ,CAAC,WAAW,CAAC,EAAE,CAAC;4BAChD,OAAO;gCACL,OAAO,EAAE,KAAK;gCACd,iBAAiB,EAAE,IAAI;gCACvB,OAAO,EAAE,IAAI;gCACb,KAAK,EACH,+FAA+F;6BAClG,CAAC;wBACJ,CAAC;wBACD,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,iBAAiB,EAAE,IAAI;4BACvB,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,MAAM,CAAC,KAAK,CAAC,OAAO,IAAI,eAAe;yBAC/C,CAAC;oBACJ,CAAC;oBAED,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,iBAAiB,EAAE,UAAU;wBAC7B,OAAO,EAAE,gBAAgB;wBACzB,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,4BAA4B,EAAE,KAAK,CAAC,CAAC;oBACrE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,iBAAiB,EAAE,IAAI;wBACvB,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,cAAc,EAAE,KAAK,EACnB,OAAY,EACZ,IAA8D,EAC9D,EAAE;gBACF,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,UAAU,EAAE,GAAG,IAAI,CAAC;gBAE9C,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,wBAAwB,UAAU,aAAa,MAAM,EAAE,CAAC,CAAC;oBAExF,WAAW;oBACX,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,mBAAmB,MAAM,EAAE;yBACnC,CAAC;oBACJ,CAAC;oBAED,yCAAyC;oBACzC,MAAM,SAAS,GAAG,MAAM,kBAAkB,EAAE,CAAC;oBAC7C,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;wBAC3B,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EACH,sFAAsF;yBACzF,CAAC;oBACJ,CAAC;oBAED,2EAA2E;oBAC3E,MAAM,WAAW,GAA2B,EAAE,WAAW,EAAE,QAAQ,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;oBAC1F,MAAM,UAAU,GAAG,WAAW,CAAC,QAAQ,CAAC,IAAI,QAAQ,CAAC;oBACrD,MAAM,eAAe,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,UAAU,CAAC,CAAC;oBAExE,IAAI,CAAC,eAAe,EAAE,CAAC;wBACrB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,aAAa,QAAQ,gCAAgC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;yBAC3G,CAAC;oBACJ,CAAC;oBAED,iDAAiD;oBACjD,MAAM,iBAAiB,GAA2B,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;oBAC1F,MAAM,gBAAgB,GAAG,iBAAiB,CAAC,eAAe,CAAC,EAAE,CAAC,IAAI,eAAe,CAAC,EAAE,CAAC;oBAErF,kEAAkE;oBAClE,MAAM,MAAM,GAAG,MAAM,eAAe,CAClC,yBAAyB,EACzB,kBAAkB,EAClB,MAAM,EACN,gBAAgB,EAChB,UAAU,CACX,CAAC;oBAEF,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;wBACjB,+EAA+E;wBAC/E,IAAI,MAAM,CAAC,KAAK,CAAC,OAAO,EAAE,QAAQ,CAAC,WAAW,CAAC,EAAE,CAAC;4BAChD,OAAO;gCACL,OAAO,EAAE,KAAK;gCACd,QAAQ,EAAE,IAAI;gCACd,OAAO,EAAE,IAAI;gCACb,KAAK,EACH,mGAAmG;6BACtG,CAAC;wBACJ,CAAC;wBACD,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,QAAQ,EAAE,IAAI;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,MAAM,CAAC,KAAK,CAAC,OAAO,IAAI,iBAAiB;yBACjD,CAAC;oBACJ,CAAC;oBAED,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,QAAQ,EAAE,MAAM,CAAC,MAAM,EAAE,QAAQ,IAAI,IAAI;wBACzC,OAAO,EAAE,uCAAuC;wBAChD,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,8BAA8B,EAAE,KAAK,CAAC,CAAC;oBACvE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,QAAQ,EAAE,IAAI;wBACd,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,cAAc,EAAE,KAAK,EACnB,OAAY,EACZ,IAA4E,EAC5E,EAAE;gBACF,MAAM,EAAE,MAAM,EAAE,QAAQ,EAAE,UAAU,EAAE,IAAI,EAAE,GAAG,IAAI,CAAC;gBAEpD,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,6BAA6B,UAAU,EAAE,CAAC,CAAC;oBAE1E,WAAW;oBACX,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,MAAM,CAAC,CAAC;oBACtC,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,UAAU,EAAE,IAAI;4BAChB,IAAI,EAAE,IAAI;4BACV,KAAK,EAAE,mBAAmB,MAAM,EAAE;yBACnC,CAAC;oBACJ,CAAC;oBAED,yCAAyC;oBACzC,MAAM,SAAS,GAAG,MAAM,kBAAkB,EAAE,CAAC;oBAC7C,IAAI,SAAS,CAAC,MAAM,KAAK,CAAC,EAAE,CAAC;wBAC3B,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,UAAU,EAAE,IAAI;4BAChB,IAAI,EAAE,IAAI;4BACV,KAAK,EACH,sFAAsF;yBACzF,CAAC;oBACJ,CAAC;oBAED,2EAA2E;oBAC3E,MAAM,WAAW,GAA2B,EAAE,WAAW,EAAE,QAAQ,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;oBAC1F,MAAM,UAAU,GAAG,WAAW,CAAC,QAAQ,CAAC,IAAI,QAAQ,CAAC;oBACrD,MAAM,eAAe,GAAG,SAAS,CAAC,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,EAAE,KAAK,UAAU,CAAC,CAAC;oBAExE,IAAI,CAAC,eAAe,EAAE,CAAC;wBACrB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,UAAU,EAAE,IAAI;4BAChB,IAAI,EAAE,IAAI;4BACV,KAAK,EAAE,aAAa,QAAQ,gCAAgC,SAAS,CAAC,GAAG,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,IAAI,CAAC,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE;yBAC3G,CAAC;oBACJ,CAAC;oBAED,iDAAiD;oBACjD,MAAM,iBAAiB,GAA2B,EAAE,MAAM,EAAE,OAAO,EAAE,OAAO,EAAE,SAAS,EAAE,CAAC;oBAC1F,MAAM,gBAAgB,GAAG,iBAAiB,CAAC,eAAe,CAAC,EAAE,CAAC,IAAI,eAAe,CAAC,EAAE,CAAC;oBAErF,6DAA6D;oBAC7D,MAAM,MAAM,GAAG,MAAM,eAAe,CAClC,mBAAmB,EACnB,mBAAmB,EACnB,MAAM,EACN,gBAAgB,EAChB,UAAU,EACV,IAAI,CACL,CAAC;oBAEF,IAAI,MAAM,CAAC,KAAK,EAAE,CAAC;wBACjB,+EAA+E;wBAC/E,IAAI,MAAM,CAAC,KAAK,CAAC,OAAO,EAAE,QAAQ,CAAC,WAAW,CAAC,EAAE,CAAC;4BAChD,OAAO;gCACL,OAAO,EAAE,KAAK;gCACd,UAAU,EAAE,IAAI;gCAChB,IAAI,EAAE,IAAI;gCACV,KAAK,EACH,qGAAqG;6BACxG,CAAC;wBACJ,CAAC;wBACD,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,UAAU,EAAE,IAAI;4BAChB,IAAI,EAAE,IAAI;4BACV,KAAK,EAAE,MAAM,CAAC,KAAK,CAAC,OAAO,IAAI,kBAAkB;yBAClD,CAAC;oBACJ,CAAC;oBAED,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,UAAU;wBACV,IAAI;wBACJ,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,+BAA+B,EAAE,KAAK,CAAC,CAAC;oBACxE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,UAAU,EAAE,IAAI;wBAChB,IAAI,EAAE,IAAI;wBACV,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,8BAA8B;YAC9B,eAAe,EAAE,KAAK,IAAI,EAAE;gBAC1B,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,uCAAuC,CAAC,CAAC;oBAExE,IAAI,CAAC,eAAe,EAAE,CAAC;wBACrB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,KAAK,EAAE,IAAI;4BACX,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,uCAAuC;yBAC/C,CAAC;oBACJ,CAAC;oBAED,+DAA+D;oBAC/D,gDAAgD;oBAChD,MAAM,MAAM,GAAG,MAAM,eAAe,CAAC,YAAY,EAAE,CAAC;oBAEpD,IAAI,MAAM,IAAI,MAAM,CAAC,WAAW,EAAE,CAAC;wBACjC,kCAAkC;wBAClC,IAAI,KAAK,GAAG,IAAI,CAAC;wBACjB,IAAI,WAAW,EAAE,CAAC;4BAChB,IAAI,CAAC;gCACH,MAAM,WAAW,GAAG,MAAM,WAAW,CAAC,cAAc,EAAE,CAAC;gCACvD,KAAK,GAAG,WAAW,EAAE,KAAK,IAAI,IAAI,CAAC;4BACrC,CAAC;4BAAC,MAAM,CAAC;gCACP,0BAA0B;4BAC5B,CAAC;wBACH,CAAC;wBAED,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,wCAAwC,KAAK,CAAC,CAAC,CAAC,OAAO,KAAK,EAAE,CAAC,CAAC,CAAC,EAAE,EAAE,CACpF,CAAC;wBACF,OAAO;4BACL,OAAO,EAAE,IAAI;4BACb,KAAK;4BACL,OAAO,EAAE,2CAA2C;4BACpD,KAAK,EAAE,IAAI;yBACZ,CAAC;oBACJ,CAAC;oBAED,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,IAAI;wBACX,OAAO,EAAE,sEAAsE;wBAC/E,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,8BAA8B,EAAE,KAAK,CAAC,CAAC;oBACvE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,KAAK,EAAE,IAAI;wBACX,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,uBAAuB;qBAChD,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,SAAS,EAAE,KAAK,IAAI,EAAE;gBACpB,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,8BAA8B,CAAC,CAAC;oBAE/D,IAAI,CAAC,eAAe,EAAE,CAAC;wBACrB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,uCAAuC;yBAC/C,CAAC;oBACJ,CAAC;oBAED,qCAAqC;oBACrC,MAAM,eAAe,CAAC,WAAW,EAAE,CAAC;oBAEpC,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,oCAAoC,CAAC,CAAC;oBACrE,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,OAAO,EAAE,2BAA2B;wBACpC,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,sBAAsB,EAAE,KAAK,CAAC,CAAC;oBAC/D,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,eAAe;qBACxC,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,+BAA+B;YAC/B,SAAS,EAAE,KAAK,EACd,OAAY,EACZ,IAKC,EACD,EAAE;gBACF,MAAM,EAAE,WAAW,EAAE,eAAe,EAAE,UAAU,GAAG,KAAK,EAAE,OAAO,GAAG,KAAK,EAAE,GAAG,IAAI,CAAC;gBAEnF,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,uBAAuB,WAAW,YAAY,eAAe,gBAAgB,UAAU,EAAE,CACxG,CAAC;oBAEF,2CAA2C;oBAC3C,IAAI,CAAC,OAAO,EAAE,CAAC;wBACb,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EACH,2FAA2F;yBAC9F,CAAC;oBACJ,CAAC;oBAED,qBAAqB;oBACrB,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;oBAC3C,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,mBAAmB,WAAW,EAAE;yBACxC,CAAC;oBACJ,CAAC;oBAED,8BAA8B;oBAC9B,MAAM,aAAa,GAAG,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,KAAK,CAAC,CAAC;oBACjF,IAAI,CAAC,aAAa,EAAE,CAAC;wBACnB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EACH,+EAA+E;yBAClF,CAAC;oBACJ,CAAC;oBAED,kCAAkC;oBAClC,IAAI,CAAC,cAAc,IAAI,OAAO,cAAc,CAAC,IAAI,KAAK,UAAU,EAAE,CAAC;wBACjE,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,gCAAgC;yBACxC,CAAC;oBACJ,CAAC;oBAED,2DAA2D;oBAC3D,IAAI,WAAW,GAAG,eAAe,CAAC;oBAClC,IAAI,aAAa,GAAG,EAAE,CAAC;oBACvB,IAAI,SAAS,GAAG,EAAE,CAAC;oBAEnB,IAAI,WAAW,IAAI,OAAO,WAAW,CAAC,cAAc,KAAK,UAAU,EAAE,CAAC;wBACpE,MAAM,QAAQ,GAAG,MAAM,WAAW,CAAC,cAAc,EAAE,CAAC;wBACpD,MAAM,eAAe,GAAG,QAAQ,EAAE,IAAI,CACpC,CAAC,CAAM,EAAE,EAAE,CACT,CAAC,CAAC,IAAI,EAAE,EAAE,KAAK,aAAa,CAAC,YAAY;4BACzC,CAAC,CAAC,aAAa,CAAC,aAAa,IAAI,CAAC,CAAC,WAAW,KAAK,aAAa,CAAC,aAAa,CAAC,CAClF,CAAC;wBACF,IAAI,eAAe,EAAE,CAAC;4BACpB,WAAW,GAAG,eAAe,CAAC,IAAI,CAAC;4BACnC,aAAa;gCACX,eAAe,CAAC,cAAc;oCAC9B,eAAe,CAAC,KAAK;oCACrB,GAAG,eAAe,CAAC,IAAI,eAAe,CAAC;4BACzC,SAAS,GAAG,eAAe,CAAC,EAAE,CAAC;wBACjC,CAAC;oBACH,CAAC;oBAED,IAAI,CAAC,aAAa,EAAE,CAAC;wBACnB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EACH,qFAAqF;yBACxF,CAAC;oBACJ,CAAC;oBAED,yDAAyD;oBACzD,cAAc;yBACX,IAAI,CAAC;wBACJ,UAAU;wBACV,mBAAmB,EAAE,WAAW;wBAChC,iBAAiB,EAAE,SAAS;wBAC5B,cAAc,EAAE,aAAa,CAAC,YAAY;wBAC1C,qBAAqB,EAAE,aAAa;wBACpC,WAAW,EAAE,IAAI,CAAC,EAAE;wBACpB,WAAW,EAAE,aAAa,CAAC,aAAa;wBACxC,WAAW,EAAE,KAAK;qBACnB,CAAC;yBACD,KAAK,CAAC,CAAC,GAAQ,EAAE,EAAE;wBAClB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,gBAAgB,EAAE,GAAG,CAAC,CAAC;oBACzD,CAAC,CAAC,CAAC;oBAEL,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,OAAO,EAAE,mBAAmB,WAAW,gCAAgC;wBACvE,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,yBAAyB,EAAE,KAAK,CAAC,CAAC;oBAClE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,sBAAsB;qBAC/C,CAAC;gBACJ,CAAC;YACH,CAAC;YAED,iCAAiC;YACjC,WAAW,EAAE,KAAK,EAChB,OAAY,EACZ,IAIC,EACD,EAAE;gBACF,MAAM,EAAE,WAAW,EAAE,eAAe,EAAE,UAAU,GAAG,KAAK,EAAE,GAAG,IAAI,CAAC;gBAElE,IAAI,CAAC;oBACH,WAAW,CAAC,IAAI,CACd,IAAI,UAAU,yBAAyB,WAAW,YAAY,eAAe,gBAAgB,UAAU,EAAE,CAC1G,CAAC;oBAEF,qBAAqB;oBACrB,MAAM,IAAI,GAAG,QAAQ,CAAC,OAAO,CAAC,WAAW,CAAC,CAAC;oBAC3C,IAAI,CAAC,IAAI,EAAE,CAAC;wBACV,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,mBAAmB,WAAW,EAAE;yBACxC,CAAC;oBACJ,CAAC;oBAED,8BAA8B;oBAC9B,MAAM,aAAa,GAAG,IAAI,CAAC,eAAe,EAAE,IAAI,CAAC,CAAC,CAAM,EAAE,EAAE,CAAC,CAAC,CAAC,MAAM,KAAK,KAAK,CAAC,CAAC;oBACjF,IAAI,CAAC,aAAa,EAAE,CAAC;wBACnB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EACH,+EAA+E;yBAClF,CAAC;oBACJ,CAAC;oBAED,kCAAkC;oBAClC,IAAI,CAAC,cAAc,IAAI,OAAO,cAAc,CAAC,IAAI,KAAK,UAAU,EAAE,CAAC;wBACjE,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EAAE,gCAAgC;yBACxC,CAAC;oBACJ,CAAC;oBAED,gCAAgC;oBAChC,IAAI,WAAW,GAAG,eAAe,CAAC;oBAClC,IAAI,aAAa,GAAG,EAAE,CAAC;oBACvB,IAAI,SAAS,GAAG,EAAE,CAAC;oBAEnB,IAAI,WAAW,IAAI,OAAO,WAAW,CAAC,cAAc,KAAK,UAAU,EAAE,CAAC;wBACpE,MAAM,QAAQ,GAAG,MAAM,WAAW,CAAC,cAAc,EAAE,CAAC;wBACpD,MAAM,eAAe,GAAG,QAAQ,EAAE,IAAI,CACpC,CAAC,CAAM,EAAE,EAAE,CACT,CAAC,CAAC,IAAI,EAAE,EAAE,KAAK,aAAa,CAAC,YAAY;4BACzC,CAAC,CAAC,aAAa,CAAC,aAAa,IAAI,CAAC,CAAC,WAAW,KAAK,aAAa,CAAC,aAAa,CAAC,CAClF,CAAC;wBACF,IAAI,eAAe,EAAE,CAAC;4BACpB,WAAW,GAAG,eAAe,CAAC,IAAI,CAAC;4BACnC,aAAa;gCACX,eAAe,CAAC,cAAc;oCAC9B,eAAe,CAAC,KAAK;oCACrB,GAAG,eAAe,CAAC,IAAI,eAAe,CAAC;4BACzC,SAAS,GAAG,eAAe,CAAC,EAAE,CAAC;wBACjC,CAAC;oBACH,CAAC;oBAED,IAAI,CAAC,aAAa,EAAE,CAAC;wBACnB,OAAO;4BACL,OAAO,EAAE,KAAK;4BACd,OAAO,EAAE,IAAI;4BACb,KAAK,EACH,qFAAqF;yBACxF,CAAC;oBACJ,CAAC;oBAED,yDAAyD;oBACzD,cAAc;yBACX,IAAI,CAAC;wBACJ,UAAU;wBACV,mBAAmB,EAAE,WAAW;wBAChC,iBAAiB,EAAE,SAAS;wBAC5B,cAAc,EAAE,aAAa,CAAC,YAAY;wBAC1C,qBAAqB,EAAE,aAAa;wBACpC,WAAW,EAAE,IAAI,CAAC,EAAE;wBACpB,WAAW,EAAE,aAAa,CAAC,aAAa;wBACxC,WAAW,EAAE,KAAK;qBACnB,CAAC;yBACD,KAAK,CAAC,CAAC,GAAQ,EAAE,EAAE;wBAClB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,gBAAgB,EAAE,GAAG,CAAC,CAAC;oBACzD,CAAC,CAAC,CAAC;oBAEL,OAAO;wBACL,OAAO,EAAE,IAAI;wBACb,OAAO,EAAE,qBAAqB,WAAW,gCAAgC;wBACzE,KAAK,EAAE,IAAI;qBACZ,CAAC;gBACJ,CAAC;gBAAC,OAAO,KAAU,EAAE,CAAC;oBACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,yBAAyB,EAAE,KAAK,CAAC,CAAC;oBAClE,OAAO;wBACL,OAAO,EAAE,KAAK;wBACd,OAAO,EAAE,IAAI;wBACb,KAAK,EAAE,KAAK,CAAC,OAAO,IAAI,sBAAsB;qBAC/C,CAAC;gBACJ,CAAC;YACH,CAAC;SACF;KACF,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,KAAK,UAAU,cAAc,CAAC,QAAuB,EAAE,MAAW;IAChE,IAAI,SAAS,EAAE,CAAC;QACd,MAAM,CAAC,IAAI,CAAC,IAAI,UAAU,8BAA8B,CAAC,CAAC;QAC1D,OAAO;IACT,CAAC;IAED,IAAI,CAAC;QACH,SAAS,GAAG,IAAI,qBAAS,CAAC,EAAE,IAAI,EAAE,sBAAU,CAAC,YAAY,EAAE,EAAE,QAAQ,EAAE,MAAM,CAAC,CAAC;QAE/E,MAAM,SAAS,CAAC,KAAK,EAAE,CAAC;QAExB,MAAM,IAAI,GAAG,SAAS,CAAC,iBAAiB,EAAE,CAAC;QAC3C,MAAM,CAAC,IAAI,CAAC,IAAI,UAAU,gCAAgC,IAAI,CAAC,IAAI,EAAE,CAAC,CAAC;QACvE,MAAM,CAAC,IAAI,CACT,IAAI,UAAU,8FAA8F,CAC7G,CAAC;QACF,MAAM,CAAC,IAAI,CAAC,IAAI,UAAU,sBAAsB,IAAI,CAAC,KAAK,CAAC,IAAI,CAAC,IAAI,CAAC,EAAE,CAAC,CAAC;IAC3E,CAAC;IAAC,OAAO,KAAU,EAAE,CAAC;QACpB,MAAM,CAAC,KAAK,CAAC,IAAI,UAAU,+BAA+B,EAAE,KAAK,CAAC,CAAC;IACrE,CAAC;AACH,CAAC;AAED;;GAEG;AACH,KAAK,UAAU,aAAa,CAAC,MAAW;IACtC,IAAI,SAAS,EAAE,CAAC;QACd,MAAM,SAAS,CAAC,IAAI,EAAE,CAAC;QACvB,SAAS,GAAG,IAAI,CAAC;QACjB,MAAM,CAAC,IAAI,CAAC,IAAI,UAAU,sBAAsB,CAAC,CAAC;IACpD,CAAC;AACH,CAAC;AAED;;GAEG;AACH,SAAS,mBAAmB,CAAC,QAAuB,EAAE,MAAW;IAC/D,wBAAwB;IACxB,kBAAO,CAAC,MAAM,CAAC,eAAe,EAAE,KAAK,IAAI,EAAE;QACzC,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,IAAI,EAAE,CAAC,EAAE,MAAM,EAAE,CAAC,EAAE,CAAC;QAChD,CAAC;QACD,OAAO,SAAS,CAAC,SAAS,EAAE,CAAC;IAC/B,CAAC,CAAC,CAAC;IAEH,sBAAsB;IACtB,kBAAO,CAAC,MAAM,CAAC,uBAAuB,EAAE,KAAK,IAAI,EAAE;QACjD,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,OAAO,IAAI,CAAC;QACd,CAAC;QACD,OAAO,SAAS,CAAC,iBAAiB,EAAE,CAAC;IACvC,CAAC,CAAC,CAAC;IAEH,mBAAmB;IACnB,kBAAO,CAAC,MAAM,CAAC,WAAW,EAAE,KAAK,IAAI,EAAE;QACrC,IAAI,CAAC;YACH,MAAM,cAAc,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;YACvC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC;QAC3B,CAAC;QAAC,OAAO,KAAU,EAAE,CAAC;YACpB,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC,OAAO,EAAE,CAAC;QAClD,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,kBAAkB;IAClB,kBAAO,CAAC,MAAM,CAAC,UAAU,EAAE,KAAK,IAAI,EAAE;QACpC,IAAI,CAAC;YACH,MAAM,aAAa,CAAC,MAAM,CAAC,CAAC;YAC5B,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC;QAC3B,CAAC;QAAC,OAAO,KAAU,EAAE,CAAC;YACpB,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC,OAAO,EAAE,CAAC;QAClD,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,qBAAqB;IACrB,kBAAO,CAAC,MAAM,CAAC,aAAa,EAAE,KAAK,IAAI,EAAE;QACvC,IAAI,CAAC;YACH,MAAM,aAAa,CAAC,MAAM,CAAC,CAAC;YAC5B,MAAM,cAAc,CAAC,QAAQ,EAAE,MAAM,CAAC,CAAC;YACvC,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,CAAC;QAC3B,CAAC;QAAC,OAAO,KAAU,EAAE,CAAC;YACpB,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC,OAAO,EAAE,CAAC;QAClD,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,wBAAwB;IACxB,kBAAO,CAAC,MAAM,CAAC,qBAAqB,EAAE,KAAK,IAAI,EAAE;QAC/C,IAAI,CAAC,SAAS,EAAE,CAAC;YACf,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,wBAAwB,EAAE,CAAC;QAC7D,CAAC;QACD,IAAI,CAAC;YACH,MAAM,QAAQ,GAAG,MAAM,SAAS,CAAC,eAAe,EAAE,CAAC;YACnD,OAAO,EAAE,OAAO,EAAE,IAAI,EAAE,KAAK,EAAE,QAAQ,EAAE,CAAC;QAC5C,CAAC;QAAC,OAAO,KAAU,EAAE,CAAC;YACpB,OAAO,EAAE,OAAO,EAAE,KAAK,EAAE,KAAK,EAAE,KAAK,CAAC,OAAO,EAAE,CAAC;QAClD,CAAC;IACH,CAAC,CAAC,CAAC;IAEH,MAAM,CAAC,IAAI,CACT,IAAI,UAAU,wHAAwH,CACvI,CAAC;AACJ,CAAC;AAED;;GAEG;AACH,mBAAyB,QAAoC;IAC3D,MAAM,QAAQ,GAAG,SAAS,CAAC,mBAAmB,EAAE,CAAC,MAAa,CAAC;IAC/D,MAAM,EAAE,WAAW,EAAE,OAAO,EAAE,GAAG,QAAQ,CAAC;IAE1C,IAAI,CAAC;QACH,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,mBAAmB,CAAC,CAAC;QAEpD,sDAAsD;QACtD,MAAM,SAAS,GAAG,eAAe,CAAC,QAAQ,CAAC,CAAC;QAC5C,OAAO,CAAC,sBAAsB,CAAC,YAAY,EAAE,QAAQ,EAAE,SAAS,CAAC,CAAC;QAClE,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,8CAA8C,CAAC,CAAC;QAE/E,kCAAkC;QAClC,MAAM,aAAa,GAAkB;YACnC,QAAQ,EAAE,QAAQ,CAAC,QAAQ;YAC3B,kBAAkB,EAAE,QAAQ,CAAC,kBAAkB;YAC/C,KAAK,EAAE,QAAQ,CAAC,KAAK;YACrB,UAAU,EAAE,QAAQ,CAAC,UAAU;YAC/B,OAAO,EAAE,QAAQ,CAAC,OAAO;YACzB,WAAW,EAAE,QAAQ,CAAC,WAAW;YACjC,OAAO,EAAE,QAAQ,CAAC,OAAO;YACzB,QAAQ,EAAE,QAAQ,CAAC,QAAQ;YAC3B,eAAe,EAAE,QAAQ,CAAC,eAAe;YACzC,UAAU,EAAE,QAAQ,CAAC,UAAU;YAC/B,iBAAiB,EAAE,QAAQ,CAAC,iBAAiB;YAC7C,8BAA8B;YAC9B,QAAQ,EAAE,QAAQ,CAAC,QAAQ;YAC3B,IAAI,EAAE,QAAQ,CAAC,IAAI;SACpB,CAAC;QAEF,sCAAsC;QACtC,yEAAyE;QACzE,8CAA8C;QAC9C,mDAAmD;QAEnD,WAAW,CAAC,IAAI,CAAC,IAAI,UAAU,4CAA4C,CAAC,CAAC;IAC/E,CAAC;IAAC,OAAO,KAAU,EAAE,CAAC;QACpB,WAAW,CAAC,KAAK,CAAC,IAAI,UAAU,yBAAyB,EAAE,KAAK,CAAC,CAAC;IACpE,CAAC;AACH,CAAC","sourcesContent":["/**\n * CLI Bridge Addon - Main Process Entry Point\n *\n * This addon extends Local's capabilities:\n * - GraphQL mutations for deleteSite, wpCli (for local-cli)\n * - MCP Server for AI tool integration (Claude Code, ChatGPT, etc.)\n */\n\nimport * as LocalMain from '@getflywheel/local/main';\nimport { ipcMain } from 'electron';\nimport gql from 'graphql-tag';\nimport { McpServer } from './mcp/McpServer';\nimport { MCP_SERVER } from '../common/constants';\nimport { LocalServices } from '../common/types';\n\nconst ADDON_NAME = 'MCP Server';\n\nlet mcpServer: McpServer | null = null;\n\n/**\n * GraphQL type definitions for CLI Bridge\n */\nconst typeDefs = gql`\n  input DeleteSiteInput {\n    \"The site ID to delete\"\n    id: ID!\n    \"Whether to move site files to trash (true) or just remove from Local (false)\"\n    trashFiles: Boolean = true\n    \"Whether to update the hosts file\"\n    updateHosts: Boolean = true\n  }\n\n  type DeleteSiteResult {\n    \"Whether the deletion was successful\"\n    success: Boolean!\n    \"Error message if deletion failed\"\n    error: String\n    \"The ID of the deleted site\"\n    siteId: ID\n  }\n\n  input WpCliInput {\n    \"The site ID to run WP-CLI against\"\n    siteId: ID!\n    \"WP-CLI command and arguments (e.g., ['plugin', 'list', '--format=json'])\"\n    args: [String!]!\n    \"Skip loading plugins (default: true)\"\n    skipPlugins: Boolean = true\n    \"Skip loading themes (default: true)\"\n    skipThemes: Boolean = true\n  }\n\n  type WpCliResult {\n    \"Whether the command executed successfully\"\n    success: Boolean!\n    \"Command output (stdout)\"\n    output: String\n    \"Error message if command failed\"\n    error: String\n  }\n\n  input CreateSiteInput {\n    \"Site name (required)\"\n    name: String!\n    \"PHP version (e.g., '8.2.10'). Uses Local default if not specified.\"\n    phpVersion: String\n    \"Web server type\"\n    webServer: String\n    \"Database type\"\n    database: String\n    \"WordPress admin username (default: admin)\"\n    wpAdminUsername: String\n    \"WordPress admin password (default: password)\"\n    wpAdminPassword: String\n    \"WordPress admin email (default: admin@local.test)\"\n    wpAdminEmail: String\n    \"Blueprint name to create site from. Use list_blueprints to see available blueprints.\"\n    blueprint: String\n  }\n\n  type CreateSiteResult {\n    \"Whether site creation was initiated successfully\"\n    success: Boolean!\n    \"Error message if creation failed\"\n    error: String\n    \"The created site ID\"\n    siteId: ID\n    \"The site name\"\n    siteName: String\n    \"The site domain\"\n    siteDomain: String\n  }\n\n  input OpenSiteInput {\n    \"The site ID to open\"\n    siteId: ID!\n    \"Path to open (default: /, use /wp-admin for admin)\"\n    path: String = \"/\"\n  }\n\n  type OpenSiteResult {\n    \"Whether the site was opened successfully\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n    \"The URL that was opened\"\n    url: String\n  }\n\n  input CloneSiteInput {\n    \"The site ID to clone\"\n    siteId: ID!\n    \"Name for the cloned site\"\n    newName: String!\n  }\n\n  type CloneSiteResult {\n    \"Whether cloning was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n    \"The new site ID\"\n    newSiteId: ID\n    \"The new site name\"\n    newSiteName: String\n    \"The new site domain\"\n    newSiteDomain: String\n  }\n\n  input ExportSiteInput {\n    \"The site ID to export\"\n    siteId: ID!\n    \"Output directory path (default: ~/Downloads)\"\n    outputPath: String\n  }\n\n  type ExportSiteResult {\n    \"Whether export was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n    \"Path to the exported zip file\"\n    exportPath: String\n  }\n\n  type Blueprint {\n    \"Blueprint name\"\n    name: String!\n    \"Last modified date\"\n    lastModified: String\n    \"PHP version\"\n    phpVersion: String\n    \"Web server type\"\n    webServer: String\n    \"Database type\"\n    database: String\n  }\n\n  type BlueprintsResult {\n    \"Whether query was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n    \"List of blueprints\"\n    blueprints: [Blueprint!]\n  }\n\n  input SaveBlueprintInput {\n    \"The site ID to save as blueprint\"\n    siteId: ID!\n    \"Name for the blueprint\"\n    name: String!\n  }\n\n  type SaveBlueprintResult {\n    \"Whether save was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n    \"The blueprint name\"\n    blueprintName: String\n  }\n\n  # Phase 8: WordPress Development Tools\n  input ExportDatabaseInput {\n    \"The site ID\"\n    siteId: ID!\n    \"Output file path (optional, defaults to ~/Downloads/<site-name>.sql)\"\n    outputPath: String\n  }\n\n  type ExportDatabaseResult {\n    \"Whether export was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n    \"Path to the exported SQL file\"\n    outputPath: String\n  }\n\n  input ImportDatabaseInput {\n    \"The site ID\"\n    siteId: ID!\n    \"Path to the SQL file to import\"\n    sqlPath: String!\n  }\n\n  type ImportDatabaseResult {\n    \"Whether import was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n  }\n\n  input OpenAdminerInput {\n    \"The site ID\"\n    siteId: ID!\n  }\n\n  type OpenAdminerResult {\n    \"Whether opening was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n  }\n\n  input TrustSslInput {\n    \"The site ID\"\n    siteId: ID!\n  }\n\n  type TrustSslResult {\n    \"Whether trust was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n  }\n\n  input McpRenameSiteInput {\n    \"The site ID\"\n    siteId: ID!\n    \"New name for the site\"\n    newName: String!\n  }\n\n  type McpRenameSiteResult {\n    \"Whether rename was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n    \"The new name\"\n    newName: String\n  }\n\n  input ChangePhpVersionInput {\n    \"The site ID\"\n    siteId: ID!\n    \"Target PHP version\"\n    phpVersion: String!\n  }\n\n  type ChangePhpVersionResult {\n    \"Whether change was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n    \"The new PHP version\"\n    phpVersion: String\n  }\n\n  input ImportSiteInput {\n    \"Path to the zip file to import\"\n    zipPath: String!\n    \"Name for the imported site (optional)\"\n    siteName: String\n  }\n\n  type ImportSiteResult {\n    \"Whether import was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n    \"The imported site ID\"\n    siteId: ID\n    \"The imported site name\"\n    siteName: String\n  }\n\n  # Phase 9: Site Configuration & Dev Tools\n  input ToggleXdebugInput {\n    \"The site ID\"\n    siteId: ID!\n    \"Whether to enable or disable Xdebug\"\n    enabled: Boolean!\n  }\n\n  type ToggleXdebugResult {\n    \"Whether toggle was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n    \"Current Xdebug state\"\n    enabled: Boolean\n  }\n\n  input GetSiteLogsInput {\n    \"The site ID\"\n    siteId: ID!\n    \"Type of logs to retrieve (php, nginx, mysql, all)\"\n    logType: String = \"php\"\n    \"Number of lines to return\"\n    lines: Int = 100\n  }\n\n  type LogEntry {\n    \"Log type\"\n    type: String!\n    \"Log content\"\n    content: String!\n    \"Log file path\"\n    path: String!\n  }\n\n  type GetSiteLogsResult {\n    \"Whether retrieval was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n    \"Log entries\"\n    logs: [LogEntry!]\n  }\n\n  type ServiceInfo {\n    \"Service role (php, database, webserver)\"\n    role: String!\n    \"Service name\"\n    name: String!\n    \"Service version\"\n    version: String!\n  }\n\n  type ListServicesResult {\n    \"Whether listing was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n    \"Available services\"\n    services: [ServiceInfo!]\n  }\n\n  extend type Mutation {\n    \"Create a new WordPress site with full WordPress installation\"\n    createSite(input: CreateSiteInput!): CreateSiteResult!\n\n    \"Delete a site from Local\"\n    deleteSite(input: DeleteSiteInput!): DeleteSiteResult!\n\n    \"Delete multiple sites from Local\"\n    deleteSites(ids: [ID!]!, trashFiles: Boolean = true): DeleteSiteResult!\n\n    \"Run a WP-CLI command against a site\"\n    wpCli(input: WpCliInput!): WpCliResult!\n\n    \"Open a site in the default browser\"\n    openSite(input: OpenSiteInput!): OpenSiteResult!\n\n    \"Clone an existing site\"\n    cloneSite(input: CloneSiteInput!): CloneSiteResult!\n\n    \"Export a site to a zip file\"\n    exportSite(input: ExportSiteInput!): ExportSiteResult!\n\n    \"Save a site as a blueprint\"\n    saveBlueprint(input: SaveBlueprintInput!): SaveBlueprintResult!\n\n    # Phase 8: WordPress Development Tools\n    \"Export site database to SQL file\"\n    exportDatabase(input: ExportDatabaseInput!): ExportDatabaseResult!\n\n    \"Import SQL file into site database\"\n    importDatabase(input: ImportDatabaseInput!): ImportDatabaseResult!\n\n    \"Open Adminer database management UI\"\n    openAdminer(input: OpenAdminerInput!): OpenAdminerResult!\n\n    \"Trust site SSL certificate\"\n    trustSsl(input: TrustSslInput!): TrustSslResult!\n\n    \"Rename a site (MCP version)\"\n    mcpRenameSite(input: McpRenameSiteInput!): McpRenameSiteResult!\n\n    \"Change site PHP version\"\n    changePhpVersion(input: ChangePhpVersionInput!): ChangePhpVersionResult!\n\n    \"Import site from zip file\"\n    importSite(input: ImportSiteInput!): ImportSiteResult!\n\n    # Phase 9: Site Configuration & Dev Tools\n    \"Toggle Xdebug for a site\"\n    toggleXdebug(input: ToggleXdebugInput!): ToggleXdebugResult!\n\n    \"Get site log files\"\n    getSiteLogs(input: GetSiteLogsInput!): GetSiteLogsResult!\n  }\n\n  extend type Query {\n    \"Run a WP-CLI command against a site (read-only operations)\"\n    wpCliQuery(input: WpCliInput!): WpCliResult!\n\n    \"List all available blueprints\"\n    blueprints: BlueprintsResult!\n\n    \"List available service versions\"\n    listServices(type: String): ListServicesResult!\n\n    # Phase 11: WP Engine Connect\n    \"Check WP Engine authentication status\"\n    wpeStatus: WpeAuthStatus!\n\n    \"List all sites from WP Engine account\"\n    listWpeSites(accountId: String): ListWpeSitesResult!\n\n    # Phase 11b: Site Linking\n    \"Get WP Engine connection details for a local site\"\n    getWpeLink(siteId: ID!): GetWpeLinkResult!\n  }\n\n  # Phase 11: WP Engine Connect Types\n  type WpeAuthStatus {\n    \"Whether authenticated with WP Engine\"\n    authenticated: Boolean!\n    \"User email if authenticated\"\n    email: String\n    \"Account ID if authenticated\"\n    accountId: String\n    \"Account name if authenticated\"\n    accountName: String\n    \"Token expiry time\"\n    tokenExpiry: String\n    \"Error message if status check failed\"\n    error: String\n  }\n\n  type WpeAuthResult {\n    \"Whether authentication was successful\"\n    success: Boolean!\n    \"User email if successful\"\n    email: String\n    \"Message about the authentication result\"\n    message: String\n    \"Error message if failed\"\n    error: String\n  }\n\n  type WpeLogoutResult {\n    \"Whether logout was successful\"\n    success: Boolean!\n    \"Message about the logout result\"\n    message: String\n    \"Error message if failed\"\n    error: String\n  }\n\n  type WpeSite {\n    \"Install ID\"\n    id: String!\n    \"Install name\"\n    name: String!\n    \"Environment (production, staging, development)\"\n    environment: String!\n    \"PHP version\"\n    phpVersion: String\n    \"Primary domain\"\n    primaryDomain: String\n    \"Account ID\"\n    accountId: String\n    \"Account name\"\n    accountName: String\n    \"SFTP host\"\n    sftpHost: String\n    \"SFTP user\"\n    sftpUser: String\n  }\n\n  type ListWpeSitesResult {\n    \"Whether query was successful\"\n    success: Boolean!\n    \"Error message if failed\"\n    error: String\n    \"List of WP Engine sites\"\n    sites: [WpeSite!]\n    \"Total count of sites\"\n    count: Int\n  }\n\n  # Phase 11b: Site Linking Types\n  type WpeConnection {\n    \"Remote install ID (UUID from WP Engine)\"\n    remoteInstallId: String!\n    \"Install name (human-readable, used in portal URLs)\"\n    installName: String\n    \"Environment (production, staging, development)\"\n    environment: String\n    \"Account ID\"\n    accountId: String\n    \"WP Engine portal URL\"\n    portalUrl: String\n    \"Primary domain/CNAME\"\n    primaryDomain: String\n  }\n\n  \"Sync capabilities available for WPE-connected sites\"\n  type WpeSyncCapabilities {\n    \"Whether user can push to WP Engine\"\n    canPush: Boolean!\n    \"Whether user can pull from WP Engine\"\n    canPull: Boolean!\n    \"Available sync modes\"\n    syncModes: [String!]!\n    \"Whether Magic Sync (select files) is available\"\n    magicSyncAvailable: Boolean!\n    \"Whether database sync is available\"\n    databaseSyncAvailable: Boolean!\n  }\n\n  type GetWpeLinkResult {\n    \"Whether site is linked to WP Engine\"\n    linked: Boolean!\n    \"Site name\"\n    siteName: String\n    \"WP Engine connections\"\n    connections: [WpeConnection!]\n    \"Number of connections\"\n    connectionCount: Int\n    \"Sync capabilities (only present if linked)\"\n    capabilities: WpeSyncCapabilities\n    \"Message (for unlinked sites)\"\n    message: String\n    \"Error message if failed\"\n    error: String\n  }\n\n  # Phase 11c: Sync Operations Types\n  type SyncHistoryEvent {\n    \"Remote install name\"\n    remoteInstallName: String\n    \"Unix timestamp\"\n    timestamp: Float!\n    \"Environment (production, staging, development)\"\n    environment: String!\n    \"Sync direction\"\n    direction: String!\n    \"Sync status\"\n    status: String\n  }\n\n  type GetSyncHistoryResult {\n    \"Whether the query was successful\"\n    success: Boolean!\n    \"Site name\"\n    siteName: String\n    \"Sync history events\"\n    events: [SyncHistoryEvent!]\n    \"Number of events\"\n    count: Int\n    \"Error message if failed\"\n    error: String\n  }\n\n  type SyncResult {\n    \"Whether the sync was initiated successfully\"\n    success: Boolean!\n    \"Status message\"\n    message: String\n    \"Error message if failed\"\n    error: String\n  }\n\n  # File change detection\n  type FileChange {\n    \"File path relative to site root\"\n    path: String!\n    \"Change type: create, upload, download, delete, modify\"\n    instruction: String!\n    \"File size in bytes\"\n    size: Int\n    \"File type: - (file) or d (directory)\"\n    type: String\n  }\n\n  type GetSiteChangesResult {\n    \"Whether the query was successful\"\n    success: Boolean!\n    \"Site name\"\n    siteName: String\n    \"Direction of comparison\"\n    direction: String\n    \"Files that would be added/uploaded\"\n    added: [FileChange!]\n    \"Files that would be modified\"\n    modified: [FileChange!]\n    \"Files that would be deleted\"\n    deleted: [FileChange!]\n    \"Total number of changes\"\n    totalChanges: Int\n    \"Summary message\"\n    message: String\n    \"Error message if failed\"\n    error: String\n  }\n\n  # Phase 10: Cloud Backup Types\n  type BackupProviderStatus {\n    \"Whether authenticated with provider\"\n    authenticated: Boolean!\n    \"Account ID\"\n    accountId: String\n    \"Account email\"\n    email: String\n  }\n\n  type BackupStatusResult {\n    \"Whether backups are available\"\n    available: Boolean!\n    \"Whether the feature is enabled\"\n    featureEnabled: Boolean!\n    \"Dropbox authentication status\"\n    dropbox: BackupProviderStatus\n    \"Google Drive authentication status\"\n    googleDrive: BackupProviderStatus\n    \"Message if backups unavailable\"\n    message: String\n    \"Error message if failed\"\n    error: String\n  }\n\n  type BackupMetadata {\n    \"Snapshot ID\"\n    snapshotId: String!\n    \"Backup timestamp (ISO format)\"\n    timestamp: String\n    \"Backup note/description\"\n    note: String\n    \"Site domain\"\n    siteDomain: String\n    \"Services info (JSON)\"\n    services: String\n  }\n\n  type ListBackupsResult {\n    \"Whether query was successful\"\n    success: Boolean!\n    \"Site name\"\n    siteName: String\n    \"Backup provider\"\n    provider: String\n    \"List of backups\"\n    backups: [BackupMetadata!]\n    \"Number of backups\"\n    count: Int\n    \"Error message if failed\"\n    error: String\n  }\n\n  type CreateBackupResult {\n    \"Whether backup was created successfully\"\n    success: Boolean!\n    \"Snapshot ID\"\n    snapshotId: String\n    \"Backup timestamp\"\n    timestamp: String\n    \"Status message\"\n    message: String\n    \"Error message if failed\"\n    error: String\n  }\n\n  type RestoreBackupResult {\n    \"Whether restore was successful\"\n    success: Boolean!\n    \"Status message\"\n    message: String\n    \"Error message if failed\"\n    error: String\n  }\n\n  type DeleteBackupResult {\n    \"Whether deletion was successful\"\n    success: Boolean!\n    \"Deleted snapshot ID\"\n    deletedSnapshotId: String\n    \"Status message\"\n    message: String\n    \"Error message if failed\"\n    error: String\n  }\n\n  type DownloadBackupResult {\n    \"Whether download was successful\"\n    success: Boolean!\n    \"Path to downloaded file\"\n    filePath: String\n    \"Status message\"\n    message: String\n    \"Error message if failed\"\n    error: String\n  }\n\n  type EditBackupNoteResult {\n    \"Whether edit was successful\"\n    success: Boolean!\n    \"Updated snapshot ID\"\n    snapshotId: String\n    \"Updated note\"\n    note: String\n    \"Error message if failed\"\n    error: String\n  }\n\n  extend type Query {\n    # Phase 10: Cloud Backups\n    \"Check if cloud backups are available and authenticated\"\n    backupStatus: BackupStatusResult!\n\n    \"List all backups for a site\"\n    listBackups(siteId: ID!, provider: String!): ListBackupsResult!\n\n    # Phase 11c: Sync Operations\n    \"Get sync history for a local site\"\n    getSyncHistory(siteId: ID!, limit: Int): GetSyncHistoryResult!\n\n    \"Get file changes between local site and WP Engine (dry-run comparison)\"\n    getSiteChanges(siteId: ID!, direction: String = \"push\"): GetSiteChangesResult!\n  }\n\n  extend type Mutation {\n    # Phase 10: Cloud Backups\n    \"Create a backup of a site to cloud storage\"\n    createBackup(siteId: ID!, provider: String!, note: String): CreateBackupResult!\n\n    \"Restore a site from a cloud backup\"\n    restoreBackup(\n      siteId: ID!\n      provider: String!\n      snapshotId: String!\n      confirm: Boolean = false\n    ): RestoreBackupResult!\n\n    \"Delete a backup from cloud storage\"\n    deleteBackup(\n      siteId: ID!\n      provider: String!\n      snapshotId: String!\n      confirm: Boolean = false\n    ): DeleteBackupResult!\n\n    \"Download a backup as a ZIP file\"\n    downloadBackup(siteId: ID!, provider: String!, snapshotId: String!): DownloadBackupResult!\n\n    \"Update the note/description for a backup\"\n    editBackupNote(\n      siteId: ID!\n      provider: String!\n      snapshotId: String!\n      note: String!\n    ): EditBackupNoteResult!\n\n    # Phase 11: WP Engine Connect\n    \"Authenticate with WP Engine (opens browser for OAuth)\"\n    wpeAuthenticate: WpeAuthResult!\n\n    \"Logout from WP Engine\"\n    wpeLogout: WpeLogoutResult!\n\n    # Phase 11c: Sync Operations\n    \"Push local site to WP Engine\"\n    pushToWpe(\n      localSiteId: ID!\n      remoteInstallId: ID!\n      includeSql: Boolean = false\n      confirm: Boolean = false\n    ): SyncResult!\n\n    \"Pull from WP Engine to local site\"\n    pullFromWpe(localSiteId: ID!, remoteInstallId: ID!, includeSql: Boolean = false): SyncResult!\n  }\n`;\n\n/**\n * Create GraphQL resolvers that use Local's internal services\n */\nfunction createResolvers(services: any) {\n  const {\n    deleteSite: deleteSiteService,\n    siteData,\n    localLogger,\n    wpCli,\n    siteProcessManager,\n    addSite: addSiteService,\n    cloneSite: cloneSiteService,\n    exportSite: exportSiteService,\n    blueprints: blueprintsService,\n    browserManager,\n    adminer,\n    x509Cert,\n    siteProvisioner,\n    importSite: importSiteService,\n    lightningServices,\n    siteDatabase,\n    importSQLFile: importSQLFileService,\n    // Phase 11: WP Engine Connect\n    wpeOAuth: wpeOAuthService,\n    capi: capiService,\n    // Phase 11c: Sync services\n    wpePush: wpePushService,\n    wpePull: wpePullService,\n    connectHistory: connectHistoryService,\n    wpeConnectBase: wpeConnectBaseService,\n    // Note: Phase 10 Cloud Backup services are accessed via IPC to the Cloud Backups addon\n    // (backupService, dropbox, googleDrive, featureFlags, userData)\n  } = services;\n\n  // Helper to invoke IPC calls to the Cloud Backups addon\n  // This uses the same pattern as the BackupAIBridge\n  // Timeout constants for backup operations (in milliseconds)\n  const BACKUP_IPC_TIMEOUT = 600000; // 10 minutes for backup operations\n  const DEFAULT_IPC_TIMEOUT = 30000; // 30 seconds for quick operations\n\n  const invokeBackupIPC = async (\n    channel: string,\n    timeoutMs: number = BACKUP_IPC_TIMEOUT,\n    ...args: any[]\n  ): Promise<any> => {\n    return new Promise((resolve, reject) => {\n      const timestamp = Date.now();\n      const random = Math.random().toString(36).substr(2, 9);\n      const successReplyChannel = `${channel}-success-${timestamp}-${random}`;\n      const errorReplyChannel = `${channel}-error-${timestamp}-${random}`;\n\n      const timeoutSeconds = Math.round(timeoutMs / 1000);\n      const timeout = setTimeout(() => {\n        ipcMain.removeAllListeners(successReplyChannel);\n        ipcMain.removeAllListeners(errorReplyChannel);\n        reject(new Error(`IPC call to ${channel} timed out after ${timeoutSeconds} seconds`));\n      }, timeoutMs);\n\n      ipcMain.once(successReplyChannel, (_event: any, result: any) => {\n        clearTimeout(timeout);\n        ipcMain.removeAllListeners(errorReplyChannel);\n        localLogger.info(`[${ADDON_NAME}] IPC success from ${channel}`);\n        resolve({ result });\n      });\n\n      ipcMain.once(errorReplyChannel, (_event: any, error: any) => {\n        clearTimeout(timeout);\n        ipcMain.removeAllListeners(successReplyChannel);\n        localLogger.error(`[${ADDON_NAME}] IPC error from ${channel}: ${error?.message}`);\n        resolve({ error });\n      });\n\n      const mockEvent = {\n        reply: (replyChannel: string, data: any) => {\n          ipcMain.emit(replyChannel, null, data);\n        },\n        sender: {\n          send: (replyChannel: string, data: any) => {\n            ipcMain.emit(replyChannel, null, data);\n          },\n        },\n      };\n\n      const replyChannels = { successReplyChannel, errorReplyChannel };\n      localLogger.info(`[${ADDON_NAME}] Invoking backup IPC: ${channel}`);\n      ipcMain.emit(channel, mockEvent, replyChannels, ...args);\n    });\n  };\n\n  // Helper to get backup providers from the Cloud Backups addon\n  const getBackupProviders = async (): Promise<Array<{ id: string; name: string }>> => {\n    try {\n      const result = await invokeBackupIPC('backups:enabled-providers', DEFAULT_IPC_TIMEOUT);\n      localLogger.info(`[${ADDON_NAME}] Raw IPC result: ${JSON.stringify(result)}`);\n\n      if (result.error) {\n        localLogger.error(\n          `[${ADDON_NAME}] Failed to get backup providers: ${result.error.message}`\n        );\n        return [];\n      }\n\n      // The response is double-nested: result.result.result contains the array\n      // Structure: { result: { result: [providers...] } }\n      let providers: any = result.result;\n\n      // Unwrap nested result if present\n      if (providers && typeof providers === 'object' && !Array.isArray(providers)) {\n        if (Array.isArray(providers.result)) {\n          providers = providers.result;\n        } else if (providers.result && typeof providers.result === 'object') {\n          // Even deeper nesting\n          providers = providers.result;\n        }\n      }\n\n      localLogger.info(`[${ADDON_NAME}] Extracted providers: ${JSON.stringify(providers)}`);\n\n      if (Array.isArray(providers)) {\n        localLogger.info(`[${ADDON_NAME}] Got ${providers.length} backup providers`);\n        return providers;\n      }\n\n      localLogger.warn(\n        `[${ADDON_NAME}] Unexpected providers format after unwrapping: ${typeof providers}`\n      );\n      return [];\n    } catch (error: any) {\n      localLogger.error(`[${ADDON_NAME}] Error getting backup providers: ${error.message}`);\n      return [];\n    }\n  };\n\n  // Shared WP-CLI execution logic\n  const executeWpCli = async (\n    _parent: any,\n    args: { input: { siteId: string; args: string[]; skipPlugins?: boolean; skipThemes?: boolean } }\n  ) => {\n    const { siteId, args: wpArgs, skipPlugins = true, skipThemes = true } = args.input;\n\n    try {\n      localLogger.info(`[${ADDON_NAME}] Running WP-CLI: wp ${wpArgs.join(' ')}`);\n\n      const site = siteData.getSite(siteId);\n      if (!site) {\n        return {\n          success: false,\n          output: null,\n          error: `Site not found: ${siteId}`,\n        };\n      }\n\n      const status = await siteProcessManager.getSiteStatus(site);\n      if (status !== 'running') {\n        return {\n          success: false,\n          output: null,\n          error: `Site \"${site.name}\" is not running. Start it first with: local-cli start ${site.name}`,\n        };\n      }\n\n      const output = await wpCli.run(site, wpArgs, {\n        skipPlugins,\n        skipThemes,\n        ignoreErrors: false,\n      });\n\n      localLogger.info(`[${ADDON_NAME}] WP-CLI completed successfully`);\n\n      return {\n        success: true,\n        output: output?.trim() || '',\n        error: null,\n      };\n    } catch (error: any) {\n      localLogger.error(`[${ADDON_NAME}] WP-CLI failed:`, error);\n      return {\n        success: false,\n        output: null,\n        error: error.message || 'Unknown error',\n      };\n    }\n  };\n\n  return {\n    Query: {\n      wpCliQuery: executeWpCli,\n\n      blueprints: async () => {\n        try {\n          localLogger.info(`[${ADDON_NAME}] Fetching blueprints`);\n\n          const blueprintsList = await blueprintsService.getBlueprints();\n\n          return {\n            success: true,\n            error: null,\n            blueprints: blueprintsList.map((bp: any) => ({\n              name: bp.name,\n              lastModified: bp.lastModified,\n              // Handle nested objects - extract just the name/type string\n              phpVersion:\n                typeof bp.phpVersion === 'object'\n                  ? bp.phpVersion?.name || bp.phpVersion?.version\n                  : bp.phpVersion,\n              webServer:\n                typeof bp.webServer === 'object'\n                  ? bp.webServer?.name || bp.webServer?.type\n                  : bp.webServer,\n              database:\n                typeof bp.database === 'object'\n                  ? bp.database?.name || bp.database?.type\n                  : bp.database,\n            })),\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to fetch blueprints:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            blueprints: [],\n          };\n        }\n      },\n\n      listServices: async (_parent: any, args: { type?: string }) => {\n        const { type = 'all' } = args;\n\n        try {\n          localLogger.info(`[${ADDON_NAME}] Listing services (type: ${type})`);\n\n          if (!lightningServices) {\n            return {\n              success: false,\n              error: 'Lightning services not available',\n              services: [],\n            };\n          }\n\n          const roleMap: Record<string, string> = {\n            php: 'php',\n            database: 'mysql',\n            webserver: 'nginx',\n          };\n\n          const roleFilter = type !== 'all' ? roleMap[type] : undefined;\n          const registeredServices = lightningServices.getRegisteredServices(roleFilter);\n\n          const serviceList: Array<{ role: string; name: string; version: string }> = [];\n\n          for (const [role, versions] of Object.entries(registeredServices)) {\n            for (const [version, info] of Object.entries(versions as Record<string, any>)) {\n              serviceList.push({\n                role,\n                name: info?.name || role,\n                version,\n              });\n            }\n          }\n\n          return {\n            success: true,\n            error: null,\n            services: serviceList,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to list services:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            services: [],\n          };\n        }\n      },\n\n      // Phase 11: WP Engine Connect\n      wpeStatus: async () => {\n        try {\n          localLogger.info(`[${ADDON_NAME}] Checking WP Engine authentication status`);\n\n          if (!wpeOAuthService) {\n            return {\n              authenticated: false,\n              email: null,\n              accountId: null,\n              accountName: null,\n              tokenExpiry: null,\n              error: 'WP Engine OAuth service not available',\n            };\n          }\n\n          // Check if we have valid credentials by trying to get access token\n          const accessToken = await wpeOAuthService.getAccessToken();\n\n          if (!accessToken) {\n            return {\n              authenticated: false,\n              email: null,\n              accountId: null,\n              accountName: null,\n              tokenExpiry: null,\n              error: null,\n            };\n          }\n\n          // Try to get user info from CAPI if available\n          let email = null;\n          if (capiService) {\n            try {\n              const currentUser = await capiService.getCurrentUser();\n              email = currentUser?.email || null;\n            } catch {\n              // User info not available, but still authenticated\n            }\n          }\n\n          return {\n            authenticated: true,\n            email,\n            accountId: null,\n            accountName: null,\n            tokenExpiry: null,\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to check WPE status:`, error);\n          return {\n            authenticated: false,\n            email: null,\n            accountId: null,\n            accountName: null,\n            tokenExpiry: null,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n\n      listWpeSites: async (_parent: any, args: { accountId?: string }) => {\n        const { accountId } = args;\n\n        try {\n          localLogger.info(\n            `[${ADDON_NAME}] Listing WP Engine sites${accountId ? ` for account ${accountId}` : ''}`\n          );\n\n          if (!wpeOAuthService) {\n            return {\n              success: false,\n              error: 'WP Engine OAuth service not available',\n              sites: [],\n              count: 0,\n            };\n          }\n\n          // Check if authenticated by trying to get access token\n          const accessToken = await wpeOAuthService.getAccessToken();\n          if (!accessToken) {\n            return {\n              success: false,\n              error: 'Not authenticated with WP Engine. Use wpe_authenticate first.',\n              sites: [],\n              count: 0,\n            };\n          }\n\n          if (!capiService) {\n            return {\n              success: false,\n              error: 'WP Engine CAPI service not available',\n              sites: [],\n              count: 0,\n            };\n          }\n\n          // Get installs from CAPI using getInstallList\n          const installs = await capiService.getInstallList();\n\n          if (!installs) {\n            return {\n              success: true,\n              error: null,\n              sites: [],\n              count: 0,\n            };\n          }\n\n          const sites = installs.map((install: any) => ({\n            id: install.id,\n            name: install.name,\n            environment: install.environment || 'production',\n            phpVersion: install.phpVersion || null,\n            primaryDomain: install.primaryDomain || install.cname || null,\n            accountId: install.accountId || accountId || null,\n            accountName: install.accountName || null,\n            sftpHost: `${install.name}.ssh.wpengine.net`,\n            sftpUser: install.name,\n          }));\n\n          return {\n            success: true,\n            error: null,\n            sites,\n            count: sites.length,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to list WPE sites:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            sites: [],\n            count: 0,\n          };\n        }\n      },\n\n      // Phase 11b: Site Linking\n      getWpeLink: async (_parent: any, args: { siteId: string }) => {\n        const { siteId } = args;\n\n        try {\n          localLogger.info(`[${ADDON_NAME}] Getting WP Engine link for site ${siteId}`);\n\n          // Get site from siteData\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              linked: false,\n              siteName: null,\n              connections: [],\n              connectionCount: 0,\n              message: null,\n              error: `Site not found: ${siteId}`,\n            };\n          }\n\n          // Get hostConnections from site\n          const hostConnections = site.hostConnections || [];\n          const wpeConnections = hostConnections.filter((c: any) => c.hostId === 'wpe');\n\n          if (wpeConnections.length === 0) {\n            return {\n              linked: false,\n              siteName: site.name,\n              connections: [],\n              connectionCount: 0,\n              message:\n                'Site is not linked to any WP Engine environment. Use Connect in Local to pull a site from WPE.',\n              error: null,\n            };\n          }\n\n          // Transform connections for output, enriching with CAPI data if available\n          const connections = await Promise.all(\n            wpeConnections.map(async (c: any) => {\n              let installName = c.remoteSiteId; // Default to UUID\n              let portalUrl = null;\n              let primaryDomain = null;\n\n              // Try to get install details from CAPI to get the actual name\n              // remoteSiteId matches install.site.id (WPE Site ID), not install.id\n              if (capiService && typeof capiService.getInstallList === 'function') {\n                try {\n                  localLogger.info(\n                    `[${ADDON_NAME}] Looking for install with site.id=${c.remoteSiteId}, env=${c.remoteSiteEnv}`\n                  );\n                  const installs = await capiService.getInstallList();\n                  localLogger.info(\n                    `[${ADDON_NAME}] Got ${installs?.length || 0} installs from CAPI`\n                  );\n                  if (installs && installs.length > 0) {\n                    // Log first install structure for debugging\n                    localLogger.info(\n                      `[${ADDON_NAME}] Sample install structure: ${JSON.stringify(installs[0], null, 2)}`\n                    );\n\n                    // Match by site.id (remoteSiteId is the WPE Site ID, not Install ID)\n                    // Also filter by environment if available\n                    const matchingInstall = installs.find(\n                      (i: any) =>\n                        i.site?.id === c.remoteSiteId &&\n                        (!c.remoteSiteEnv || i.environment === c.remoteSiteEnv)\n                    );\n\n                    if (matchingInstall) {\n                      localLogger.info(`[${ADDON_NAME}] Found match: ${matchingInstall.name}`);\n                      installName = matchingInstall.name;\n                      portalUrl = `https://my.wpengine.com/installs/${matchingInstall.name}`;\n                      primaryDomain =\n                        matchingInstall.primary_domain || matchingInstall.cname || null;\n                    } else {\n                      localLogger.warn(\n                        `[${ADDON_NAME}] No matching install found for site.id=${c.remoteSiteId}`\n                      );\n                    }\n                  }\n                } catch (e: any) {\n                  localLogger.warn(\n                    `[${ADDON_NAME}] Could not look up install from CAPI: ${e.message}`\n                  );\n                }\n              } else {\n                localLogger.warn(`[${ADDON_NAME}] capiService or getInstallList not available`);\n              }\n\n              return {\n                remoteInstallId: c.remoteSiteId,\n                installName,\n                environment: c.remoteSiteEnv || null,\n                accountId: c.accountId || null,\n                portalUrl,\n                primaryDomain,\n              };\n            })\n          );\n\n          // Capabilities are always the same for WPE-connected sites\n          const capabilities = {\n            canPush: true,\n            canPull: true,\n            syncModes: ['all_files', 'select_files', 'database_only'],\n            magicSyncAvailable: true,\n            databaseSyncAvailable: true,\n          };\n\n          return {\n            linked: true,\n            siteName: site.name,\n            connections,\n            connectionCount: connections.length,\n            capabilities,\n            message: null,\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to get WPE link:`, error);\n          return {\n            linked: false,\n            siteName: null,\n            connections: [],\n            connectionCount: 0,\n            message: null,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n\n      // Phase 10: Cloud Backups\n      backupStatus: async () => {\n        try {\n          localLogger.info(`[${ADDON_NAME}] Checking backup status`);\n\n          // Get providers from Cloud Backups addon via IPC\n          const providers = await getBackupProviders();\n          localLogger.info(`[${ADDON_NAME}] Got ${providers.length} backup providers`);\n\n          if (providers.length === 0) {\n            return {\n              available: false,\n              featureEnabled: false,\n              dropbox: null,\n              googleDrive: null,\n              message:\n                'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub (hub.localwp.com/addons/cloud-backups).',\n              error: null,\n            };\n          }\n\n          // Map provider info to our response format\n          const dropboxProvider = providers.find(\n            (p: any) => p.id === 'dropbox' || p.name?.toLowerCase().includes('dropbox')\n          ) as any;\n          const googleProvider = providers.find(\n            (p: any) => p.id === 'google' || p.name?.toLowerCase().includes('google')\n          ) as any;\n\n          const dropboxStatus = dropboxProvider\n            ? {\n                authenticated: true,\n                accountId: dropboxProvider.id,\n                email: dropboxProvider.email || null,\n              }\n            : {\n                authenticated: false,\n                accountId: null as string | null,\n                email: null as string | null,\n              };\n\n          const googleDriveStatus = googleProvider\n            ? {\n                authenticated: true,\n                accountId: googleProvider.id,\n                email: googleProvider.email || null,\n              }\n            : {\n                authenticated: false,\n                accountId: null as string | null,\n                email: null as string | null,\n              };\n\n          const hasProvider = providers.length > 0;\n\n          return {\n            available: hasProvider,\n            featureEnabled: true,\n            dropbox: dropboxStatus,\n            googleDrive: googleDriveStatus,\n            message: hasProvider\n              ? null\n              : 'No cloud storage provider authenticated. Connect Dropbox or Google Drive in Local settings.',\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to check backup status:`, error);\n          return {\n            available: false,\n            featureEnabled: false,\n            dropbox: null,\n            googleDrive: null,\n            message: null,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n\n      listBackups: async (_parent: any, args: { siteId: string; provider: string }) => {\n        const { siteId, provider } = args;\n\n        try {\n          localLogger.info(`[${ADDON_NAME}] Listing backups for site ${siteId} from ${provider}`);\n\n          // Get site\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              siteName: null,\n              provider,\n              backups: [],\n              count: 0,\n              error: `Site not found: ${siteId}`,\n            };\n          }\n\n          // Get providers from Cloud Backups addon\n          const providers = await getBackupProviders();\n          if (providers.length === 0) {\n            return {\n              success: false,\n              siteName: site.name,\n              provider,\n              backups: [],\n              count: 0,\n              error:\n                'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub.',\n            };\n          }\n\n          // Find the matching provider (map 'googleDrive' to 'google' for the addon)\n          const providerMap: Record<string, string> = { googleDrive: 'google', dropbox: 'dropbox' };\n          const providerId = providerMap[provider] || provider;\n          const matchedProvider = providers.find((p: any) => p.id === providerId);\n\n          if (!matchedProvider) {\n            return {\n              success: false,\n              siteName: site.name,\n              provider,\n              backups: [],\n              count: 0,\n              error: `Provider '${provider}' not configured. Available: ${providers.map((p: any) => p.name).join(', ')}`,\n            };\n          }\n\n          // For listing snapshots, use the Hub provider ID directly (e.g., 'google')\n          // NOT the rclone backend name ('drive') - the Hub queries expect the OAuth provider name\n          // Also pass pageOffset parameter (0 for first page)\n          const result = await invokeBackupIPC(\n            'backups:provider-snapshots',\n            DEFAULT_IPC_TIMEOUT,\n            siteId,\n            matchedProvider.id,\n            0\n          );\n          localLogger.info(\n            `[${ADDON_NAME}] Provider snapshots raw result: ${JSON.stringify(result)}`\n          );\n\n          if (result.error) {\n            return {\n              success: false,\n              siteName: site.name,\n              provider,\n              backups: [],\n              count: 0,\n              error: result.error.message || 'Failed to list backups',\n            };\n          }\n\n          // Unwrap nested result structure (similar to providers)\n          let backupsData = result.result;\n          if (backupsData && typeof backupsData === 'object' && !Array.isArray(backupsData)) {\n            // Check for nested result or snapshots array\n            if (Array.isArray(backupsData.result)) {\n              backupsData = backupsData.result;\n            } else if (Array.isArray(backupsData.snapshots)) {\n              backupsData = backupsData.snapshots;\n            } else if (backupsData.result && Array.isArray(backupsData.result.snapshots)) {\n              backupsData = backupsData.result.snapshots;\n            }\n          }\n\n          const backups = Array.isArray(backupsData) ? backupsData : [];\n          localLogger.info(`[${ADDON_NAME}] Extracted ${backups.length} backups`);\n\n          return {\n            success: true,\n            siteName: site.name,\n            provider,\n            backups: backups.map((b: any) => ({\n              // Use hash for snapshotId as that's what restic uses for restore/delete operations\n              // The Hub ID (b.id) is just a database identifier\n              snapshotId: b.hash || b.snapshotId || b.short_id,\n              timestamp: b.updatedAt || b.createdAt || b.timestamp || b.time || b.created,\n              note:\n                b.configObject?.description || b.note || b.description || b.tags?.description || '',\n              siteDomain: b.configObject?.name\n                ? `${b.configObject.name}.local`\n                : b.siteDomain || site.domain,\n              services: JSON.stringify(b.configObject?.services || b.services || {}),\n            })),\n            count: backups.length,\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to list backups:`, error);\n          return {\n            success: false,\n            siteName: null,\n            provider,\n            backups: [],\n            count: 0,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n\n      // Phase 11c: Sync History\n      getSyncHistory: async (_parent: any, args: { siteId: string; limit?: number }) => {\n        const { siteId, limit = 30 } = args;\n\n        try {\n          localLogger.info(`[${ADDON_NAME}] Getting sync history for site ${siteId}`);\n\n          // Get site to verify it exists\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              siteName: null,\n              events: [],\n              count: 0,\n              error: `Site not found: ${siteId}`,\n            };\n          }\n\n          // Check if connectHistory service is available\n          if (!connectHistoryService || typeof connectHistoryService.getEvents !== 'function') {\n            return {\n              success: false,\n              siteName: site.name,\n              events: [],\n              count: 0,\n              error: 'Sync history service not available',\n            };\n          }\n\n          const events = connectHistoryService.getEvents(siteId);\n          const limitedEvents = events.slice(0, limit);\n\n          return {\n            success: true,\n            siteName: site.name,\n            events: limitedEvents.map((e: any) => ({\n              remoteInstallName: e.remoteInstallName || null,\n              timestamp: e.timestamp,\n              environment: e.environment,\n              direction: e.direction,\n              status: e.status || null,\n            })),\n            count: limitedEvents.length,\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to get sync history:`, error);\n          return {\n            success: false,\n            siteName: null,\n            events: [],\n            count: 0,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n\n      // Get file changes between local and WPE (dry-run comparison)\n      getSiteChanges: async (_parent: any, args: { siteId: string; direction?: string }) => {\n        const { siteId, direction = 'push' } = args;\n\n        try {\n          localLogger.info(\n            `[${ADDON_NAME}] Getting site changes for ${siteId}, direction=${direction}`\n          );\n\n          // Validate direction\n          if (direction !== 'push' && direction !== 'pull') {\n            return {\n              success: false,\n              siteName: null,\n              direction,\n              added: [],\n              modified: [],\n              deleted: [],\n              totalChanges: 0,\n              message: null,\n              error: 'Invalid direction. Must be \"push\" or \"pull\".',\n            };\n          }\n\n          // Get site\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              siteName: null,\n              direction,\n              added: [],\n              modified: [],\n              deleted: [],\n              totalChanges: 0,\n              message: null,\n              error: `Site not found: ${siteId}`,\n            };\n          }\n\n          // Check WPE connection\n          const wpeConnection = site.hostConnections?.find((c: any) => c.hostId === 'wpe');\n          if (!wpeConnection) {\n            return {\n              success: false,\n              siteName: site.name,\n              direction,\n              added: [],\n              modified: [],\n              deleted: [],\n              totalChanges: 0,\n              message: null,\n              error:\n                'Site is not linked to WP Engine. Use Connect in Local to link the site first.',\n            };\n          }\n\n          // Check service availability\n          if (\n            !wpeConnectBaseService ||\n            typeof wpeConnectBaseService.listModifications !== 'function'\n          ) {\n            return {\n              success: false,\n              siteName: site.name,\n              direction,\n              added: [],\n              modified: [],\n              deleted: [],\n              totalChanges: 0,\n              message: null,\n              error: 'WPE Connect service not available',\n            };\n          }\n\n          // Get install details from CAPI\n          let installName = wpeConnection.remoteSiteId;\n          let primaryDomain = '';\n          let installId = '';\n\n          if (capiService && typeof capiService.getInstallList === 'function') {\n            const installs = await capiService.getInstallList();\n            const matchingInstall = installs?.find(\n              (i: any) =>\n                i.site?.id === wpeConnection.remoteSiteId &&\n                (!wpeConnection.remoteSiteEnv || i.environment === wpeConnection.remoteSiteEnv)\n            );\n            if (matchingInstall) {\n              installName = matchingInstall.name;\n              primaryDomain =\n                matchingInstall.primary_domain ||\n                matchingInstall.cname ||\n                `${matchingInstall.name}.wpengine.com`;\n              installId = matchingInstall.id;\n            }\n          }\n\n          if (!primaryDomain) {\n            return {\n              success: false,\n              siteName: site.name,\n              direction,\n              added: [],\n              modified: [],\n              deleted: [],\n              totalChanges: 0,\n              message: null,\n              error:\n                'Could not determine WP Engine install details. Please ensure you are authenticated.',\n            };\n          }\n\n          // Call listModifications (dry-run rsync comparison)\n          localLogger.info(`[${ADDON_NAME}] Calling listModifications for ${installName}`);\n          const modifications = await wpeConnectBaseService.listModifications({\n            connectArgs: {\n              wpengineInstallName: installName,\n              wpengineInstallId: installId,\n              wpengineSiteId: wpeConnection.remoteSiteId,\n              wpenginePrimaryDomain: primaryDomain,\n              localSiteId: site.id,\n            },\n            direction: direction as 'push' | 'pull',\n            includeIgnored: false,\n          });\n\n          // Categorize changes\n          const added = modifications\n            .filter(\n              (f: any) =>\n                f.instruction === 'create' ||\n                f.instruction === 'upload' ||\n                f.instruction === 'download'\n            )\n            .map((f: any) => ({\n              path: f.path,\n              instruction: f.instruction,\n              size: f.size,\n              type: f.type,\n            }));\n\n          const modified = modifications\n            .filter((f: any) => f.instruction === 'modify')\n            .map((f: any) => ({\n              path: f.path,\n              instruction: f.instruction,\n              size: f.size,\n              type: f.type,\n            }));\n\n          const deleted = modifications\n            .filter((f: any) => f.instruction === 'delete')\n            .map((f: any) => ({\n              path: f.path,\n              instruction: f.instruction,\n              size: f.size,\n              type: f.type,\n            }));\n\n          const totalChanges = added.length + modified.length + deleted.length;\n          const directionLabel = direction === 'push' ? 'local → WPE' : 'WPE → local';\n\n          return {\n            success: true,\n            siteName: site.name,\n            direction,\n            added,\n            modified,\n            deleted,\n            totalChanges,\n            message:\n              totalChanges > 0\n                ? `${totalChanges} file(s) changed (${directionLabel}): ${added.length} added, ${modified.length} modified, ${deleted.length} deleted`\n                : `No changes detected (${directionLabel})`,\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to get site changes:`, error);\n          return {\n            success: false,\n            siteName: null,\n            direction,\n            added: [],\n            modified: [],\n            deleted: [],\n            totalChanges: 0,\n            message: null,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n    },\n    Mutation: {\n      wpCli: executeWpCli,\n\n      createSite: async (\n        _parent: any,\n        args: {\n          input: {\n            name: string;\n            phpVersion?: string;\n            webServer?: string;\n            database?: string;\n            wpAdminUsername?: string;\n            wpAdminPassword?: string;\n            wpAdminEmail?: string;\n            blueprint?: string;\n          };\n        }\n      ) => {\n        // DEBUG: Log raw args received\n        localLogger.info(`[${ADDON_NAME}] createSite called with args: ${JSON.stringify(args)}`);\n\n        const {\n          name,\n          phpVersion,\n          webServer = 'nginx',\n          database = 'mysql',\n          wpAdminUsername = 'admin',\n          wpAdminPassword = 'password',\n          wpAdminEmail = 'admin@local.test',\n          blueprint,\n        } = args.input;\n\n        // DEBUG: Log destructured values\n        localLogger.info(\n          `[${ADDON_NAME}] Destructured - name: ${name}, blueprint: ${blueprint}, typeof blueprint: ${typeof blueprint}`\n        );\n\n        try {\n          localLogger.info(\n            `[${ADDON_NAME}] Creating site: ${name}${blueprint ? ` from blueprint: ${blueprint}` : ''}`\n          );\n\n          // Generate slug and domain from name\n          const siteSlug = name\n            .toLowerCase()\n            .replace(/[^a-z0-9]+/g, '-')\n            .replace(/^-|-$/g, '');\n          const siteDomain = `${siteSlug}.local`;\n\n          const os = require('os');\n          const path = require('path');\n          const fs = require('fs');\n          const sitePath = path.join(os.homedir(), 'Local Sites', siteSlug);\n\n          // If blueprint is provided, use importSiteService instead of addSiteService\n          if (blueprint) {\n            localLogger.info(`[${ADDON_NAME}] Blueprint parameter received: ${blueprint}`);\n\n            // Get the userDataPath from electron app\n            const { app } = require('electron');\n            const userDataPath = app.getPath('userData');\n            const blueprintZipPath = path.join(userDataPath, 'blueprints', `${blueprint}.zip`);\n\n            localLogger.info(`[${ADDON_NAME}] Looking for blueprint at: ${blueprintZipPath}`);\n\n            // Verify blueprint exists\n            if (!fs.existsSync(blueprintZipPath)) {\n              localLogger.error(`[${ADDON_NAME}] Blueprint not found at: ${blueprintZipPath}`);\n              return {\n                success: false,\n                error: `Blueprint not found: ${blueprint}. Use list_blueprints to see available blueprints.`,\n                siteId: null,\n                siteName: name,\n                siteDomain: null,\n              };\n            }\n\n            localLogger.info(`[${ADDON_NAME}] Found blueprint at: ${blueprintZipPath}`);\n\n            // Read the local-site.json from the blueprint zip to get manifest\n            let localSiteJSON: any;\n            try {\n              const StreamZip = require('node-stream-zip');\n              localLogger.info(`[${ADDON_NAME}] node-stream-zip loaded successfully`);\n\n              const zip = new StreamZip.async({ file: blueprintZipPath });\n              const entries = await zip.entries();\n              localLogger.info(\n                `[${ADDON_NAME}] Zip entries loaded, count: ${Object.keys(entries).length}`\n              );\n\n              const filename = entries['local-site.json']\n                ? 'local-site.json'\n                : 'pressmatic-site.json';\n              localLogger.info(`[${ADDON_NAME}] Reading manifest file: ${filename}`);\n\n              const data = await zip.entryData(filename);\n              localSiteJSON = JSON.parse(data.toString('utf8'));\n              await zip.close();\n              localLogger.info(\n                `[${ADDON_NAME}] Successfully read manifest:`,\n                JSON.stringify(localSiteJSON).substring(0, 200)\n              );\n            } catch (zipError: any) {\n              localLogger.error(\n                `[${ADDON_NAME}] Failed to read blueprint zip: ${zipError.message}`,\n                zipError\n              );\n              return {\n                success: false,\n                error: `Failed to read blueprint manifest: ${zipError.message}`,\n                siteId: null,\n                siteName: name,\n                siteDomain: null,\n              };\n            }\n\n            // Build import settings\n            const importSettings: any = {\n              siteName: name,\n              siteDomain: siteDomain,\n              sitePath: sitePath,\n              zip: blueprintZipPath,\n              importData: {\n                type: 'local-blueprint',\n                oldSite: localSiteJSON,\n              },\n              environment: localSiteJSON.environment || 'flywheel',\n              blueprint: blueprint,\n            };\n\n            // Copy service versions from blueprint if available\n            if (localSiteJSON.services) {\n              // Extract PHP version\n              const phpService = Object.values(localSiteJSON.services).find(\n                (s: any) => s.role === 'php'\n              ) as any;\n              if (phpService) {\n                importSettings.phpVersion = phpService.version;\n              }\n\n              // Extract database\n              const dbService = Object.values(localSiteJSON.services).find(\n                (s: any) => s.role === 'database' || s.role === 'db'\n              ) as any;\n              if (dbService) {\n                importSettings.database = `${dbService.name}-${dbService.version}`;\n              }\n\n              // Extract web server\n              const webService = Object.values(localSiteJSON.services).find(\n                (s: any) => s.role === 'http' || s.role === 'web'\n              ) as any;\n              if (webService) {\n                importSettings.webServer = `${webService.name}-${webService.version}`;\n              }\n            } else if (localSiteJSON.phpVersion) {\n              importSettings.phpVersion = localSiteJSON.phpVersion;\n            }\n\n            localLogger.info(\n              `[${ADDON_NAME}] Import settings prepared:`,\n              JSON.stringify(importSettings).substring(0, 500)\n            );\n\n            if (!importSiteService) {\n              localLogger.error(`[${ADDON_NAME}] importSiteService is not available!`);\n              return {\n                success: false,\n                error: 'Import service not available',\n                siteId: null,\n                siteName: name,\n                siteDomain: null,\n              };\n            }\n\n            localLogger.info(`[${ADDON_NAME}] Calling importSiteService.run()...`);\n\n            // Use the importSiteService to create from blueprint\n            const importResult = await importSiteService.run(importSettings);\n\n            localLogger.info(\n              `[${ADDON_NAME}] Import result:`,\n              JSON.stringify(importResult || 'null').substring(0, 500)\n            );\n\n            if (importResult && importResult.id) {\n              localLogger.info(\n                `[${ADDON_NAME}] Successfully created site from blueprint: ${name} (${importResult.id})`\n              );\n              return {\n                success: true,\n                error: null,\n                siteId: importResult.id,\n                siteName: name,\n                siteDomain: siteDomain,\n              };\n            } else {\n              localLogger.warn(`[${ADDON_NAME}] Import returned but no site ID found`);\n              return {\n                success: true,\n                error: null,\n                siteId: null,\n                siteName: name,\n                siteDomain: siteDomain,\n              };\n            }\n          }\n\n          // No blueprint - create a fresh site\n          const newSiteInfo: any = {\n            siteName: name,\n            siteDomain: siteDomain,\n            sitePath: sitePath,\n            webServer: webServer,\n            database: database,\n          };\n\n          if (phpVersion) {\n            newSiteInfo.phpVersion = phpVersion;\n          }\n\n          const wpCredentials = {\n            adminUsername: wpAdminUsername,\n            adminPassword: wpAdminPassword,\n            adminEmail: wpAdminEmail,\n          };\n\n          const site = await addSiteService.addSite({\n            newSiteInfo,\n            wpCredentials,\n            goToSite: false,\n          });\n\n          localLogger.info(`[${ADDON_NAME}] Successfully created site: ${name} (${site.id})`);\n\n          return {\n            success: true,\n            error: null,\n            siteId: site.id,\n            siteName: name,\n            siteDomain: siteDomain,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to create site:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            siteId: null,\n            siteName: name,\n            siteDomain: null,\n          };\n        }\n      },\n\n      deleteSite: async (\n        _parent: any,\n        args: { input: { id: string; trashFiles?: boolean; updateHosts?: boolean } }\n      ) => {\n        const { id, trashFiles = true, updateHosts = true } = args.input;\n\n        try {\n          localLogger.info(`[${ADDON_NAME}] Deleting site: ${id}`);\n\n          const site = siteData.getSite(id);\n          if (!site) {\n            return {\n              success: false,\n              error: `Site not found: ${id}`,\n              siteId: id,\n            };\n          }\n\n          await deleteSiteService.deleteSite({\n            site,\n            trashFiles,\n            updateHosts,\n          });\n\n          localLogger.info(`[${ADDON_NAME}] Successfully deleted site: ${site.name}`);\n\n          return {\n            success: true,\n            error: null,\n            siteId: id,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to delete site:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            siteId: id,\n          };\n        }\n      },\n\n      deleteSites: async (_parent: any, args: { ids: string[]; trashFiles?: boolean }) => {\n        const { ids, trashFiles = true } = args;\n\n        try {\n          localLogger.info(`[${ADDON_NAME}] Deleting ${ids.length} sites`);\n\n          await deleteSiteService.deleteSites({\n            siteIds: ids,\n            trashFiles,\n            updateHosts: true,\n          });\n\n          localLogger.info(`[${ADDON_NAME}] Successfully deleted ${ids.length} sites`);\n\n          return {\n            success: true,\n            error: null,\n            siteId: ids.join(','),\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to delete sites:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            siteId: ids.join(','),\n          };\n        }\n      },\n\n      openSite: async (_parent: any, args: { input: { siteId: string; path?: string } }) => {\n        const { siteId, path = '/' } = args.input;\n\n        try {\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              error: `Site not found: ${siteId}`,\n              url: null,\n            };\n          }\n\n          // Check if site is running\n          const status = await siteProcessManager.getSiteStatus(site);\n          if (status !== 'running') {\n            return {\n              success: false,\n              error: `Site \"${site.name}\" must be running to open in browser. Start it first.`,\n              url: null,\n            };\n          }\n\n          const protocol = site.isStarred ? 'https' : 'http';\n          const url = `${protocol}://${site.domain}${path}`;\n\n          localLogger.info(`[${ADDON_NAME}] Opening site in browser: ${url}`);\n\n          if (browserManager) {\n            await browserManager.openInBrowser(url);\n          } else {\n            // Fallback to shell.openExternal\n            const { shell } = require('electron');\n            await shell.openExternal(url);\n          }\n\n          return {\n            success: true,\n            error: null,\n            url,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to open site:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            url: null,\n          };\n        }\n      },\n\n      cloneSite: async (_parent: any, args: { input: { siteId: string; newName: string } }) => {\n        const { siteId, newName } = args.input;\n\n        try {\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              error: `Site not found: ${siteId}`,\n              newSiteId: null,\n              newSiteName: null,\n              newSiteDomain: null,\n            };\n          }\n\n          // Check if site is running - needed for database cloning\n          const status = await siteProcessManager.getSiteStatus(site);\n          if (status !== 'running') {\n            return {\n              success: false,\n              error: `Site \"${site.name}\" must be running to clone. Start it first.`,\n              newSiteId: null,\n              newSiteName: null,\n              newSiteDomain: null,\n            };\n          }\n\n          localLogger.info(`[${ADDON_NAME}] Cloning site ${site.name} to ${newName}`);\n\n          const newSite = await cloneSiteService.cloneSite({\n            site,\n            newSiteName: newName,\n          });\n\n          localLogger.info(\n            `[${ADDON_NAME}] Successfully cloned site: ${newSite.name} (${newSite.id})`\n          );\n\n          return {\n            success: true,\n            error: null,\n            newSiteId: newSite.id,\n            newSiteName: newSite.name,\n            newSiteDomain: newSite.domain,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to clone site:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            newSiteId: null,\n            newSiteName: null,\n            newSiteDomain: null,\n          };\n        }\n      },\n\n      exportSite: async (\n        _parent: any,\n        args: { input: { siteId: string; outputPath?: string } }\n      ) => {\n        const { siteId, outputPath } = args.input;\n        const os = require('os');\n        const path = require('path');\n\n        try {\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              error: `Site not found: ${siteId}`,\n              exportPath: null,\n            };\n          }\n\n          // Check if site is running - needed for database export\n          const status = await siteProcessManager.getSiteStatus(site);\n          if (status !== 'running') {\n            return {\n              success: false,\n              error: `Site \"${site.name}\" must be running to export. Start it first.`,\n              exportPath: null,\n            };\n          }\n\n          // Default to Downloads folder\n          const outputDir = outputPath || path.join(os.homedir(), 'Downloads');\n          const timestamp = new Date().toISOString().replace(/[:.]/g, '-').slice(0, 19);\n          const fileName = `${site.name}-${timestamp}.zip`;\n          const fullPath = path.join(outputDir, fileName);\n\n          localLogger.info(`[${ADDON_NAME}] Exporting site ${site.name} to ${fullPath}`);\n\n          // Use default export filter (excludes archive files)\n          const defaultExportFilter = '*.zip, *.tar.gz, *.bz2, *.tgz';\n\n          await exportSiteService.exportSite({\n            site,\n            outputPath: fullPath,\n            filter: defaultExportFilter,\n          });\n\n          localLogger.info(`[${ADDON_NAME}] Successfully exported site to: ${fullPath}`);\n\n          return {\n            success: true,\n            error: null,\n            exportPath: fullPath,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to export site:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            exportPath: null,\n          };\n        }\n      },\n\n      saveBlueprint: async (_parent: any, args: { input: { siteId: string; name: string } }) => {\n        const { siteId, name } = args.input;\n\n        try {\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              error: `Site not found: ${siteId}`,\n              blueprintName: null,\n            };\n          }\n\n          // Check if site is running - needed for database export\n          const status = await siteProcessManager.getSiteStatus(site);\n          if (status !== 'running') {\n            return {\n              success: false,\n              error: `Site \"${site.name}\" must be running to save as blueprint. Start it first.`,\n              blueprintName: null,\n            };\n          }\n\n          localLogger.info(`[${ADDON_NAME}] Saving site ${site.name} as blueprint: ${name}`);\n\n          // Use default export filter (excludes archive files)\n          const defaultFilter = '*.zip, *.tar.gz, *.bz2, *.tgz';\n\n          await blueprintsService.saveBlueprint({\n            name,\n            siteId,\n            filter: defaultFilter,\n          });\n\n          localLogger.info(`[${ADDON_NAME}] Successfully saved blueprint: ${name}`);\n\n          return {\n            success: true,\n            error: null,\n            blueprintName: name,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to save blueprint:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            blueprintName: null,\n          };\n        }\n      },\n\n      // Phase 8: WordPress Development Tools\n      exportDatabase: async (\n        _parent: any,\n        args: { input: { siteId: string; outputPath?: string } }\n      ) => {\n        const { siteId, outputPath } = args.input;\n        const os = require('os');\n        const pathModule = require('path');\n\n        try {\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              error: `Site not found: ${siteId}`,\n              outputPath: null,\n            };\n          }\n\n          // Check if site is running - database must be accessible\n          const status = await siteProcessManager.getSiteStatus(site);\n          if (status !== 'running') {\n            return {\n              success: false,\n              error: `Site \"${site.name}\" must be running to export database. Start it first.`,\n              outputPath: null,\n            };\n          }\n\n          // Default to Downloads folder with site name\n          const defaultPath = pathModule.join(\n            os.homedir(),\n            'Downloads',\n            `${site.name.replace(/[^a-z0-9]/gi, '-')}.sql`\n          );\n          const finalPath = outputPath || defaultPath;\n\n          localLogger.info(`[${ADDON_NAME}] Exporting database for ${site.name} to ${finalPath}`);\n\n          // Use siteDatabase.dump() which properly sets up MySQL environment\n          if (!siteDatabase) {\n            return {\n              success: false,\n              error: 'Database service not available',\n              outputPath: null,\n            };\n          }\n\n          await siteDatabase.dump(site, finalPath);\n\n          localLogger.info(`[${ADDON_NAME}] Successfully exported database to: ${finalPath}`);\n\n          return {\n            success: true,\n            error: null,\n            outputPath: finalPath,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to export database:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            outputPath: null,\n          };\n        }\n      },\n\n      importDatabase: async (\n        _parent: any,\n        args: { input: { siteId: string; sqlPath: string } }\n      ) => {\n        const { siteId, sqlPath } = args.input;\n        const fs = require('fs');\n\n        try {\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              error: `Site not found: ${siteId}`,\n            };\n          }\n\n          if (!fs.existsSync(sqlPath)) {\n            return {\n              success: false,\n              error: `SQL file not found: ${sqlPath}`,\n            };\n          }\n\n          // Check if site is running - database must be accessible\n          const status = await siteProcessManager.getSiteStatus(site);\n          if (status !== 'running') {\n            return {\n              success: false,\n              error: `Site \"${site.name}\" must be running to import database. Start it first.`,\n            };\n          }\n\n          localLogger.info(`[${ADDON_NAME}] Importing database for ${site.name} from ${sqlPath}`);\n\n          // Use importSQLFile service which properly sets up MySQL environment\n          if (!importSQLFileService) {\n            return {\n              success: false,\n              error: 'Import SQL file service not available',\n            };\n          }\n\n          await importSQLFileService(site, sqlPath);\n\n          localLogger.info(`[${ADDON_NAME}] Successfully imported database from: ${sqlPath}`);\n\n          return {\n            success: true,\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to import database:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n\n      openAdminer: async (_parent: any, args: { input: { siteId: string } }) => {\n        const { siteId } = args.input;\n\n        try {\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              error: `Site not found: ${siteId}`,\n            };\n          }\n\n          // Check if site is running - database must be accessible\n          const status = await siteProcessManager.getSiteStatus(site);\n          if (status !== 'running') {\n            return {\n              success: false,\n              error: `Site \"${site.name}\" must be running to open Adminer. Start it first.`,\n            };\n          }\n\n          localLogger.info(`[${ADDON_NAME}] Opening Adminer for ${site.name}`);\n\n          if (adminer) {\n            await adminer.open(site);\n          } else {\n            return {\n              success: false,\n              error: 'Adminer service not available',\n            };\n          }\n\n          return {\n            success: true,\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to open Adminer:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n\n      trustSsl: async (_parent: any, args: { input: { siteId: string } }) => {\n        const { siteId } = args.input;\n\n        try {\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              error: `Site not found: ${siteId}`,\n            };\n          }\n\n          localLogger.info(`[${ADDON_NAME}] Trusting SSL for ${site.name}`);\n\n          if (x509Cert) {\n            await x509Cert.trustCert(site);\n          } else {\n            return {\n              success: false,\n              error: 'X509 certificate service not available',\n            };\n          }\n\n          return {\n            success: true,\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to trust SSL:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n\n      mcpRenameSite: async (_parent: any, args: { input: { siteId: string; newName: string } }) => {\n        const { siteId, newName } = args.input;\n\n        try {\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              error: `Site not found: ${siteId}`,\n              newName: null,\n            };\n          }\n\n          localLogger.info(`[${ADDON_NAME}] Renaming ${site.name} to ${newName}`);\n\n          // Update site name via siteData\n          site.name = newName;\n          await siteData.updateSite(siteId, { name: newName });\n\n          localLogger.info(`[${ADDON_NAME}] Successfully renamed site to: ${newName}`);\n\n          return {\n            success: true,\n            error: null,\n            newName,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to rename site:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            newName: null,\n          };\n        }\n      },\n\n      changePhpVersion: async (\n        _parent: any,\n        args: { input: { siteId: string; phpVersion: string } }\n      ) => {\n        const { siteId, phpVersion } = args.input;\n\n        try {\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              error: `Site not found: ${siteId}`,\n              phpVersion: null,\n            };\n          }\n\n          localLogger.info(\n            `[${ADDON_NAME}] Changing PHP version for ${site.name} to ${phpVersion}`\n          );\n\n          if (siteProvisioner) {\n            await siteProvisioner.swapService(site, 'php', phpVersion);\n          } else {\n            return {\n              success: false,\n              error: 'Site provisioner service not available',\n              phpVersion: null,\n            };\n          }\n\n          localLogger.info(`[${ADDON_NAME}] Successfully changed PHP version to: ${phpVersion}`);\n\n          return {\n            success: true,\n            error: null,\n            phpVersion,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to change PHP version:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            phpVersion: null,\n          };\n        }\n      },\n\n      importSite: async (_parent: any, args: { input: { zipPath: string; siteName?: string } }) => {\n        const { zipPath, siteName } = args.input;\n        const fs = require('fs');\n\n        try {\n          if (!fs.existsSync(zipPath)) {\n            return {\n              success: false,\n              error: `Zip file not found: ${zipPath}`,\n              siteId: null,\n              siteName: null,\n            };\n          }\n\n          localLogger.info(`[${ADDON_NAME}] Importing site from ${zipPath}`);\n\n          if (!importSiteService) {\n            return {\n              success: false,\n              error: 'Import site service not available',\n              siteId: null,\n              siteName: null,\n            };\n          }\n\n          const result = await importSiteService.run({\n            zipPath,\n            siteName: siteName || undefined,\n          });\n\n          localLogger.info(`[${ADDON_NAME}] Successfully imported site: ${result.name}`);\n\n          return {\n            success: true,\n            error: null,\n            siteId: result.id,\n            siteName: result.name,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to import site:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            siteId: null,\n            siteName: null,\n          };\n        }\n      },\n\n      // Phase 9: Site Configuration & Dev Tools\n      toggleXdebug: async (_parent: any, args: { input: { siteId: string; enabled: boolean } }) => {\n        const { siteId, enabled } = args.input;\n\n        try {\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              error: `Site not found: ${siteId}`,\n              enabled: null,\n            };\n          }\n\n          localLogger.info(\n            `[${ADDON_NAME}] ${enabled ? 'Enabling' : 'Disabling'} Xdebug for ${site.name}`\n          );\n\n          // Update the site's xdebugEnabled property\n          await siteData.updateSite(siteId, { xdebugEnabled: enabled });\n\n          // Restart the site if it's running to apply the change\n          const status = await siteProcessManager.getSiteStatus(site);\n          if (status === 'running') {\n            localLogger.info(`[${ADDON_NAME}] Restarting site to apply Xdebug change`);\n            await siteProcessManager.restart(site);\n          }\n\n          localLogger.info(\n            `[${ADDON_NAME}] Successfully ${enabled ? 'enabled' : 'disabled'} Xdebug`\n          );\n\n          return {\n            success: true,\n            error: null,\n            enabled,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to toggle Xdebug:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            enabled: null,\n          };\n        }\n      },\n\n      getSiteLogs: async (\n        _parent: any,\n        args: { input: { siteId: string; logType?: string; lines?: number } }\n      ) => {\n        const { siteId, logType = 'php', lines = 100 } = args.input;\n        const fs = require('fs');\n        const fsPromises = fs.promises;\n        const pathModule = require('path');\n\n        // Helper for async file existence check\n        const fileExists = async (filePath: string): Promise<boolean> => {\n          try {\n            await fsPromises.access(filePath);\n            return true;\n          } catch {\n            return false;\n          }\n        };\n\n        // Helper to read last N lines of a file\n        const readLastLines = async (filePath: string, numLines: number): Promise<string> => {\n          try {\n            const content = await fsPromises.readFile(filePath, 'utf-8');\n            const logLines = content.split('\\n');\n            return logLines.slice(-numLines).join('\\n') || '(empty)';\n          } catch {\n            return '';\n          }\n        };\n\n        try {\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              error: `Site not found: ${siteId}`,\n              logs: [],\n            };\n          }\n\n          localLogger.info(`[${ADDON_NAME}] Getting ${logType} logs for ${site.name}`);\n\n          const logs: Array<{ type: string; content: string; path: string }> = [];\n          const logsDir = pathModule.join(site.path, 'logs');\n\n          const logFiles: Record<string, string[]> = {\n            php: ['php', 'php-fpm'],\n            nginx: ['nginx'],\n            mysql: ['mysql'],\n            all: ['php', 'php-fpm', 'nginx', 'mysql'],\n          };\n\n          const targetLogs = logFiles[logType] || logFiles.php;\n\n          for (const logName of targetLogs) {\n            // Check for error and access logs\n            for (const suffix of ['error.log', 'access.log', '.log']) {\n              const logPath = pathModule.join(\n                logsDir,\n                `${logName}${suffix === '.log' ? '' : '/'}${suffix}`\n              );\n              const altLogPath = pathModule.join(logsDir, `${logName}${suffix}`);\n\n              let finalPath: string | null = null;\n              if (await fileExists(logPath)) {\n                finalPath = logPath;\n              } else if (await fileExists(altLogPath)) {\n                finalPath = altLogPath;\n              }\n\n              if (finalPath) {\n                const content = await readLastLines(finalPath, lines);\n                if (content) {\n                  logs.push({\n                    type: logName,\n                    content,\n                    path: finalPath,\n                  });\n                }\n              }\n            }\n          }\n\n          if (logs.length === 0) {\n            // Try to find any log files\n            if (await fileExists(logsDir)) {\n              const entries = await fsPromises.readdir(logsDir, { withFileTypes: true });\n              for (const entry of entries) {\n                if (entry.isDirectory()) {\n                  const subDir = pathModule.join(logsDir, entry.name);\n                  const subEntries = await fsPromises.readdir(subDir);\n                  for (const subFile of subEntries) {\n                    if (subFile.endsWith('.log')) {\n                      const logPath = pathModule.join(subDir, subFile);\n                      const content = await readLastLines(logPath, lines);\n                      if (content) {\n                        logs.push({\n                          type: entry.name,\n                          content,\n                          path: logPath,\n                        });\n                      }\n                    }\n                  }\n                } else if (entry.name.endsWith('.log')) {\n                  const logPath = pathModule.join(logsDir, entry.name);\n                  const content = await readLastLines(logPath, lines);\n                  if (content) {\n                    logs.push({\n                      type: entry.name.replace('.log', ''),\n                      content,\n                      path: logPath,\n                    });\n                  }\n                }\n              }\n            }\n          }\n\n          return {\n            success: true,\n            error: null,\n            logs,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to get logs:`, error);\n          return {\n            success: false,\n            error: error.message || 'Unknown error',\n            logs: [],\n          };\n        }\n      },\n\n      // Phase 10: Cloud Backup Mutations\n      createBackup: async (\n        _parent: any,\n        args: { siteId: string; provider: string; note?: string }\n      ) => {\n        const { siteId, provider, note } = args;\n\n        try {\n          localLogger.info(`[${ADDON_NAME}] Creating backup for site ${siteId} to ${provider}`);\n\n          // Get site\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              snapshotId: null,\n              timestamp: null,\n              message: null,\n              error: `Site not found: ${siteId}`,\n            };\n          }\n\n          // Get providers from Cloud Backups addon\n          const providers = await getBackupProviders();\n          if (providers.length === 0) {\n            return {\n              success: false,\n              snapshotId: null,\n              timestamp: null,\n              message: null,\n              error:\n                'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub.',\n            };\n          }\n\n          // Find the matching provider (map 'googleDrive' to 'google' for the addon)\n          const providerMap: Record<string, string> = { googleDrive: 'google', dropbox: 'dropbox' };\n          const providerId = providerMap[provider] || provider;\n          const matchedProvider = providers.find((p: any) => p.id === providerId);\n\n          if (!matchedProvider) {\n            return {\n              success: false,\n              snapshotId: null,\n              timestamp: null,\n              message: null,\n              error: `Provider '${provider}' not configured. Available: ${providers.map((p: any) => p.name).join(', ')}`,\n            };\n          }\n\n          // Map the Hub provider ID to rclone backend name\n          // The addon uses 'google' in enabled-providers but expects 'drive' for backup operations\n          const backupProviderMap: Record<string, string> = { google: 'drive', dropbox: 'dropbox' };\n          const backupProviderId = backupProviderMap[matchedProvider.id] || matchedProvider.id;\n          localLogger.info(\n            `[${ADDON_NAME}] Using backup provider ID: ${backupProviderId} (from ${matchedProvider.id})`\n          );\n\n          // Create backup via IPC (use long timeout for backup operations)\n          const description = note || 'Backup created via MCP';\n          const result = await invokeBackupIPC(\n            'backups:backup-site',\n            BACKUP_IPC_TIMEOUT,\n            siteId,\n            backupProviderId,\n            description\n          );\n          localLogger.info(`[${ADDON_NAME}] Backup IPC result: ${JSON.stringify(result)}`);\n\n          // Check for top-level IPC error\n          if (result.error) {\n            return {\n              success: false,\n              snapshotId: null,\n              timestamp: null,\n              message: null,\n              error: result.error.message || 'Backup creation failed',\n            };\n          }\n\n          // Unwrap nested result structure - the actual result is at result.result\n          const backupResult = result.result;\n\n          // Check if the backup result contains an error (nested at result.result.error)\n          if (backupResult?.error) {\n            const errorMsg =\n              backupResult.error.message || backupResult.error.original?.message || 'Backup failed';\n            return {\n              success: false,\n              snapshotId: null,\n              timestamp: null,\n              message: null,\n              error: errorMsg,\n            };\n          }\n\n          // Try to extract snapshot ID (may be nested in result.result.result)\n          let snapshotId = backupResult?.snapshotId || backupResult?.id;\n          if (!snapshotId && backupResult?.result) {\n            snapshotId = backupResult.result.snapshotId || backupResult.result.id;\n          }\n\n          // If no error was returned, the backup succeeded even if no snapshot ID is provided\n          // The addon doesn't always return the snapshot ID in its IPC response\n          return {\n            success: true,\n            snapshotId: snapshotId || null,\n            timestamp: new Date().toISOString(),\n            message: `Backup created successfully to ${matchedProvider.name}`,\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to create backup:`, error);\n          return {\n            success: false,\n            snapshotId: null,\n            timestamp: null,\n            message: null,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n\n      restoreBackup: async (\n        _parent: any,\n        args: { siteId: string; provider: string; snapshotId: string; confirm?: boolean }\n      ) => {\n        const { siteId, provider, snapshotId, confirm = false } = args;\n\n        try {\n          localLogger.info(`[${ADDON_NAME}] Restoring backup ${snapshotId} for site ${siteId}`);\n\n          // Check confirmation\n          if (!confirm) {\n            return {\n              success: false,\n              message: null,\n              error:\n                'Restore requires confirm=true to prevent accidental data loss. Current site files and database will be overwritten.',\n            };\n          }\n\n          // Get site\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              message: null,\n              error: `Site not found: ${siteId}`,\n            };\n          }\n\n          // Get providers from Cloud Backups addon\n          const providers = await getBackupProviders();\n          if (providers.length === 0) {\n            return {\n              success: false,\n              message: null,\n              error:\n                'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub.',\n            };\n          }\n\n          // Find the matching provider (map 'googleDrive' to 'google' for the addon)\n          const providerMap: Record<string, string> = { googleDrive: 'google', dropbox: 'dropbox' };\n          const providerId = providerMap[provider] || provider;\n          const matchedProvider = providers.find((p: any) => p.id === providerId);\n\n          if (!matchedProvider) {\n            return {\n              success: false,\n              message: null,\n              error: `Provider '${provider}' not configured. Available: ${providers.map((p: any) => p.name).join(', ')}`,\n            };\n          }\n\n          // Map the Hub provider ID to rclone backend name\n          const backupProviderMap: Record<string, string> = { google: 'drive', dropbox: 'dropbox' };\n          const backupProviderId = backupProviderMap[matchedProvider.id] || matchedProvider.id;\n\n          // Restore backup via IPC (use long timeout for restore operations)\n          const result = await invokeBackupIPC(\n            'backups:restore-backup',\n            BACKUP_IPC_TIMEOUT,\n            siteId,\n            backupProviderId,\n            snapshotId\n          );\n          localLogger.info(`[${ADDON_NAME}] Restore result: ${JSON.stringify(result)}`);\n\n          // Check for errors - can be at result.error or result.result.error (IPC async pattern)\n          const ipcError = result.error || result.result?.error;\n          if (ipcError) {\n            const errorMessage =\n              typeof ipcError === 'string' ? ipcError : ipcError.message || 'Restore failed';\n            return {\n              success: false,\n              message: null,\n              error: errorMessage,\n            };\n          }\n\n          return {\n            success: true,\n            message: `Site restored from backup ${snapshotId}`,\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to restore backup:`, error);\n          return {\n            success: false,\n            message: null,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n\n      deleteBackup: async (\n        _parent: any,\n        args: { siteId: string; provider: string; snapshotId: string; confirm?: boolean }\n      ) => {\n        const { siteId, provider, snapshotId, confirm = false } = args;\n\n        try {\n          localLogger.info(`[${ADDON_NAME}] Deleting backup ${snapshotId} for site ${siteId}`);\n\n          // Check confirmation\n          if (!confirm) {\n            return {\n              success: false,\n              deletedSnapshotId: null,\n              message: null,\n              error: 'Delete requires confirm=true to prevent accidental deletion.',\n            };\n          }\n\n          // Get site\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              deletedSnapshotId: null,\n              message: null,\n              error: `Site not found: ${siteId}`,\n            };\n          }\n\n          // Get providers from Cloud Backups addon\n          const providers = await getBackupProviders();\n          if (providers.length === 0) {\n            return {\n              success: false,\n              deletedSnapshotId: null,\n              message: null,\n              error:\n                'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub.',\n            };\n          }\n\n          // Find the matching provider (map 'googleDrive' to 'google' for the addon)\n          const providerMap: Record<string, string> = { googleDrive: 'google', dropbox: 'dropbox' };\n          const providerId = providerMap[provider] || provider;\n          const matchedProvider = providers.find((p: any) => p.id === providerId);\n\n          if (!matchedProvider) {\n            return {\n              success: false,\n              deletedSnapshotId: null,\n              message: null,\n              error: `Provider '${provider}' not configured. Available: ${providers.map((p: any) => p.name).join(', ')}`,\n            };\n          }\n\n          // Map the Hub provider ID to rclone backend name\n          const backupProviderMap: Record<string, string> = { google: 'drive', dropbox: 'dropbox' };\n          const backupProviderId = backupProviderMap[matchedProvider.id] || matchedProvider.id;\n\n          // Try to delete backup via IPC (may not be supported by the addon)\n          const result = await invokeBackupIPC(\n            'backups:delete-backup',\n            DEFAULT_IPC_TIMEOUT,\n            siteId,\n            backupProviderId,\n            snapshotId\n          );\n\n          if (result.error) {\n            // If the IPC channel doesn't exist or isn't supported, provide helpful message\n            if (result.error.message?.includes('timed out')) {\n              return {\n                success: false,\n                deletedSnapshotId: null,\n                message: null,\n                error:\n                  'Delete backup operation is not available via MCP. Please delete backups through the Local UI.',\n              };\n            }\n            return {\n              success: false,\n              deletedSnapshotId: null,\n              message: null,\n              error: result.error.message || 'Delete failed',\n            };\n          }\n\n          return {\n            success: true,\n            deletedSnapshotId: snapshotId,\n            message: 'Backup deleted',\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to delete backup:`, error);\n          return {\n            success: false,\n            deletedSnapshotId: null,\n            message: null,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n\n      downloadBackup: async (\n        _parent: any,\n        args: { siteId: string; provider: string; snapshotId: string }\n      ) => {\n        const { siteId, provider, snapshotId } = args;\n\n        try {\n          localLogger.info(`[${ADDON_NAME}] Downloading backup ${snapshotId} for site ${siteId}`);\n\n          // Get site\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              filePath: null,\n              message: null,\n              error: `Site not found: ${siteId}`,\n            };\n          }\n\n          // Get providers from Cloud Backups addon\n          const providers = await getBackupProviders();\n          if (providers.length === 0) {\n            return {\n              success: false,\n              filePath: null,\n              message: null,\n              error:\n                'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub.',\n            };\n          }\n\n          // Find the matching provider (map 'googleDrive' to 'google' for the addon)\n          const providerMap: Record<string, string> = { googleDrive: 'google', dropbox: 'dropbox' };\n          const providerId = providerMap[provider] || provider;\n          const matchedProvider = providers.find((p: any) => p.id === providerId);\n\n          if (!matchedProvider) {\n            return {\n              success: false,\n              filePath: null,\n              message: null,\n              error: `Provider '${provider}' not configured. Available: ${providers.map((p: any) => p.name).join(', ')}`,\n            };\n          }\n\n          // Map the Hub provider ID to rclone backend name\n          const backupProviderMap: Record<string, string> = { google: 'drive', dropbox: 'dropbox' };\n          const backupProviderId = backupProviderMap[matchedProvider.id] || matchedProvider.id;\n\n          // Try to download backup via IPC (use long timeout for downloads)\n          const result = await invokeBackupIPC(\n            'backups:download-backup',\n            BACKUP_IPC_TIMEOUT,\n            siteId,\n            backupProviderId,\n            snapshotId\n          );\n\n          if (result.error) {\n            // If the IPC channel doesn't exist or isn't supported, provide helpful message\n            if (result.error.message?.includes('timed out')) {\n              return {\n                success: false,\n                filePath: null,\n                message: null,\n                error:\n                  'Download backup operation is not available via MCP. Please download backups through the Local UI.',\n              };\n            }\n            return {\n              success: false,\n              filePath: null,\n              message: null,\n              error: result.error.message || 'Download failed',\n            };\n          }\n\n          return {\n            success: true,\n            filePath: result.result?.filePath || null,\n            message: 'Backup downloaded to Downloads folder',\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to download backup:`, error);\n          return {\n            success: false,\n            filePath: null,\n            message: null,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n\n      editBackupNote: async (\n        _parent: any,\n        args: { siteId: string; provider: string; snapshotId: string; note: string }\n      ) => {\n        const { siteId, provider, snapshotId, note } = args;\n\n        try {\n          localLogger.info(`[${ADDON_NAME}] Editing backup note for ${snapshotId}`);\n\n          // Get site\n          const site = siteData.getSite(siteId);\n          if (!site) {\n            return {\n              success: false,\n              snapshotId: null,\n              note: null,\n              error: `Site not found: ${siteId}`,\n            };\n          }\n\n          // Get providers from Cloud Backups addon\n          const providers = await getBackupProviders();\n          if (providers.length === 0) {\n            return {\n              success: false,\n              snapshotId: null,\n              note: null,\n              error:\n                'No cloud storage providers configured. Connect Google Drive or Dropbox in Local Hub.',\n            };\n          }\n\n          // Find the matching provider (map 'googleDrive' to 'google' for the addon)\n          const providerMap: Record<string, string> = { googleDrive: 'google', dropbox: 'dropbox' };\n          const providerId = providerMap[provider] || provider;\n          const matchedProvider = providers.find((p: any) => p.id === providerId);\n\n          if (!matchedProvider) {\n            return {\n              success: false,\n              snapshotId: null,\n              note: null,\n              error: `Provider '${provider}' not configured. Available: ${providers.map((p: any) => p.name).join(', ')}`,\n            };\n          }\n\n          // Map the Hub provider ID to rclone backend name\n          const backupProviderMap: Record<string, string> = { google: 'drive', dropbox: 'dropbox' };\n          const backupProviderId = backupProviderMap[matchedProvider.id] || matchedProvider.id;\n\n          // Try to edit backup note via IPC (quick metadata operation)\n          const result = await invokeBackupIPC(\n            'backups:edit-note',\n            DEFAULT_IPC_TIMEOUT,\n            siteId,\n            backupProviderId,\n            snapshotId,\n            note\n          );\n\n          if (result.error) {\n            // If the IPC channel doesn't exist or isn't supported, provide helpful message\n            if (result.error.message?.includes('timed out')) {\n              return {\n                success: false,\n                snapshotId: null,\n                note: null,\n                error:\n                  'Edit backup note operation is not available via MCP. Please edit backup notes through the Local UI.',\n              };\n            }\n            return {\n              success: false,\n              snapshotId: null,\n              note: null,\n              error: result.error.message || 'Edit note failed',\n            };\n          }\n\n          return {\n            success: true,\n            snapshotId,\n            note,\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to edit backup note:`, error);\n          return {\n            success: false,\n            snapshotId: null,\n            note: null,\n            error: error.message || 'Unknown error',\n          };\n        }\n      },\n\n      // Phase 11: WP Engine Connect\n      wpeAuthenticate: async () => {\n        try {\n          localLogger.info(`[${ADDON_NAME}] Initiating WP Engine authentication`);\n\n          if (!wpeOAuthService) {\n            return {\n              success: false,\n              email: null,\n              message: null,\n              error: 'WP Engine OAuth service not available',\n            };\n          }\n\n          // Trigger OAuth flow - this will open browser for user consent\n          // authenticate() returns OAuthTokens on success\n          const tokens = await wpeOAuthService.authenticate();\n\n          if (tokens && tokens.accessToken) {\n            // Try to get user email from CAPI\n            let email = null;\n            if (capiService) {\n              try {\n                const currentUser = await capiService.getCurrentUser();\n                email = currentUser?.email || null;\n              } catch {\n                // User info not available\n              }\n            }\n\n            localLogger.info(\n              `[${ADDON_NAME}] Successfully authenticated with WPE${email ? ` as ${email}` : ''}`\n            );\n            return {\n              success: true,\n              email,\n              message: 'Successfully authenticated with WP Engine',\n              error: null,\n            };\n          }\n\n          return {\n            success: true,\n            email: null,\n            message: 'Authentication initiated. Please complete the login in your browser.',\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] WPE authentication failed:`, error);\n          return {\n            success: false,\n            email: null,\n            message: null,\n            error: error.message || 'Authentication failed',\n          };\n        }\n      },\n\n      wpeLogout: async () => {\n        try {\n          localLogger.info(`[${ADDON_NAME}] Logging out from WP Engine`);\n\n          if (!wpeOAuthService) {\n            return {\n              success: false,\n              message: null,\n              error: 'WP Engine OAuth service not available',\n            };\n          }\n\n          // clearTokens() is the logout method\n          await wpeOAuthService.clearTokens();\n\n          localLogger.info(`[${ADDON_NAME}] Successfully logged out from WPE`);\n          return {\n            success: true,\n            message: 'Logged out from WP Engine',\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] WPE logout failed:`, error);\n          return {\n            success: false,\n            message: null,\n            error: error.message || 'Logout failed',\n          };\n        }\n      },\n\n      // Phase 11c: Push to WP Engine\n      pushToWpe: async (\n        _parent: any,\n        args: {\n          localSiteId: string;\n          remoteInstallId: string;\n          includeSql?: boolean;\n          confirm?: boolean;\n        }\n      ) => {\n        const { localSiteId, remoteInstallId, includeSql = false, confirm = false } = args;\n\n        try {\n          localLogger.info(\n            `[${ADDON_NAME}] Push to WPE: site=${localSiteId}, remote=${remoteInstallId}, includeSql=${includeSql}`\n          );\n\n          // Require confirmation for push operations\n          if (!confirm) {\n            return {\n              success: false,\n              message: null,\n              error:\n                'Push requires confirm=true to prevent accidental overwrites. Set confirm=true to proceed.',\n            };\n          }\n\n          // Verify site exists\n          const site = siteData.getSite(localSiteId);\n          if (!site) {\n            return {\n              success: false,\n              message: null,\n              error: `Site not found: ${localSiteId}`,\n            };\n          }\n\n          // Check WPE connection exists\n          const wpeConnection = site.hostConnections?.find((c: any) => c.hostId === 'wpe');\n          if (!wpeConnection) {\n            return {\n              success: false,\n              message: null,\n              error:\n                'Site is not linked to WP Engine. Use Connect in Local to link the site first.',\n            };\n          }\n\n          // Check push service availability\n          if (!wpePushService || typeof wpePushService.push !== 'function') {\n            return {\n              success: false,\n              message: null,\n              error: 'WPE Push service not available',\n            };\n          }\n\n          // Get install details from CAPI to get required parameters\n          let installName = remoteInstallId;\n          let primaryDomain = '';\n          let installId = '';\n\n          if (capiService && typeof capiService.getInstallList === 'function') {\n            const installs = await capiService.getInstallList();\n            const matchingInstall = installs?.find(\n              (i: any) =>\n                i.site?.id === wpeConnection.remoteSiteId &&\n                (!wpeConnection.remoteSiteEnv || i.environment === wpeConnection.remoteSiteEnv)\n            );\n            if (matchingInstall) {\n              installName = matchingInstall.name;\n              primaryDomain =\n                matchingInstall.primary_domain ||\n                matchingInstall.cname ||\n                `${matchingInstall.name}.wpengine.com`;\n              installId = matchingInstall.id;\n            }\n          }\n\n          if (!primaryDomain) {\n            return {\n              success: false,\n              message: null,\n              error:\n                'Could not determine WP Engine install details. Please ensure you are authenticated.',\n            };\n          }\n\n          // Start the push operation (async - returns immediately)\n          wpePushService\n            .push({\n              includeSql,\n              wpengineInstallName: installName,\n              wpengineInstallId: installId,\n              wpengineSiteId: wpeConnection.remoteSiteId,\n              wpenginePrimaryDomain: primaryDomain,\n              localSiteId: site.id,\n              environment: wpeConnection.remoteSiteEnv,\n              isMagicSync: false,\n            })\n            .catch((err: any) => {\n              localLogger.error(`[${ADDON_NAME}] Push failed:`, err);\n            });\n\n          return {\n            success: true,\n            message: `Push started to ${installName}. Check Local UI for progress.`,\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to start push:`, error);\n          return {\n            success: false,\n            message: null,\n            error: error.message || 'Failed to start push',\n          };\n        }\n      },\n\n      // Phase 11c: Pull from WP Engine\n      pullFromWpe: async (\n        _parent: any,\n        args: {\n          localSiteId: string;\n          remoteInstallId: string;\n          includeSql?: boolean;\n        }\n      ) => {\n        const { localSiteId, remoteInstallId, includeSql = false } = args;\n\n        try {\n          localLogger.info(\n            `[${ADDON_NAME}] Pull from WPE: site=${localSiteId}, remote=${remoteInstallId}, includeSql=${includeSql}`\n          );\n\n          // Verify site exists\n          const site = siteData.getSite(localSiteId);\n          if (!site) {\n            return {\n              success: false,\n              message: null,\n              error: `Site not found: ${localSiteId}`,\n            };\n          }\n\n          // Check WPE connection exists\n          const wpeConnection = site.hostConnections?.find((c: any) => c.hostId === 'wpe');\n          if (!wpeConnection) {\n            return {\n              success: false,\n              message: null,\n              error:\n                'Site is not linked to WP Engine. Use Connect in Local to link the site first.',\n            };\n          }\n\n          // Check pull service availability\n          if (!wpePullService || typeof wpePullService.pull !== 'function') {\n            return {\n              success: false,\n              message: null,\n              error: 'WPE Pull service not available',\n            };\n          }\n\n          // Get install details from CAPI\n          let installName = remoteInstallId;\n          let primaryDomain = '';\n          let installId = '';\n\n          if (capiService && typeof capiService.getInstallList === 'function') {\n            const installs = await capiService.getInstallList();\n            const matchingInstall = installs?.find(\n              (i: any) =>\n                i.site?.id === wpeConnection.remoteSiteId &&\n                (!wpeConnection.remoteSiteEnv || i.environment === wpeConnection.remoteSiteEnv)\n            );\n            if (matchingInstall) {\n              installName = matchingInstall.name;\n              primaryDomain =\n                matchingInstall.primary_domain ||\n                matchingInstall.cname ||\n                `${matchingInstall.name}.wpengine.com`;\n              installId = matchingInstall.id;\n            }\n          }\n\n          if (!primaryDomain) {\n            return {\n              success: false,\n              message: null,\n              error:\n                'Could not determine WP Engine install details. Please ensure you are authenticated.',\n            };\n          }\n\n          // Start the pull operation (async - returns immediately)\n          wpePullService\n            .pull({\n              includeSql,\n              wpengineInstallName: installName,\n              wpengineInstallId: installId,\n              wpengineSiteId: wpeConnection.remoteSiteId,\n              wpenginePrimaryDomain: primaryDomain,\n              localSiteId: site.id,\n              environment: wpeConnection.remoteSiteEnv,\n              isMagicSync: false,\n            })\n            .catch((err: any) => {\n              localLogger.error(`[${ADDON_NAME}] Pull failed:`, err);\n            });\n\n          return {\n            success: true,\n            message: `Pull started from ${installName}. Check Local UI for progress.`,\n            error: null,\n          };\n        } catch (error: any) {\n          localLogger.error(`[${ADDON_NAME}] Failed to start pull:`, error);\n          return {\n            success: false,\n            message: null,\n            error: error.message || 'Failed to start pull',\n          };\n        }\n      },\n    },\n  };\n}\n\n/**\n * Start the MCP server\n */\nasync function startMcpServer(services: LocalServices, logger: any): Promise<void> {\n  if (mcpServer) {\n    logger.warn(`[${ADDON_NAME}] MCP server already running`);\n    return;\n  }\n\n  try {\n    mcpServer = new McpServer({ port: MCP_SERVER.DEFAULT_PORT }, services, logger);\n\n    await mcpServer.start();\n\n    const info = mcpServer.getConnectionInfo();\n    logger.info(`[${ADDON_NAME}] MCP server started on port ${info.port}`);\n    logger.info(\n      `[${ADDON_NAME}] MCP connection info saved to: ~/Library/Application Support/Local/mcp-connection-info.json`\n    );\n    logger.info(`[${ADDON_NAME}] Available tools: ${info.tools.join(', ')}`);\n  } catch (error: any) {\n    logger.error(`[${ADDON_NAME}] Failed to start MCP server:`, error);\n  }\n}\n\n/**\n * Stop the MCP server\n */\nasync function stopMcpServer(logger: any): Promise<void> {\n  if (mcpServer) {\n    await mcpServer.stop();\n    mcpServer = null;\n    logger.info(`[${ADDON_NAME}] MCP server stopped`);\n  }\n}\n\n/**\n * Register IPC handlers for renderer communication\n */\nfunction registerIpcHandlers(services: LocalServices, logger: any): void {\n  // Get MCP server status\n  ipcMain.handle('mcp:getStatus', async () => {\n    if (!mcpServer) {\n      return { running: false, port: 0, uptime: 0 };\n    }\n    return mcpServer.getStatus();\n  });\n\n  // Get connection info\n  ipcMain.handle('mcp:getConnectionInfo', async () => {\n    if (!mcpServer) {\n      return null;\n    }\n    return mcpServer.getConnectionInfo();\n  });\n\n  // Start MCP server\n  ipcMain.handle('mcp:start', async () => {\n    try {\n      await startMcpServer(services, logger);\n      return { success: true };\n    } catch (error: any) {\n      return { success: false, error: error.message };\n    }\n  });\n\n  // Stop MCP server\n  ipcMain.handle('mcp:stop', async () => {\n    try {\n      await stopMcpServer(logger);\n      return { success: true };\n    } catch (error: any) {\n      return { success: false, error: error.message };\n    }\n  });\n\n  // Restart MCP server\n  ipcMain.handle('mcp:restart', async () => {\n    try {\n      await stopMcpServer(logger);\n      await startMcpServer(services, logger);\n      return { success: true };\n    } catch (error: any) {\n      return { success: false, error: error.message };\n    }\n  });\n\n  // Regenerate auth token\n  ipcMain.handle('mcp:regenerateToken', async () => {\n    if (!mcpServer) {\n      return { success: false, error: 'MCP server not running' };\n    }\n    try {\n      const newToken = await mcpServer.regenerateToken();\n      return { success: true, token: newToken };\n    } catch (error: any) {\n      return { success: false, error: error.message };\n    }\n  });\n\n  logger.info(\n    `[${ADDON_NAME}] Registered IPC handlers: mcp:getStatus, mcp:getConnectionInfo, mcp:start, mcp:stop, mcp:restart, mcp:regenerateToken`\n  );\n}\n\n/**\n * Main addon initialization function\n */\nexport default function (_context: LocalMain.AddonMainContext): void {\n  const services = LocalMain.getServiceContainer().cradle as any;\n  const { localLogger, graphql } = services;\n\n  try {\n    localLogger.info(`[${ADDON_NAME}] Initializing...`);\n\n    // Register GraphQL extensions (for local-cli and MCP)\n    const resolvers = createResolvers(services);\n    graphql.registerGraphQLService('mcp-server', typeDefs, resolvers);\n    localLogger.info(`[${ADDON_NAME}] Registered GraphQL: 29 tools (Phase 1-11b)`);\n\n    // Start MCP server (for AI tools)\n    const localServices: LocalServices = {\n      siteData: services.siteData,\n      siteProcessManager: services.siteProcessManager,\n      wpCli: services.wpCli,\n      deleteSite: services.deleteSite,\n      addSite: services.addSite,\n      localLogger: services.localLogger,\n      adminer: services.adminer,\n      x509Cert: services.x509Cert,\n      siteProvisioner: services.siteProvisioner,\n      importSite: services.importSite,\n      lightningServices: services.lightningServices,\n      // Phase 11: WP Engine Connect\n      wpeOAuth: services.wpeOAuth,\n      capi: services.capi,\n    };\n\n    // MCP server disabled - CLI-only mode\n    // To enable MCP server for AI tool integration, uncomment the following:\n    // startMcpServer(localServices, localLogger);\n    // registerIpcHandlers(localServices, localLogger);\n\n    localLogger.info(`[${ADDON_NAME}] Successfully initialized (CLI-only mode)`);\n  } catch (error: any) {\n    localLogger.error(`[${ADDON_NAME}] Failed to initialize:`, error);\n  }\n}\n"]}
|