@_davideast/stitch-mcp 0.5.2 → 0.5.3
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- package/dist/chunk-22bymmh9.js +137 -0
- package/dist/chunk-22bymmh9.js.map +10 -0
- package/dist/chunk-387yyye2.js +248 -0
- package/dist/chunk-387yyye2.js.map +14 -0
- package/dist/chunk-3ff2k44g.js +19 -0
- package/dist/chunk-3ff2k44g.js.map +9 -0
- package/dist/chunk-46cay1zp.js +273 -0
- package/dist/chunk-46cay1zp.js.map +10 -0
- package/dist/chunk-48e1jpm8.js +167 -0
- package/dist/chunk-48e1jpm8.js.map +10 -0
- package/dist/chunk-4arzkk1s.js +69 -0
- package/dist/chunk-4arzkk1s.js.map +10 -0
- package/dist/chunk-4js7vw6h.js +415 -0
- package/dist/chunk-4js7vw6h.js.map +20 -0
- package/dist/chunk-4p1wfk3t.js +736 -0
- package/dist/chunk-4p1wfk3t.js.map +16 -0
- package/dist/chunk-4zg969tc.js +370 -0
- package/dist/chunk-4zg969tc.js.map +16 -0
- package/dist/chunk-5jjt7tgw.js +247 -0
- package/dist/chunk-5jjt7tgw.js.map +14 -0
- package/dist/chunk-5t2h8z2n.js +273 -0
- package/dist/chunk-5t2h8z2n.js.map +10 -0
- package/dist/chunk-5zy3et1m.js +759 -0
- package/dist/chunk-5zy3et1m.js.map +19 -0
- package/dist/chunk-6wvst7z8.js +125 -0
- package/dist/chunk-6wvst7z8.js.map +12 -0
- package/dist/chunk-8mm08arm.js +256 -0
- package/dist/chunk-8mm08arm.js.map +11 -0
- package/dist/chunk-8nv4wwv0.js +19 -0
- package/dist/chunk-8nv4wwv0.js.map +9 -0
- package/dist/chunk-9cjny9z2.js +19 -0
- package/dist/chunk-9cjny9z2.js.map +9 -0
- package/dist/chunk-9ggfw78s.js +19 -0
- package/dist/chunk-9ggfw78s.js.map +9 -0
- package/dist/chunk-9nyyn1hn.js +125 -0
- package/dist/chunk-9nyyn1hn.js.map +12 -0
- package/dist/chunk-9nzbvp6r.js +19 -0
- package/dist/chunk-9nzbvp6r.js.map +9 -0
- package/dist/chunk-a0gmbh8e.js +415 -0
- package/dist/chunk-a0gmbh8e.js.map +20 -0
- package/dist/chunk-acv998mp.js +941 -0
- package/dist/chunk-acv998mp.js.map +27 -0
- package/dist/chunk-cpjsvven.js +10 -0
- package/dist/chunk-cpjsvven.js.map +9 -0
- package/dist/chunk-dmrahbs1.js +31467 -0
- package/dist/chunk-dmrahbs1.js.map +245 -0
- package/dist/chunk-dts6851a.js +137 -0
- package/dist/chunk-dts6851a.js.map +10 -0
- package/dist/chunk-efd0rm6g.js +69 -0
- package/dist/chunk-efd0rm6g.js.map +10 -0
- package/dist/chunk-f1kjn6cd.js +16554 -0
- package/dist/chunk-f1kjn6cd.js.map +98 -0
- package/dist/chunk-f5f6ekgp.js +109 -0
- package/dist/chunk-f5f6ekgp.js.map +10 -0
- package/dist/chunk-fmewhfh3.js +137 -0
- package/dist/chunk-fmewhfh3.js.map +10 -0
- package/dist/chunk-fmhrn6cg.js +6382 -0
- package/dist/chunk-fmhrn6cg.js.map +109 -0
- package/dist/chunk-gh7dzfhb.js +256 -0
- package/dist/chunk-gh7dzfhb.js.map +11 -0
- package/dist/chunk-gwhtc0fe.js +370 -0
- package/dist/chunk-gwhtc0fe.js.map +16 -0
- package/dist/chunk-hc7sdwmn.js +449 -0
- package/dist/chunk-hc7sdwmn.js.map +17 -0
- package/dist/chunk-k4xwb3wp.js +44184 -0
- package/dist/chunk-k4xwb3wp.js.map +237 -0
- package/dist/chunk-k751yfm6.js +736 -0
- package/dist/chunk-k751yfm6.js.map +16 -0
- package/dist/chunk-kkc2tvar.js +19 -0
- package/dist/chunk-kkc2tvar.js.map +9 -0
- package/dist/chunk-kva47mgc.js +10 -0
- package/dist/chunk-kva47mgc.js.map +9 -0
- package/dist/chunk-nv2j020p.js +45210 -0
- package/dist/chunk-nv2j020p.js.map +258 -0
- package/dist/chunk-pdj9s41r.js +683 -0
- package/dist/chunk-pdj9s41r.js.map +17 -0
- package/dist/chunk-ppphsy4t.js +273 -0
- package/dist/chunk-ppphsy4t.js.map +10 -0
- package/dist/chunk-psmw9zpa.js +941 -0
- package/dist/chunk-psmw9zpa.js.map +27 -0
- package/dist/chunk-pz31v3ma.js +24 -0
- package/dist/chunk-pz31v3ma.js.map +9 -0
- package/dist/chunk-qbt0906e.js +1495 -0
- package/dist/chunk-qbt0906e.js.map +23 -0
- package/dist/chunk-rh3k09h7.js +19 -0
- package/dist/chunk-rh3k09h7.js.map +9 -0
- package/dist/chunk-saznae7w.js +10 -0
- package/dist/chunk-saznae7w.js.map +9 -0
- package/dist/chunk-sscqdg1j.js +69 -0
- package/dist/chunk-sscqdg1j.js.map +10 -0
- package/dist/chunk-stgj6y85.js +835 -0
- package/dist/chunk-stgj6y85.js.map +19 -0
- package/dist/chunk-t3tqmxyj.js +42 -0
- package/dist/chunk-t3tqmxyj.js.map +9 -0
- package/dist/chunk-v8ak35zd.js +94 -0
- package/dist/chunk-v8ak35zd.js.map +10 -0
- package/dist/chunk-w9acw256.js +167 -0
- package/dist/chunk-w9acw256.js.map +10 -0
- package/dist/chunk-wq60thjn.js +167 -0
- package/dist/chunk-wq60thjn.js.map +10 -0
- package/dist/chunk-wzkdeb8a.js +736 -0
- package/dist/chunk-wzkdeb8a.js.map +16 -0
- package/dist/chunk-xxv6j815.js +256 -0
- package/dist/chunk-xxv6j815.js.map +11 -0
- package/dist/chunk-yjnj35q8.js +1495 -0
- package/dist/chunk-yjnj35q8.js.map +23 -0
- package/dist/chunk-yvzzrczy.js +45206 -0
- package/dist/chunk-yvzzrczy.js.map +258 -0
- package/dist/chunk-ywax1akt.js +370 -0
- package/dist/chunk-ywax1akt.js.map +16 -0
- package/dist/chunk-zakq9pc0.js +4989 -0
- package/dist/chunk-zakq9pc0.js.map +65 -0
- package/dist/commands/doctor/command.js +1 -1
- package/dist/commands/doctor/steps/AdcProjectCheckStep.d.ts +8 -0
- package/dist/commands/init/command.js +1 -1
- package/dist/commands/logout/command.js +1 -1
- package/dist/commands/proxy/command.js +1 -1
- package/dist/commands/screens/command.js +2 -2
- package/dist/commands/serve/command.js +3 -3
- package/dist/commands/site/command.js +1 -1
- package/dist/commands/snapshot/command.js +1 -1
- package/dist/commands/tool/command.js +1 -1
- package/dist/commands/view/command.js +1 -1
- package/dist/index.js +4 -4
- package/dist/lib/server/AssetGateway.d.ts +12 -0
- package/package.json +1 -1
|
@@ -0,0 +1,256 @@
|
|
|
1
|
+
import {
|
|
2
|
+
require_lib
|
|
3
|
+
} from "./chunk-w8q7nsm7.js";
|
|
4
|
+
import {
|
|
5
|
+
theme
|
|
6
|
+
} from "./chunk-kbtqrkwh.js";
|
|
7
|
+
import"./chunk-3sfn889r.js";
|
|
8
|
+
import {
|
|
9
|
+
__require,
|
|
10
|
+
__toESM
|
|
11
|
+
} from "./chunk-9wyra8hs.js";
|
|
12
|
+
|
|
13
|
+
// src/commands/snapshot/handler.ts
|
|
14
|
+
var import_fs_extra = __toESM(require_lib(), 1);
|
|
15
|
+
|
|
16
|
+
// src/framework/MockUI.ts
|
|
17
|
+
class MockUI {
|
|
18
|
+
data;
|
|
19
|
+
constructor(data) {
|
|
20
|
+
this.data = data;
|
|
21
|
+
}
|
|
22
|
+
async promptMcpClient() {
|
|
23
|
+
if (!this.data.mcpClient) {
|
|
24
|
+
throw new Error("MockUI: Missing data for mcpClient");
|
|
25
|
+
}
|
|
26
|
+
return this.data.mcpClient;
|
|
27
|
+
}
|
|
28
|
+
async promptAuthMode() {
|
|
29
|
+
if (!this.data.authMode) {
|
|
30
|
+
throw new Error("MockUI: Missing data for authMode");
|
|
31
|
+
}
|
|
32
|
+
return this.data.authMode;
|
|
33
|
+
}
|
|
34
|
+
async promptTransportType(authMode) {
|
|
35
|
+
if (this.data.transportType) {
|
|
36
|
+
return this.data.transportType;
|
|
37
|
+
}
|
|
38
|
+
throw new Error("MockUI: Missing data for transportType");
|
|
39
|
+
}
|
|
40
|
+
async promptApiKeyStorage() {
|
|
41
|
+
if (!this.data.apiKeyStorage) {
|
|
42
|
+
throw new Error("MockUI: Missing data for apiKeyStorage");
|
|
43
|
+
}
|
|
44
|
+
return this.data.apiKeyStorage;
|
|
45
|
+
}
|
|
46
|
+
async promptApiKey() {
|
|
47
|
+
if (!this.data.apiKey) {
|
|
48
|
+
throw new Error("MockUI: Missing data for apiKey");
|
|
49
|
+
}
|
|
50
|
+
return this.data.apiKey;
|
|
51
|
+
}
|
|
52
|
+
async promptConfirm(message, defaultYes) {
|
|
53
|
+
if (typeof this.data.confirm === "boolean") {
|
|
54
|
+
return this.data.confirm;
|
|
55
|
+
}
|
|
56
|
+
return defaultYes ?? false;
|
|
57
|
+
}
|
|
58
|
+
log(message) {
|
|
59
|
+
console.log(message);
|
|
60
|
+
}
|
|
61
|
+
warn(message) {
|
|
62
|
+
console.log(`WARN: ${message}`);
|
|
63
|
+
}
|
|
64
|
+
error(message) {
|
|
65
|
+
console.error(`ERROR: ${message}`);
|
|
66
|
+
}
|
|
67
|
+
success(message) {
|
|
68
|
+
console.log(`SUCCESS: ${message}`);
|
|
69
|
+
}
|
|
70
|
+
}
|
|
71
|
+
|
|
72
|
+
// src/commands/snapshot/handler.ts
|
|
73
|
+
var SCHEMAS = {
|
|
74
|
+
init: {
|
|
75
|
+
description: "Data schema for 'init' command",
|
|
76
|
+
type: "object",
|
|
77
|
+
properties: {
|
|
78
|
+
mcpClient: { type: "string", enum: ["vscode", "cursor", "claude-code", "gemini-cli", "codex", "opencode"] },
|
|
79
|
+
authMode: { type: "string", enum: ["apiKey", "oauth"] },
|
|
80
|
+
transportType: { type: "string", enum: ["http", "stdio"] },
|
|
81
|
+
apiKeyStorage: { type: "string", enum: ["config", "skip", ".env"] },
|
|
82
|
+
apiKey: { type: "string" },
|
|
83
|
+
confirm: { type: "boolean" },
|
|
84
|
+
inputArgs: {
|
|
85
|
+
type: "object",
|
|
86
|
+
description: "Arguments to pass to the command execution",
|
|
87
|
+
properties: {
|
|
88
|
+
local: { type: "boolean" },
|
|
89
|
+
defaults: { type: "boolean" },
|
|
90
|
+
autoVerify: { type: "boolean" },
|
|
91
|
+
client: { type: "string" },
|
|
92
|
+
transport: { type: "string" }
|
|
93
|
+
}
|
|
94
|
+
}
|
|
95
|
+
},
|
|
96
|
+
required: ["mcpClient", "authMode"]
|
|
97
|
+
},
|
|
98
|
+
doctor: {
|
|
99
|
+
description: "Data schema for 'doctor' command",
|
|
100
|
+
type: "object",
|
|
101
|
+
properties: {
|
|
102
|
+
confirm: { type: "boolean" },
|
|
103
|
+
inputArgs: {
|
|
104
|
+
type: "object",
|
|
105
|
+
description: "Arguments to pass to the command execution",
|
|
106
|
+
properties: {
|
|
107
|
+
verbose: { type: "boolean" }
|
|
108
|
+
}
|
|
109
|
+
}
|
|
110
|
+
}
|
|
111
|
+
},
|
|
112
|
+
site: {
|
|
113
|
+
description: "Data schema for 'site' command",
|
|
114
|
+
type: "object",
|
|
115
|
+
properties: {
|
|
116
|
+
screens: {
|
|
117
|
+
type: "array",
|
|
118
|
+
items: {
|
|
119
|
+
type: "object",
|
|
120
|
+
properties: {
|
|
121
|
+
id: { type: "string" },
|
|
122
|
+
title: { type: "string" },
|
|
123
|
+
description: { type: "string" },
|
|
124
|
+
type: { type: "string" },
|
|
125
|
+
content: { type: "string" }
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
},
|
|
129
|
+
inputArgs: {
|
|
130
|
+
type: "object",
|
|
131
|
+
properties: {
|
|
132
|
+
projectId: { type: "string" },
|
|
133
|
+
outputDir: { type: "string" }
|
|
134
|
+
}
|
|
135
|
+
}
|
|
136
|
+
}
|
|
137
|
+
}
|
|
138
|
+
};
|
|
139
|
+
|
|
140
|
+
class SnapshotHandler {
|
|
141
|
+
services;
|
|
142
|
+
constructor(services) {
|
|
143
|
+
this.services = services;
|
|
144
|
+
}
|
|
145
|
+
async execute(input) {
|
|
146
|
+
if (input.schema) {
|
|
147
|
+
if (input.command) {
|
|
148
|
+
const schema = SCHEMAS[input.command];
|
|
149
|
+
if (!schema) {
|
|
150
|
+
return {
|
|
151
|
+
success: false,
|
|
152
|
+
error: { message: `No schema found for command '${input.command}'` }
|
|
153
|
+
};
|
|
154
|
+
}
|
|
155
|
+
console.log(JSON.stringify(schema, null, 2));
|
|
156
|
+
} else {
|
|
157
|
+
console.log(JSON.stringify(Object.keys(SCHEMAS), null, 2));
|
|
158
|
+
}
|
|
159
|
+
return { success: true };
|
|
160
|
+
}
|
|
161
|
+
if (!input.command) {
|
|
162
|
+
return { success: false, error: { message: "Command (-c) is required unless using -s" } };
|
|
163
|
+
}
|
|
164
|
+
if (!input.data) {
|
|
165
|
+
return { success: false, error: { message: "Data file (-d) is required unless using -s" } };
|
|
166
|
+
}
|
|
167
|
+
let data;
|
|
168
|
+
try {
|
|
169
|
+
if (await import_fs_extra.default.pathExists(input.data)) {
|
|
170
|
+
data = await import_fs_extra.default.readJson(input.data);
|
|
171
|
+
} else {
|
|
172
|
+
try {
|
|
173
|
+
data = JSON.parse(input.data);
|
|
174
|
+
} catch {
|
|
175
|
+
return { success: false, error: { message: `Data file not found at '${input.data}' and content is not valid JSON` } };
|
|
176
|
+
}
|
|
177
|
+
}
|
|
178
|
+
} catch (e) {
|
|
179
|
+
return { success: false, error: { message: `Failed to read data: ${e instanceof Error ? e.message : String(e)}` } };
|
|
180
|
+
}
|
|
181
|
+
const mockUI = new MockUI(data);
|
|
182
|
+
try {
|
|
183
|
+
switch (input.command) {
|
|
184
|
+
case "init": {
|
|
185
|
+
const { InitHandler } = await import("./chunk-kkc2tvar.js");
|
|
186
|
+
const handler = new InitHandler(this.services?.gcloud, this.services?.mcpConfig, this.services?.project, this.services?.stitch, mockUI);
|
|
187
|
+
const initInput = {
|
|
188
|
+
local: false,
|
|
189
|
+
defaults: false,
|
|
190
|
+
autoVerify: true,
|
|
191
|
+
...data.inputArgs
|
|
192
|
+
};
|
|
193
|
+
const result = await handler.execute(initInput);
|
|
194
|
+
if (!result.success) {
|
|
195
|
+
console.error("Init failed:", result.error);
|
|
196
|
+
}
|
|
197
|
+
break;
|
|
198
|
+
}
|
|
199
|
+
case "doctor": {
|
|
200
|
+
const { DoctorHandler } = await import("./chunk-9cjny9z2.js");
|
|
201
|
+
const handler = new DoctorHandler(this.services?.gcloud, this.services?.stitch, mockUI);
|
|
202
|
+
const doctorInput = {
|
|
203
|
+
verbose: false,
|
|
204
|
+
...data.inputArgs
|
|
205
|
+
};
|
|
206
|
+
const result = await handler.execute(doctorInput);
|
|
207
|
+
if (!result.success) {
|
|
208
|
+
console.error("Doctor failed:", result.error);
|
|
209
|
+
}
|
|
210
|
+
break;
|
|
211
|
+
}
|
|
212
|
+
case "site": {
|
|
213
|
+
const { SiteBuilder } = await import("./chunk-3ff2k44g.js");
|
|
214
|
+
const { createMockStitch, createMockProject, createMockScreen } = await import("./chunk-98trrxk2.js");
|
|
215
|
+
const mockScreens = (data.screens || []).map((s) => createMockScreen({
|
|
216
|
+
screenId: s.name,
|
|
217
|
+
title: s.title,
|
|
218
|
+
getHtml: () => Promise.resolve(s.htmlCode?.downloadUrl || null)
|
|
219
|
+
}));
|
|
220
|
+
const mockClient = createMockStitch(createMockProject(data.inputArgs?.projectId || "mock-project", mockScreens));
|
|
221
|
+
try {
|
|
222
|
+
const { render } = await import("./chunk-v8ak35zd.js");
|
|
223
|
+
const React = await import("./chunk-edp6faw2.js");
|
|
224
|
+
const projectId = data.inputArgs?.projectId || "mock-project";
|
|
225
|
+
const { lastFrame, unmount } = render(React.createElement(SiteBuilder, {
|
|
226
|
+
projectId,
|
|
227
|
+
client: mockClient,
|
|
228
|
+
onExit: () => {}
|
|
229
|
+
}));
|
|
230
|
+
await new Promise((resolve) => setTimeout(resolve, 1000));
|
|
231
|
+
console.log(lastFrame());
|
|
232
|
+
unmount();
|
|
233
|
+
} catch (e) {
|
|
234
|
+
if (e.code === "ERR_MODULE_NOT_FOUND") {
|
|
235
|
+
console.warn(theme.yellow("ink-testing-library not found. Install dev dependencies to snapshot site command."));
|
|
236
|
+
} else {
|
|
237
|
+
throw e;
|
|
238
|
+
}
|
|
239
|
+
}
|
|
240
|
+
break;
|
|
241
|
+
}
|
|
242
|
+
default:
|
|
243
|
+
return { success: false, error: { message: `Unsupported command '${input.command}'` } };
|
|
244
|
+
}
|
|
245
|
+
} catch (error) {
|
|
246
|
+
console.error(theme.red("Command execution failed:"), error);
|
|
247
|
+
return { success: true };
|
|
248
|
+
}
|
|
249
|
+
return { success: true };
|
|
250
|
+
}
|
|
251
|
+
}
|
|
252
|
+
export {
|
|
253
|
+
SnapshotHandler
|
|
254
|
+
};
|
|
255
|
+
|
|
256
|
+
//# debugId=F78CBF4C733E0A4464756E2164756E21
|
|
@@ -0,0 +1,11 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../src/commands/snapshot/handler.ts", "../src/framework/MockUI.ts"],
|
|
4
|
+
"sourcesContent": [
|
|
5
|
+
"import fs from 'fs-extra';\nimport { type SnapshotCommand, type SnapshotInput, type SnapshotResult } from './spec.js';\nimport { MockUI } from '../../framework/MockUI.js';\nimport { theme } from '../../ui/theme.js';\nimport { type GcloudService } from '../../services/gcloud/spec.js';\nimport { type StitchService } from '../../services/stitch/spec.js';\nimport { type McpConfigService } from '../../services/mcp-config/spec.js';\nimport { type ProjectService } from '../../services/project/spec.js';\n\ninterface SnapshotServices {\n gcloud?: GcloudService;\n stitch?: StitchService;\n mcpConfig?: McpConfigService;\n project?: ProjectService;\n}\n\n// Schemas for the commands to be printed with -s\nconst SCHEMAS: Record<string, any> = {\n init: {\n description: \"Data schema for 'init' command\",\n type: \"object\",\n properties: {\n mcpClient: { type: \"string\", enum: [\"vscode\", \"cursor\", \"claude-code\", \"gemini-cli\", \"codex\", \"opencode\"] },\n authMode: { type: \"string\", enum: [\"apiKey\", \"oauth\"] },\n transportType: { type: \"string\", enum: [\"http\", \"stdio\"] },\n apiKeyStorage: { type: \"string\", enum: [\"config\", \"skip\", \".env\"] },\n apiKey: { type: \"string\" },\n confirm: { type: \"boolean\" },\n inputArgs: {\n type: \"object\",\n description: \"Arguments to pass to the command execution\",\n properties: {\n local: { type: \"boolean\" },\n defaults: { type: \"boolean\" },\n autoVerify: { type: \"boolean\" },\n client: { type: \"string\" },\n transport: { type: \"string\" }\n }\n }\n },\n required: [\"mcpClient\", \"authMode\"]\n },\n doctor: {\n description: \"Data schema for 'doctor' command\",\n type: \"object\",\n properties: {\n confirm: { type: \"boolean\" },\n inputArgs: {\n type: \"object\",\n description: \"Arguments to pass to the command execution\",\n properties: {\n verbose: { type: \"boolean\" }\n }\n }\n }\n },\n site: {\n description: \"Data schema for 'site' command\",\n type: \"object\",\n properties: {\n screens: {\n type: \"array\",\n items: {\n type: \"object\",\n properties: {\n id: { type: \"string\" },\n title: { type: \"string\" },\n description: { type: \"string\" },\n type: { type: \"string\" },\n content: { type: \"string\" }\n }\n }\n },\n inputArgs: {\n type: \"object\",\n properties: {\n projectId: { type: \"string\" },\n outputDir: { type: \"string\" }\n }\n }\n }\n }\n};\n\nexport class SnapshotHandler implements SnapshotCommand {\n constructor(private readonly services?: SnapshotServices) {}\n\n async execute(input: SnapshotInput): Promise<SnapshotResult> {\n if (input.schema) {\n if (input.command) {\n const schema = SCHEMAS[input.command];\n if (!schema) {\n return {\n success: false,\n error: { message: `No schema found for command '${input.command}'` }\n };\n }\n console.log(JSON.stringify(schema, null, 2));\n } else {\n console.log(JSON.stringify(Object.keys(SCHEMAS), null, 2));\n }\n return { success: true };\n }\n\n if (!input.command) {\n return { success: false, error: { message: \"Command (-c) is required unless using -s\" } };\n }\n if (!input.data) {\n return { success: false, error: { message: \"Data file (-d) is required unless using -s\" } };\n }\n\n // Load data\n let data: any;\n try {\n if (await fs.pathExists(input.data)) {\n data = await fs.readJson(input.data);\n } else {\n // Try parsing as JSON string\n try {\n data = JSON.parse(input.data);\n } catch {\n return { success: false, error: { message: `Data file not found at '${input.data}' and content is not valid JSON` } };\n }\n }\n } catch (e) {\n return { success: false, error: { message: `Failed to read data: ${e instanceof Error ? e.message : String(e)}` } };\n }\n\n const mockUI = new MockUI(data);\n\n try {\n switch (input.command) {\n case 'init': {\n const { InitHandler } = await import('../init/handler.js');\n const handler = new InitHandler(\n this.services?.gcloud,\n this.services?.mcpConfig,\n this.services?.project,\n this.services?.stitch,\n mockUI\n );\n const initInput = {\n local: false,\n defaults: false,\n autoVerify: true,\n ...data.inputArgs\n };\n const result = await handler.execute(initInput);\n if (!result.success) {\n console.error('Init failed:', result.error);\n }\n break;\n }\n case 'doctor': {\n const { DoctorHandler } = await import('../doctor/handler.js');\n const handler = new DoctorHandler(\n this.services?.gcloud,\n this.services?.stitch,\n mockUI\n );\n const doctorInput = {\n verbose: false,\n ...data.inputArgs\n };\n const result = await handler.execute(doctorInput);\n if (!result.success) {\n console.error('Doctor failed:', result.error);\n }\n break;\n }\n case 'site': {\n const { SiteBuilder } = await import('../site/ui/SiteBuilder.js');\n const { createMockStitch, createMockProject, createMockScreen } = await import('../../services/stitch-sdk/MockStitchSDK.js');\n\n const mockScreens = (data.screens || []).map((s: any) => createMockScreen({\n screenId: s.name,\n title: s.title,\n getHtml: (() => Promise.resolve(s.htmlCode?.downloadUrl || null)) as any,\n }));\n const mockClient = createMockStitch(createMockProject(data.inputArgs?.projectId || 'mock-project', mockScreens));\n\n // Use ink-testing-library to render and snapshot the UI\n try {\n // We use require to avoid type checking issues if it's not installed in prod build env,\n // although this command is a dev tool.\n const { render } = await import('ink-testing-library');\n const React = await import('react');\n\n const projectId = data.inputArgs?.projectId || 'mock-project';\n\n const { lastFrame, unmount } = render(\n React.createElement(SiteBuilder, {\n projectId,\n client: mockClient,\n onExit: () => {}\n })\n );\n\n // Wait for async operations (loading screens)\n // Simple delay for now, or check frame content until \"Loading\" is gone\n // Since we can't easily wait for a specific state without more complex logic,\n // let's try a short delay to allow useEffect to run.\n await new Promise(resolve => setTimeout(resolve, 1000));\n\n console.log(lastFrame());\n unmount();\n } catch (e) {\n if ((e as any).code === 'ERR_MODULE_NOT_FOUND') {\n console.warn(theme.yellow('ink-testing-library not found. Install dev dependencies to snapshot site command.'));\n } else {\n throw e;\n }\n }\n break;\n }\n default:\n return { success: false, error: { message: `Unsupported command '${input.command}'` } };\n }\n } catch (error) {\n console.error(theme.red('Command execution failed:'), error);\n // Return success: true because the snapshotting process ran, even if the command failed.\n // The error is part of the snapshot.\n return { success: true };\n }\n\n return { success: true };\n }\n}\n",
|
|
6
|
+
"import { type UserInterface } from './UserInterface.js';\nimport type { McpClient } from '../ui/wizard.js';\n\nexport class MockUI implements UserInterface {\n constructor(private readonly data: Record<string, any>) {}\n\n async promptMcpClient(): Promise<McpClient> {\n if (!this.data.mcpClient) {\n throw new Error('MockUI: Missing data for mcpClient');\n }\n return this.data.mcpClient;\n }\n\n async promptAuthMode(): Promise<'apiKey' | 'oauth'> {\n if (!this.data.authMode) {\n throw new Error('MockUI: Missing data for authMode');\n }\n return this.data.authMode;\n }\n\n async promptTransportType(authMode?: 'apiKey' | 'oauth'): Promise<'http' | 'stdio'> {\n if (this.data.transportType) {\n return this.data.transportType;\n }\n throw new Error('MockUI: Missing data for transportType');\n }\n\n async promptApiKeyStorage(): Promise<'config' | 'skip' | '.env'> {\n if (!this.data.apiKeyStorage) {\n throw new Error('MockUI: Missing data for apiKeyStorage');\n }\n return this.data.apiKeyStorage;\n }\n\n async promptApiKey(): Promise<string> {\n if (!this.data.apiKey) {\n throw new Error('MockUI: Missing data for apiKey');\n }\n return this.data.apiKey;\n }\n\n async promptConfirm(message: string, defaultYes?: boolean): Promise<boolean> {\n if (typeof this.data.confirm === 'boolean') {\n return this.data.confirm;\n }\n // If specific confirm keys are provided in data, could use that too.\n // For now, default to defaultYes or false.\n return defaultYes ?? false;\n }\n\n log(message: string): void {\n console.log(message);\n }\n\n warn(message: string): void {\n console.log(`WARN: ${message}`);\n }\n\n error(message: string): void {\n console.error(`ERROR: ${message}`);\n }\n\n success(message: string): void {\n console.log(`SUCCESS: ${message}`);\n }\n}\n"
|
|
7
|
+
],
|
|
8
|
+
"mappings": "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",
|
|
9
|
+
"debugId": "F78CBF4C733E0A4464756E2164756E21",
|
|
10
|
+
"names": []
|
|
11
|
+
}
|
|
@@ -0,0 +1,370 @@
|
|
|
1
|
+
import {
|
|
2
|
+
SiteBuilder,
|
|
3
|
+
SiteManifest
|
|
4
|
+
} from "./chunk-wzkdeb8a.js";
|
|
5
|
+
import {
|
|
6
|
+
require_jsx_dev_runtime
|
|
7
|
+
} from "./chunk-krfqppg2.js";
|
|
8
|
+
import {
|
|
9
|
+
AssetGateway
|
|
10
|
+
} from "./chunk-nv2j020p.js";
|
|
11
|
+
import {
|
|
12
|
+
require_lib
|
|
13
|
+
} from "./chunk-w8q7nsm7.js";
|
|
14
|
+
import {
|
|
15
|
+
render_default
|
|
16
|
+
} from "./chunk-f1kjn6cd.js";
|
|
17
|
+
import"./chunk-4jwmvjb4.js";
|
|
18
|
+
import {
|
|
19
|
+
fetchWithRetry
|
|
20
|
+
} from "./chunk-6gw9apqb.js";
|
|
21
|
+
import {
|
|
22
|
+
stitch
|
|
23
|
+
} from "./chunk-a1mk24q9.js";
|
|
24
|
+
import {
|
|
25
|
+
pLimit
|
|
26
|
+
} from "./chunk-a5xra9jn.js";
|
|
27
|
+
import"./chunk-nq68kghz.js";
|
|
28
|
+
import"./chunk-hc7sdwmn.js";
|
|
29
|
+
import"./chunk-3sfn889r.js";
|
|
30
|
+
import {
|
|
31
|
+
exports_external
|
|
32
|
+
} from "./chunk-c6ge431q.js";
|
|
33
|
+
import {
|
|
34
|
+
__toESM
|
|
35
|
+
} from "./chunk-9wyra8hs.js";
|
|
36
|
+
|
|
37
|
+
// src/lib/services/site/SiteService.ts
|
|
38
|
+
var import_fs_extra = __toESM(require_lib(), 1);
|
|
39
|
+
import path from "path";
|
|
40
|
+
class SiteService {
|
|
41
|
+
static toUIScreens(screens) {
|
|
42
|
+
return screens.filter((s) => s.htmlCode && s.htmlCode.downloadUrl).map((s) => ({
|
|
43
|
+
id: s.name,
|
|
44
|
+
title: s.title,
|
|
45
|
+
downloadUrl: s.htmlCode.downloadUrl,
|
|
46
|
+
status: "ignored",
|
|
47
|
+
route: ""
|
|
48
|
+
}));
|
|
49
|
+
}
|
|
50
|
+
static async generateSite(config, htmlContent, assetGateway, outputDir = ".") {
|
|
51
|
+
await import_fs_extra.default.ensureDir(path.join(outputDir, "src/pages"));
|
|
52
|
+
await import_fs_extra.default.ensureDir(path.join(outputDir, "src/layouts"));
|
|
53
|
+
await import_fs_extra.default.ensureDir(path.join(outputDir, "public/assets"));
|
|
54
|
+
const pkgJson = {
|
|
55
|
+
name: "stitch-site",
|
|
56
|
+
type: "module",
|
|
57
|
+
version: "0.0.1",
|
|
58
|
+
scripts: {
|
|
59
|
+
dev: "astro dev",
|
|
60
|
+
start: "astro dev",
|
|
61
|
+
build: "astro build",
|
|
62
|
+
preview: "astro preview",
|
|
63
|
+
astro: "astro"
|
|
64
|
+
},
|
|
65
|
+
dependencies: {
|
|
66
|
+
astro: "^5.0.0"
|
|
67
|
+
}
|
|
68
|
+
};
|
|
69
|
+
await import_fs_extra.default.writeJson(path.join(outputDir, "package.json"), pkgJson, { spaces: 2 });
|
|
70
|
+
const astroConfig = `import { defineConfig } from 'astro/config';
|
|
71
|
+
export default defineConfig({});`;
|
|
72
|
+
await import_fs_extra.default.writeFile(path.join(outputDir, "astro.config.mjs"), astroConfig);
|
|
73
|
+
const layout = `---
|
|
74
|
+
interface Props {
|
|
75
|
+
title: string;
|
|
76
|
+
}
|
|
77
|
+
|
|
78
|
+
const { title } = Astro.props;
|
|
79
|
+
---
|
|
80
|
+
|
|
81
|
+
<!doctype html>
|
|
82
|
+
<html lang="en">
|
|
83
|
+
<head>
|
|
84
|
+
<meta charset="UTF-8" />
|
|
85
|
+
<meta name="description" content="Astro description" />
|
|
86
|
+
<meta name="viewport" content="width=device-width" />
|
|
87
|
+
<link rel="icon" type="image/svg+xml" href="/favicon.svg" />
|
|
88
|
+
<meta name="generator" content={Astro.generator} />
|
|
89
|
+
<title>{title}</title>
|
|
90
|
+
</head>
|
|
91
|
+
<body>
|
|
92
|
+
<slot />
|
|
93
|
+
</body>
|
|
94
|
+
</html>
|
|
95
|
+
`;
|
|
96
|
+
await import_fs_extra.default.writeFile(path.join(outputDir, "src/layouts/Layout.astro"), layout);
|
|
97
|
+
const limit = pLimit(10);
|
|
98
|
+
const tasks = config.routes.map((route) => limit(async () => {
|
|
99
|
+
if (route.status !== "included")
|
|
100
|
+
return;
|
|
101
|
+
const html = htmlContent.get(route.screenId);
|
|
102
|
+
if (!html) {
|
|
103
|
+
console.warn(`No HTML content found for screen ${route.screenId}`);
|
|
104
|
+
return;
|
|
105
|
+
}
|
|
106
|
+
const { html: rewrittenHtml, assets } = await assetGateway.rewriteHtmlForBuild(html);
|
|
107
|
+
const assetsDir = path.join(outputDir, "public/assets");
|
|
108
|
+
for (const asset of assets) {
|
|
109
|
+
await assetGateway.copyAssetTo(asset.url, path.join(assetsDir, asset.filename));
|
|
110
|
+
}
|
|
111
|
+
let filePath = route.route;
|
|
112
|
+
if (filePath === "/") {
|
|
113
|
+
filePath = "index";
|
|
114
|
+
} else {
|
|
115
|
+
if (filePath.startsWith("/"))
|
|
116
|
+
filePath = filePath.substring(1);
|
|
117
|
+
}
|
|
118
|
+
const fullPath = path.join(outputDir, "src/pages", `${filePath}.astro`);
|
|
119
|
+
await import_fs_extra.default.ensureDir(path.dirname(fullPath));
|
|
120
|
+
await import_fs_extra.default.writeFile(fullPath, rewrittenHtml);
|
|
121
|
+
}));
|
|
122
|
+
await Promise.all(tasks);
|
|
123
|
+
}
|
|
124
|
+
static slugify(text) {
|
|
125
|
+
return text.toString().toLowerCase().replace(/\s+/g, "-").replace(/[^\w\-]+/g, "").replace(/\-\-+/g, "-").replace(/^-+/, "").replace(/-+$/, "");
|
|
126
|
+
}
|
|
127
|
+
}
|
|
128
|
+
|
|
129
|
+
// src/commands/site/utils/suggestRoute.ts
|
|
130
|
+
var HOME_SLUGS = new Set(["home", "landing", "landing-page", "index"]);
|
|
131
|
+
function suggestRoute(title) {
|
|
132
|
+
const slug = title.toLowerCase().replace(/[^a-z0-9\s-]/g, "").trim().replace(/\s+/g, "-").replace(/-+/g, "-");
|
|
133
|
+
return HOME_SLUGS.has(slug) ? "/" : `/${slug}`;
|
|
134
|
+
}
|
|
135
|
+
|
|
136
|
+
// src/commands/site/list-screens/handler.ts
|
|
137
|
+
class ListScreensHandler {
|
|
138
|
+
client;
|
|
139
|
+
constructor(client) {
|
|
140
|
+
this.client = client;
|
|
141
|
+
}
|
|
142
|
+
async execute(input) {
|
|
143
|
+
try {
|
|
144
|
+
const project = this.client.project(input.projectId);
|
|
145
|
+
const sdkScreens = await project.screens();
|
|
146
|
+
const limit = pLimit(3);
|
|
147
|
+
const screens = await Promise.all(sdkScreens.map((s) => limit(async () => {
|
|
148
|
+
const htmlUrl = await s.getHtml().catch(() => null);
|
|
149
|
+
return {
|
|
150
|
+
screenId: s.screenId,
|
|
151
|
+
title: s.title ?? s.screenId,
|
|
152
|
+
suggestedRoute: suggestRoute(s.title ?? s.screenId),
|
|
153
|
+
hasHtml: htmlUrl !== null
|
|
154
|
+
};
|
|
155
|
+
})));
|
|
156
|
+
return { success: true, projectId: input.projectId, screens };
|
|
157
|
+
} catch (e) {
|
|
158
|
+
return {
|
|
159
|
+
success: false,
|
|
160
|
+
error: {
|
|
161
|
+
code: "SCREENS_FETCH_FAILED",
|
|
162
|
+
message: e.message,
|
|
163
|
+
recoverable: false
|
|
164
|
+
}
|
|
165
|
+
};
|
|
166
|
+
}
|
|
167
|
+
}
|
|
168
|
+
}
|
|
169
|
+
|
|
170
|
+
// src/commands/site/list-screens/spec.ts
|
|
171
|
+
var ListScreensInputSchema = exports_external.object({
|
|
172
|
+
projectId: exports_external.string().min(1, "projectId is required")
|
|
173
|
+
});
|
|
174
|
+
var ListScreensErrorCode = exports_external.enum([
|
|
175
|
+
"PROJECT_NOT_FOUND",
|
|
176
|
+
"SCREENS_FETCH_FAILED"
|
|
177
|
+
]);
|
|
178
|
+
|
|
179
|
+
// src/commands/site/generate/handler.ts
|
|
180
|
+
class GenerateHandler {
|
|
181
|
+
client;
|
|
182
|
+
fetchHtml;
|
|
183
|
+
constructor(client, fetchHtml = fetchWithRetry) {
|
|
184
|
+
this.client = client;
|
|
185
|
+
this.fetchHtml = fetchHtml;
|
|
186
|
+
}
|
|
187
|
+
async execute(input) {
|
|
188
|
+
try {
|
|
189
|
+
const project = this.client.project(input.projectId);
|
|
190
|
+
const sdkScreens = await project.screens();
|
|
191
|
+
const screenMap = new Map(sdkScreens.map((s) => [s.screenId, s]));
|
|
192
|
+
const missingIds = input.routesJson.map((r) => r.screenId).filter((id) => !screenMap.has(id));
|
|
193
|
+
if (missingIds.length > 0) {
|
|
194
|
+
return {
|
|
195
|
+
success: false,
|
|
196
|
+
error: {
|
|
197
|
+
code: "SCREEN_NOT_FOUND",
|
|
198
|
+
message: `Screen IDs not found in project: ${missingIds.join(", ")}`,
|
|
199
|
+
hint: `Run stitch site -p ${input.projectId} --list-screens to see available screen IDs.`,
|
|
200
|
+
recoverable: true
|
|
201
|
+
}
|
|
202
|
+
};
|
|
203
|
+
}
|
|
204
|
+
const limit = pLimit(3);
|
|
205
|
+
const htmlContent = new Map;
|
|
206
|
+
const errors = [];
|
|
207
|
+
await Promise.all(input.routesJson.map((r) => limit(async () => {
|
|
208
|
+
const screen = screenMap.get(r.screenId);
|
|
209
|
+
try {
|
|
210
|
+
const htmlUrl = await screen.getHtml();
|
|
211
|
+
const html = htmlUrl ? await this.fetchHtml(htmlUrl) : "";
|
|
212
|
+
htmlContent.set(r.screenId, html);
|
|
213
|
+
} catch (e) {
|
|
214
|
+
errors.push(`${r.screenId}: ${e.message}`);
|
|
215
|
+
}
|
|
216
|
+
})));
|
|
217
|
+
if (errors.length > 0) {
|
|
218
|
+
return {
|
|
219
|
+
success: false,
|
|
220
|
+
error: {
|
|
221
|
+
code: "HTML_FETCH_FAILED",
|
|
222
|
+
message: `Failed to fetch HTML for screens: ${errors.join("; ")}`,
|
|
223
|
+
recoverable: false
|
|
224
|
+
}
|
|
225
|
+
};
|
|
226
|
+
}
|
|
227
|
+
const config = {
|
|
228
|
+
projectId: input.projectId,
|
|
229
|
+
routes: input.routesJson.map((r) => ({ screenId: r.screenId, route: r.route, status: "included" }))
|
|
230
|
+
};
|
|
231
|
+
const assetGateway = new AssetGateway;
|
|
232
|
+
await SiteService.generateSite(config, htmlContent, assetGateway, input.outputDir);
|
|
233
|
+
return {
|
|
234
|
+
success: true,
|
|
235
|
+
outputDir: input.outputDir,
|
|
236
|
+
pages: input.routesJson.map((r) => ({ screenId: r.screenId, route: r.route }))
|
|
237
|
+
};
|
|
238
|
+
} catch (e) {
|
|
239
|
+
return {
|
|
240
|
+
success: false,
|
|
241
|
+
error: {
|
|
242
|
+
code: "GENERATE_FAILED",
|
|
243
|
+
message: e.message,
|
|
244
|
+
recoverable: false
|
|
245
|
+
}
|
|
246
|
+
};
|
|
247
|
+
}
|
|
248
|
+
}
|
|
249
|
+
}
|
|
250
|
+
|
|
251
|
+
// src/commands/site/generate/spec.ts
|
|
252
|
+
var RouteEntrySchema = exports_external.object({
|
|
253
|
+
screenId: exports_external.string().min(1, "screenId is required"),
|
|
254
|
+
route: exports_external.string().startsWith("/", "route must start with /")
|
|
255
|
+
});
|
|
256
|
+
var RoutesJsonSchema = exports_external.string().transform((str, ctx) => {
|
|
257
|
+
try {
|
|
258
|
+
const parsed = JSON.parse(str);
|
|
259
|
+
const result = exports_external.array(RouteEntrySchema).safeParse(parsed);
|
|
260
|
+
if (!result.success) {
|
|
261
|
+
ctx.addIssue({ code: exports_external.ZodIssueCode.custom, message: result.error.issues[0]?.message ?? "Invalid routes array" });
|
|
262
|
+
return exports_external.NEVER;
|
|
263
|
+
}
|
|
264
|
+
return result.data;
|
|
265
|
+
} catch {
|
|
266
|
+
ctx.addIssue({ code: exports_external.ZodIssueCode.custom, message: "routes must be valid JSON" });
|
|
267
|
+
return exports_external.NEVER;
|
|
268
|
+
}
|
|
269
|
+
});
|
|
270
|
+
var GenerateInputSchema = exports_external.object({
|
|
271
|
+
projectId: exports_external.string().min(1, "projectId is required"),
|
|
272
|
+
routesJson: RoutesJsonSchema,
|
|
273
|
+
outputDir: exports_external.string().default(".")
|
|
274
|
+
});
|
|
275
|
+
var GenerateErrorCode = exports_external.enum([
|
|
276
|
+
"INVALID_ROUTES",
|
|
277
|
+
"SCREEN_NOT_FOUND",
|
|
278
|
+
"HTML_FETCH_FAILED",
|
|
279
|
+
"GENERATE_FAILED"
|
|
280
|
+
]);
|
|
281
|
+
|
|
282
|
+
// src/commands/site/index.tsx
|
|
283
|
+
var jsx_dev_runtime = __toESM(require_jsx_dev_runtime(), 1);
|
|
284
|
+
|
|
285
|
+
class SiteCommandHandler {
|
|
286
|
+
client;
|
|
287
|
+
constructor(client = stitch) {
|
|
288
|
+
this.client = client;
|
|
289
|
+
}
|
|
290
|
+
async execute(options) {
|
|
291
|
+
if (options.listScreens) {
|
|
292
|
+
const input = ListScreensInputSchema.safeParse({ projectId: options.projectId });
|
|
293
|
+
if (!input.success) {
|
|
294
|
+
console.log(JSON.stringify({ success: false, error: { code: "INVALID_INPUT", message: input.error.issues[0]?.message } }, null, 2));
|
|
295
|
+
return;
|
|
296
|
+
}
|
|
297
|
+
const result = await new ListScreensHandler(this.client).execute(input.data);
|
|
298
|
+
console.log(JSON.stringify(result, null, 2));
|
|
299
|
+
return;
|
|
300
|
+
}
|
|
301
|
+
if (options.routes !== undefined) {
|
|
302
|
+
const input = GenerateInputSchema.safeParse({
|
|
303
|
+
projectId: options.projectId,
|
|
304
|
+
routesJson: options.routes,
|
|
305
|
+
outputDir: options.outputDir
|
|
306
|
+
});
|
|
307
|
+
if (!input.success) {
|
|
308
|
+
console.log(JSON.stringify({ success: false, error: { code: "INVALID_INPUT", message: input.error.issues[0]?.message } }, null, 2));
|
|
309
|
+
return;
|
|
310
|
+
}
|
|
311
|
+
const result = await new GenerateHandler(this.client).execute(input.data);
|
|
312
|
+
console.log(JSON.stringify(result, null, 2));
|
|
313
|
+
return;
|
|
314
|
+
}
|
|
315
|
+
if (options.export) {
|
|
316
|
+
const project = this.client.project(options.projectId);
|
|
317
|
+
const sdkScreens = await project.screens();
|
|
318
|
+
const uiScreens = await Promise.all(sdkScreens.map(async (s) => ({
|
|
319
|
+
id: s.screenId,
|
|
320
|
+
title: s.title ?? s.screenId,
|
|
321
|
+
status: "ignored",
|
|
322
|
+
route: "",
|
|
323
|
+
downloadUrl: await s.getHtml().catch(() => null)
|
|
324
|
+
})));
|
|
325
|
+
const validScreens = uiScreens.filter((s) => !!s.downloadUrl);
|
|
326
|
+
const siteManifest = new SiteManifest(options.projectId);
|
|
327
|
+
const saved = await siteManifest.load();
|
|
328
|
+
for (const screen of validScreens) {
|
|
329
|
+
const state = saved.get(screen.id);
|
|
330
|
+
if (state?.status)
|
|
331
|
+
screen.status = state.status;
|
|
332
|
+
if (state?.route)
|
|
333
|
+
screen.route = state.route;
|
|
334
|
+
}
|
|
335
|
+
const included = validScreens.filter((s) => s.status === "included");
|
|
336
|
+
const exportData = {
|
|
337
|
+
projectId: options.projectId,
|
|
338
|
+
routes: included.map((s) => ({
|
|
339
|
+
screenId: s.id,
|
|
340
|
+
route: s.route
|
|
341
|
+
}))
|
|
342
|
+
};
|
|
343
|
+
console.log(JSON.stringify(exportData, null, 2));
|
|
344
|
+
return;
|
|
345
|
+
}
|
|
346
|
+
let resultConfig = null;
|
|
347
|
+
let resultHtml;
|
|
348
|
+
const { waitUntilExit } = render_default(/* @__PURE__ */ jsx_dev_runtime.jsxDEV(SiteBuilder, {
|
|
349
|
+
projectId: options.projectId,
|
|
350
|
+
client: this.client,
|
|
351
|
+
onExit: (config, html) => {
|
|
352
|
+
resultConfig = config;
|
|
353
|
+
resultHtml = html;
|
|
354
|
+
}
|
|
355
|
+
}, undefined, false, undefined, this));
|
|
356
|
+
await waitUntilExit();
|
|
357
|
+
if (resultConfig && resultHtml) {
|
|
358
|
+
console.log("Generating site...");
|
|
359
|
+
const assetGateway = new AssetGateway;
|
|
360
|
+
const outputDir = options.outputDir || ".";
|
|
361
|
+
await SiteService.generateSite(resultConfig, resultHtml, assetGateway, outputDir);
|
|
362
|
+
console.log("Site generated successfully!");
|
|
363
|
+
} else {}
|
|
364
|
+
}
|
|
365
|
+
}
|
|
366
|
+
export {
|
|
367
|
+
SiteCommandHandler
|
|
368
|
+
};
|
|
369
|
+
|
|
370
|
+
//# debugId=38734F280B97272E64756E2164756E21
|