happo 6.5.2 → 6.6.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/dist/cli/cancelJob-DETLA3XZ.js +10 -0
- package/dist/cli/chunk-2LNQIG6R.js +84 -0
- package/dist/cli/chunk-2LNQIG6R.js.map +7 -0
- package/dist/cli/{chunk-JZDVA76O.js → chunk-JMN6VM22.js} +3 -3
- package/dist/cli/{chunk-SB3TDZLE.js → chunk-MCYQSPED.js} +2 -2
- package/dist/cli/{chunk-I357LIPJ.js → chunk-MGNFR3W2.js} +2 -2
- package/dist/cli/{chunk-DSAGPJIH.js → chunk-OJHKEE3W.js} +4 -3
- package/dist/cli/chunk-OJHKEE3W.js.map +7 -0
- package/dist/cli/{chunk-IOLNNTKP.js → chunk-SE7XKHF6.js} +2 -2
- package/dist/cli/createAsyncComparison-B3USWOAI.js +10 -0
- package/dist/cli/{createAsyncReport-4M7HVIS3.js → createAsyncReport-3FKLGSYD.js} +4 -4
- package/dist/cli/{getFlakes-QCVM7BHM.js → getFlakes-5DA34KWE.js} +4 -4
- package/dist/cli/index.d.ts.map +1 -1
- package/dist/cli/main.js +11 -11
- package/dist/cli/main.js.map +2 -2
- package/dist/cli/package-CY7G2FGG.js +7 -0
- package/dist/cli/{prepareSnapRequests-NQGLK5M6.js → prepareSnapRequests-RSZU5TQG.js} +158 -73
- package/dist/cli/prepareSnapRequests-RSZU5TQG.js.map +7 -0
- package/dist/cli/startJob-OPTUE4PO.js +10 -0
- package/dist/cli/{wrapper-AGHA5SGK.js → wrapper-QAPBXQCV.js} +7 -9
- package/dist/cli/wrapper-QAPBXQCV.js.map +7 -0
- package/dist/config/RemoteBrowserTarget.d.ts.map +1 -1
- package/dist/cypress/task.js +191 -79
- package/dist/cypress/task.js.map +4 -4
- package/dist/e2e/controller.d.ts.map +1 -1
- package/dist/e2e/wrapper.d.ts.map +1 -1
- package/dist/network/getSignedToken.d.ts +3 -0
- package/dist/network/getSignedToken.d.ts.map +1 -0
- package/dist/network/makeHappoAPIRequest.d.ts +0 -1
- package/dist/network/makeHappoAPIRequest.d.ts.map +1 -1
- package/dist/network/uploadAssets.d.ts.map +1 -1
- package/dist/playwright/index.js +191 -79
- package/dist/playwright/index.js.map +4 -4
- package/package.json +3 -2
- package/dist/cli/cancelJob-RTDZ3IL3.js +0 -10
- package/dist/cli/chunk-DSAGPJIH.js.map +0 -7
- package/dist/cli/chunk-JEFG3R6O.js +0 -54
- package/dist/cli/chunk-JEFG3R6O.js.map +0 -7
- package/dist/cli/createAsyncComparison-3ZFRRUDB.js +0 -10
- package/dist/cli/package-OPMNMDIG.js +0 -7
- package/dist/cli/prepareSnapRequests-NQGLK5M6.js.map +0 -7
- package/dist/cli/startJob-GO6BQDB4.js +0 -10
- package/dist/cli/wrapper-AGHA5SGK.js.map +0 -7
- /package/dist/cli/{cancelJob-RTDZ3IL3.js.map → cancelJob-DETLA3XZ.js.map} +0 -0
- /package/dist/cli/{chunk-JZDVA76O.js.map → chunk-JMN6VM22.js.map} +0 -0
- /package/dist/cli/{chunk-SB3TDZLE.js.map → chunk-MCYQSPED.js.map} +0 -0
- /package/dist/cli/{chunk-I357LIPJ.js.map → chunk-MGNFR3W2.js.map} +0 -0
- /package/dist/cli/{chunk-IOLNNTKP.js.map → chunk-SE7XKHF6.js.map} +0 -0
- /package/dist/cli/{createAsyncComparison-3ZFRRUDB.js.map → createAsyncComparison-B3USWOAI.js.map} +0 -0
- /package/dist/cli/{createAsyncReport-4M7HVIS3.js.map → createAsyncReport-3FKLGSYD.js.map} +0 -0
- /package/dist/cli/{getFlakes-QCVM7BHM.js.map → getFlakes-5DA34KWE.js.map} +0 -0
- /package/dist/cli/{package-OPMNMDIG.js.map → package-CY7G2FGG.js.map} +0 -0
- /package/dist/cli/{startJob-GO6BQDB4.js.map → startJob-OPTUE4PO.js.map} +0 -0
|
@@ -1,8 +1,10 @@
|
|
|
1
1
|
import {
|
|
2
2
|
makeHappoAPIRequest
|
|
3
|
-
} from "./chunk-
|
|
4
|
-
import
|
|
5
|
-
|
|
3
|
+
} from "./chunk-2LNQIG6R.js";
|
|
4
|
+
import {
|
|
5
|
+
ErrorWithStatusCode
|
|
6
|
+
} from "./chunk-SE7XKHF6.js";
|
|
7
|
+
import "./chunk-OJHKEE3W.js";
|
|
6
8
|
|
|
7
9
|
// src/network/prepareSnapRequests.ts
|
|
8
10
|
import fs6 from "node:fs";
|
|
@@ -16,6 +18,7 @@ function createHash(data) {
|
|
|
16
18
|
|
|
17
19
|
// src/config/RemoteBrowserTarget.ts
|
|
18
20
|
var VIEWPORT_PATTERN = /^([0-9]+)x([0-9]+)$/;
|
|
21
|
+
var MAX_BULK_ITEMS_PER_REQUEST = 50;
|
|
19
22
|
function computeDefaultChunks(estimatedSnapCount) {
|
|
20
23
|
if (!Number.isFinite(estimatedSnapCount) || estimatedSnapCount <= 0) {
|
|
21
24
|
return 1;
|
|
@@ -36,6 +39,71 @@ function getPageSlices(pages, chunks) {
|
|
|
36
39
|
}
|
|
37
40
|
return result;
|
|
38
41
|
}
|
|
42
|
+
function buildChunkItem({
|
|
43
|
+
slice,
|
|
44
|
+
chunk,
|
|
45
|
+
pageSlice,
|
|
46
|
+
browserName,
|
|
47
|
+
viewport,
|
|
48
|
+
maxHeight,
|
|
49
|
+
otherOptions,
|
|
50
|
+
globalCSS,
|
|
51
|
+
staticPackage,
|
|
52
|
+
assetsPackage,
|
|
53
|
+
targetName
|
|
54
|
+
}) {
|
|
55
|
+
const payloadString = JSON.stringify({
|
|
56
|
+
viewport,
|
|
57
|
+
maxHeight,
|
|
58
|
+
...otherOptions,
|
|
59
|
+
globalCSS,
|
|
60
|
+
snapPayloads: slice,
|
|
61
|
+
chunk,
|
|
62
|
+
staticPackage,
|
|
63
|
+
assetsPackage,
|
|
64
|
+
pages: pageSlice,
|
|
65
|
+
extendsSha: pageSlice ? pageSlice.extendsSha : void 0
|
|
66
|
+
});
|
|
67
|
+
const payloadHash = createHash(payloadString + (pageSlice ? Math.random() : ""));
|
|
68
|
+
const type = pageSlice && pageSlice.extendsSha ? "extends-report" : `browser-${browserName}`;
|
|
69
|
+
const item = { type, targetName, payloadString, payloadHash };
|
|
70
|
+
if (pageSlice?.extendsSha) {
|
|
71
|
+
item.extendsSha = pageSlice.extendsSha;
|
|
72
|
+
}
|
|
73
|
+
return item;
|
|
74
|
+
}
|
|
75
|
+
async function sendIndividualSnapRequest(item, config) {
|
|
76
|
+
const formData = {
|
|
77
|
+
type: item.type,
|
|
78
|
+
targetName: item.targetName,
|
|
79
|
+
payloadHash: item.payloadHash,
|
|
80
|
+
payload: new File([item.payloadString], "payload.json", {
|
|
81
|
+
type: "application/json"
|
|
82
|
+
})
|
|
83
|
+
};
|
|
84
|
+
if (item.extendsSha) {
|
|
85
|
+
formData.extendsSha = item.extendsSha;
|
|
86
|
+
}
|
|
87
|
+
const requestResult = await makeHappoAPIRequest(
|
|
88
|
+
{
|
|
89
|
+
path: `/api/snap-requests?payloadHash=${item.payloadHash}`,
|
|
90
|
+
method: "POST",
|
|
91
|
+
formData
|
|
92
|
+
},
|
|
93
|
+
config,
|
|
94
|
+
{ retryCount: 5 }
|
|
95
|
+
);
|
|
96
|
+
if (!requestResult) {
|
|
97
|
+
throw new Error("No requestResult");
|
|
98
|
+
}
|
|
99
|
+
if (!("requestId" in requestResult)) {
|
|
100
|
+
throw new Error("No requestId in requestResult");
|
|
101
|
+
}
|
|
102
|
+
if (typeof requestResult.requestId !== "number") {
|
|
103
|
+
throw new TypeError("requestId is not a number");
|
|
104
|
+
}
|
|
105
|
+
return requestResult.requestId;
|
|
106
|
+
}
|
|
39
107
|
var RemoteBrowserTarget = class {
|
|
40
108
|
chunks;
|
|
41
109
|
browserName;
|
|
@@ -74,73 +142,30 @@ var RemoteBrowserTarget = class {
|
|
|
74
142
|
targetName,
|
|
75
143
|
estimatedSnapsCount
|
|
76
144
|
}, config) {
|
|
77
|
-
const
|
|
78
|
-
|
|
79
|
-
|
|
80
|
-
|
|
81
|
-
|
|
82
|
-
|
|
83
|
-
|
|
84
|
-
|
|
85
|
-
|
|
86
|
-
globalCSS,
|
|
87
|
-
snapPayloads: slice,
|
|
88
|
-
chunk,
|
|
89
|
-
staticPackage,
|
|
90
|
-
assetsPackage,
|
|
91
|
-
pages: pageSlice,
|
|
92
|
-
extendsSha: pageSlice ? pageSlice.extendsSha : void 0
|
|
93
|
-
});
|
|
94
|
-
const payloadHash = createHash(
|
|
95
|
-
payloadString + (pageSlice ? Math.random() : "")
|
|
96
|
-
);
|
|
97
|
-
const formData = {
|
|
98
|
-
type: pageSlice && pageSlice.extendsSha ? "extends-report" : `browser-${this.browserName}`,
|
|
99
|
-
targetName,
|
|
100
|
-
payloadHash,
|
|
101
|
-
payload: new File([payloadString], "payload.json", {
|
|
102
|
-
type: "application/json"
|
|
103
|
-
})
|
|
104
|
-
};
|
|
105
|
-
if (pageSlice && pageSlice.extendsSha) {
|
|
106
|
-
formData.extendsSha = pageSlice.extendsSha;
|
|
107
|
-
}
|
|
108
|
-
const requestResult = await makeHappoAPIRequest(
|
|
109
|
-
{
|
|
110
|
-
path: `/api/snap-requests?payloadHash=${payloadHash}`,
|
|
111
|
-
method: "POST",
|
|
112
|
-
json: true,
|
|
113
|
-
formData
|
|
114
|
-
},
|
|
115
|
-
config,
|
|
116
|
-
{ retryCount: 5 }
|
|
117
|
-
);
|
|
118
|
-
if (!requestResult) {
|
|
119
|
-
throw new Error("No requestResult");
|
|
120
|
-
}
|
|
121
|
-
if (!("requestId" in requestResult)) {
|
|
122
|
-
throw new Error("No requestId in requestResult");
|
|
123
|
-
}
|
|
124
|
-
if (typeof requestResult.requestId !== "number") {
|
|
125
|
-
throw new TypeError("requestId is not a number");
|
|
126
|
-
}
|
|
127
|
-
return requestResult.requestId;
|
|
145
|
+
const buildItemParams = {
|
|
146
|
+
browserName: this.browserName,
|
|
147
|
+
viewport: this.viewport,
|
|
148
|
+
maxHeight: this.maxHeight,
|
|
149
|
+
otherOptions: this.otherOptions,
|
|
150
|
+
globalCSS,
|
|
151
|
+
staticPackage,
|
|
152
|
+
assetsPackage,
|
|
153
|
+
targetName
|
|
128
154
|
};
|
|
129
|
-
const
|
|
155
|
+
const items = [];
|
|
130
156
|
if (staticPackage) {
|
|
131
157
|
const effectiveChunks = this.chunks ?? Math.max(1, computeDefaultChunks(estimatedSnapsCount ?? 0));
|
|
132
158
|
for (let i = 0; i < effectiveChunks; i += 1) {
|
|
133
|
-
|
|
134
|
-
|
|
135
|
-
|
|
136
|
-
|
|
159
|
+
items.push(
|
|
160
|
+
buildChunkItem({
|
|
161
|
+
...buildItemParams,
|
|
162
|
+
chunk: effectiveChunks > 1 ? { index: i, total: effectiveChunks } : void 0
|
|
163
|
+
})
|
|
164
|
+
);
|
|
137
165
|
}
|
|
138
166
|
} else if (pages) {
|
|
139
167
|
for (const pageSlice of getPageSlices(pages, this.chunks ?? 1)) {
|
|
140
|
-
|
|
141
|
-
pageSlice
|
|
142
|
-
});
|
|
143
|
-
requestIds.push(requestId);
|
|
168
|
+
items.push(buildChunkItem({ ...buildItemParams, pageSlice }));
|
|
144
169
|
}
|
|
145
170
|
} else {
|
|
146
171
|
const effectiveChunks = this.chunks ?? 1;
|
|
@@ -150,12 +175,63 @@ var RemoteBrowserTarget = class {
|
|
|
150
175
|
i * snapsPerChunk,
|
|
151
176
|
i * snapsPerChunk + snapsPerChunk
|
|
152
177
|
);
|
|
153
|
-
|
|
154
|
-
slice
|
|
155
|
-
});
|
|
156
|
-
requestIds.push(requestId);
|
|
178
|
+
items.push(buildChunkItem({ ...buildItemParams, slice }));
|
|
157
179
|
}
|
|
158
180
|
}
|
|
181
|
+
if (items.length === 0) {
|
|
182
|
+
return [];
|
|
183
|
+
}
|
|
184
|
+
try {
|
|
185
|
+
const requestIds2 = Array.from({
|
|
186
|
+
length: items.length
|
|
187
|
+
});
|
|
188
|
+
for (let batchStart = 0; batchStart < items.length; batchStart += MAX_BULK_ITEMS_PER_REQUEST) {
|
|
189
|
+
const batch = items.slice(
|
|
190
|
+
batchStart,
|
|
191
|
+
batchStart + MAX_BULK_ITEMS_PER_REQUEST
|
|
192
|
+
);
|
|
193
|
+
const result = await makeHappoAPIRequest(
|
|
194
|
+
{
|
|
195
|
+
path: "/api/snap-requests/bulk",
|
|
196
|
+
method: "POST",
|
|
197
|
+
body: { items: batch }
|
|
198
|
+
},
|
|
199
|
+
config,
|
|
200
|
+
{ retryCount: 5 }
|
|
201
|
+
);
|
|
202
|
+
if (result && "results" in result && Array.isArray(result.results) && result.results.length === batch.length) {
|
|
203
|
+
const bulkResults = result.results;
|
|
204
|
+
for (const [i, r] of bulkResults.entries()) {
|
|
205
|
+
requestIds2[batchStart + i] = typeof r.requestId === "number" ? r.requestId : void 0;
|
|
206
|
+
}
|
|
207
|
+
} else {
|
|
208
|
+
throw new Error(
|
|
209
|
+
"Bulk snap-requests endpoint returned an unexpected payload shape; aborting to avoid duplicate snap-requests."
|
|
210
|
+
);
|
|
211
|
+
}
|
|
212
|
+
}
|
|
213
|
+
for (const [i, item] of items.entries()) {
|
|
214
|
+
if (requestIds2[i] === void 0) {
|
|
215
|
+
requestIds2[i] = await sendIndividualSnapRequest(item, config);
|
|
216
|
+
}
|
|
217
|
+
}
|
|
218
|
+
return requestIds2.map((id, index) => {
|
|
219
|
+
if (id === void 0) {
|
|
220
|
+
throw new Error(
|
|
221
|
+
`Failed to obtain snap request ID for item at index ${index}`
|
|
222
|
+
);
|
|
223
|
+
}
|
|
224
|
+
return id;
|
|
225
|
+
});
|
|
226
|
+
} catch (error) {
|
|
227
|
+
if (!(error instanceof ErrorWithStatusCode && (error.statusCode === 404 || error.statusCode === 501))) {
|
|
228
|
+
throw error;
|
|
229
|
+
}
|
|
230
|
+
}
|
|
231
|
+
const requestIds = [];
|
|
232
|
+
for (const item of items) {
|
|
233
|
+
requestIds.push(await sendIndividualSnapRequest(item, config));
|
|
234
|
+
}
|
|
159
235
|
return requestIds;
|
|
160
236
|
}
|
|
161
237
|
};
|
|
@@ -589,6 +665,7 @@ var Logger = class {
|
|
|
589
665
|
};
|
|
590
666
|
|
|
591
667
|
// src/network/uploadAssets.ts
|
|
668
|
+
import { createHash as createHash2 } from "node:crypto";
|
|
592
669
|
import retry from "async-retry";
|
|
593
670
|
async function uploadAssets(buffer, options, config) {
|
|
594
671
|
const { project } = config;
|
|
@@ -596,8 +673,7 @@ async function uploadAssets(buffer, options, config) {
|
|
|
596
673
|
const signedUrlRes = await makeHappoAPIRequest(
|
|
597
674
|
{
|
|
598
675
|
path: `/api/snap-requests/assets/${hash}/signed-url`,
|
|
599
|
-
method: "GET"
|
|
600
|
-
json: true
|
|
676
|
+
method: "GET"
|
|
601
677
|
},
|
|
602
678
|
config,
|
|
603
679
|
{ retryCount: 3 }
|
|
@@ -623,7 +699,8 @@ async function uploadAssets(buffer, options, config) {
|
|
|
623
699
|
body: buffer,
|
|
624
700
|
headers: {
|
|
625
701
|
"Content-Type": "application/zip"
|
|
626
|
-
}
|
|
702
|
+
},
|
|
703
|
+
signal: AbortSignal.timeout(6e4)
|
|
627
704
|
});
|
|
628
705
|
if (!res.ok) {
|
|
629
706
|
const error = new Error(
|
|
@@ -635,6 +712,15 @@ async function uploadAssets(buffer, options, config) {
|
|
|
635
712
|
}
|
|
636
713
|
throw error;
|
|
637
714
|
}
|
|
715
|
+
const etag = res.headers.get("etag");
|
|
716
|
+
const expectedEtag = createHash2("md5").update(buffer).digest("hex");
|
|
717
|
+
if (!etag || !etag.includes(expectedEtag)) {
|
|
718
|
+
const error = new Error(
|
|
719
|
+
`S3 upload verification failed: expected ETag to include ${expectedEtag}, got ${etag ?? "(none)"}. A firewall may be intercepting the upload.`
|
|
720
|
+
);
|
|
721
|
+
bail(error);
|
|
722
|
+
return;
|
|
723
|
+
}
|
|
638
724
|
return res;
|
|
639
725
|
},
|
|
640
726
|
{
|
|
@@ -649,8 +735,7 @@ async function uploadAssets(buffer, options, config) {
|
|
|
649
735
|
const finalizeRes = await makeHappoAPIRequest(
|
|
650
736
|
{
|
|
651
737
|
path: `/api/snap-requests/assets/${hash}/signed-url/finalize`,
|
|
652
|
-
method: "POST"
|
|
653
|
-
json: true
|
|
738
|
+
method: "POST"
|
|
654
739
|
},
|
|
655
740
|
config,
|
|
656
741
|
{ retryCount: 3 }
|
|
@@ -783,4 +868,4 @@ async function prepareSnapRequests(config) {
|
|
|
783
868
|
export {
|
|
784
869
|
prepareSnapRequests as default
|
|
785
870
|
};
|
|
786
|
-
//# sourceMappingURL=prepareSnapRequests-
|
|
871
|
+
//# sourceMappingURL=prepareSnapRequests-RSZU5TQG.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../src/network/prepareSnapRequests.ts", "../../src/utils/createHash.ts", "../../src/config/RemoteBrowserTarget.ts", "../../src/storybook/index.ts", "../../src/storybook/getStorybookBuildCommandParts.ts", "../../src/storybook/getStorybookStoryCount.ts", "../../src/storybook/getStorybookVersionFromPackageJson.ts", "../../src/utils/deterministicArchive.ts", "../../src/utils/validateArchive.ts", "../../src/utils/Logger.ts", "../../src/network/uploadAssets.ts"],
|
|
4
|
+
"sourcesContent": ["import fs from 'node:fs';\nimport path from 'node:path';\n\nimport type { ConfigWithDefaults } from '../config/index.ts';\nimport RemoteBrowserTarget, {\n type ExecuteParams,\n} from '../config/RemoteBrowserTarget.ts';\nimport buildStorybookPackage from '../storybook/index.ts';\nimport deterministicArchive from '../utils/deterministicArchive.ts';\nimport Logger, { logTag } from '../utils/Logger.ts';\nimport uploadAssets from './uploadAssets.ts';\n\nasync function fileExists(path: string): Promise<boolean> {\n try {\n await fs.promises.stat(path);\n return true;\n } catch (error) {\n if (error instanceof Error && 'code' in error && error.code === 'ENOENT') {\n return false;\n }\n\n throw error;\n }\n}\n\nasync function createIframeHTML(\n rootDir: string,\n entryPoint: string,\n logger: Logger,\n): Promise<void> {\n const iframePath = path.join(rootDir, 'iframe.html');\n\n if (await fileExists(iframePath)) {\n logger.info(`Using existing iframe.html at '${iframePath}'`);\n return;\n }\n\n const iframeContent = `<!DOCTYPE html>\n<html lang=\"en\" dir=\"ltr\">\n <head>\n <title>Happo</title>\n <meta charset=\"utf-8\">\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1\">\n </head>\n <body>\n <script src=\"${entryPoint}\"></script>\n </body>\n</html>`;\n\n await fs.promises.mkdir(rootDir, { recursive: true });\n await fs.promises.writeFile(iframePath, iframeContent);\n}\n\ninterface BuildPackageResult {\n packageDir: string;\n estimatedSnapsCount?: number;\n}\n\nasync function buildPackage(\n { integration }: ConfigWithDefaults,\n logger: Logger,\n): Promise<BuildPackageResult> {\n if (integration.type === 'custom') {\n const { rootDir, entryPoint, estimatedSnapsCount } = await integration.build();\n await createIframeHTML(rootDir, entryPoint, logger);\n\n const result: BuildPackageResult = { packageDir: rootDir };\n if (estimatedSnapsCount != null) {\n result.estimatedSnapsCount = estimatedSnapsCount;\n }\n return result;\n }\n\n if (integration.type === 'storybook') {\n return await buildStorybookPackage(integration);\n }\n\n throw new Error(`Unsupported integration type: ${integration.type}`);\n}\n\nasync function validatePackage(packageDir: string): Promise<void> {\n const iframePath = path.join(packageDir, 'iframe.html');\n\n if (!(await fileExists(iframePath))) {\n throw new Error(\n `Could not find iframe.html in static package at '${iframePath}'`,\n );\n }\n}\n\ninterface PreparePackageResult {\n packagePath: string;\n estimatedSnapsCount?: number;\n}\n\nasync function preparePackage(\n config: ConfigWithDefaults,\n logger: Logger,\n): Promise<PreparePackageResult> {\n const { packageDir, estimatedSnapsCount } = await buildPackage(config, logger);\n\n await validatePackage(packageDir);\n\n const { buffer, hash } = await deterministicArchive([packageDir]);\n const packagePath = await uploadAssets(\n buffer,\n {\n hash,\n logger,\n },\n config,\n );\n\n const result: PreparePackageResult = { packagePath };\n if (estimatedSnapsCount != null) {\n result.estimatedSnapsCount = estimatedSnapsCount;\n }\n return result;\n}\n\nexport default async function prepareSnapRequests(\n config: ConfigWithDefaults,\n): Promise<Array<number>> {\n const logger = new Logger();\n const prepareResult =\n config.integration.type === 'pages'\n ? null\n : await preparePackage(config, logger);\n\n const targetNames = Object.keys(config.targets);\n const tl = targetNames.length;\n logger.info(\n `${logTag(config.project)}Generating screenshots in ${tl} target${\n tl > 1 ? 's' : ''\n }...`,\n );\n const outerStartTime = Date.now();\n const results: Array<number> = [];\n await Promise.all(\n targetNames.map(async (name) => {\n const startTime = Date.now();\n\n if (!config.targets[name]) {\n throw new Error(`Target ${name} not found in config`);\n }\n\n const target = new RemoteBrowserTarget(\n config.targets[name].type,\n config.targets[name],\n );\n\n const targetParams: ExecuteParams = {\n targetName: name,\n };\n\n if (prepareResult) {\n targetParams.staticPackage = prepareResult.packagePath;\n\n if (prepareResult.estimatedSnapsCount != null) {\n targetParams.estimatedSnapsCount = prepareResult.estimatedSnapsCount;\n }\n }\n\n if (config.integration.type === 'pages') {\n targetParams.pages = config.integration.pages;\n }\n\n const snapRequestIds = await target.execute(targetParams, config);\n logger.start(` - ${logTag(config.project)}${name}`, { startTime });\n logger.success();\n results.push(...snapRequestIds);\n }),\n );\n logger.start(undefined, { startTime: outerStartTime });\n logger.success();\n return results;\n}\n", "import crypto from 'node:crypto';\n\n/**\n * Creates an MD5 hash of the input data\n * @param data - The data to hash (string, Buffer, or TypedArray)\n * @returns The MD5 hash as a hexadecimal string\n */\nexport default function createHash(\n data: string | Buffer | NodeJS.TypedArray,\n): string {\n return crypto.createHash('md5').update(data).digest('hex');\n}\n", "import { ErrorWithStatusCode } from '../network/fetchWithRetry.ts';\nimport makeHappoAPIRequest from '../network/makeHappoAPIRequest.ts';\nimport createHash from '../utils/createHash.ts';\nimport type {\n BrowserType,\n ConfigWithDefaults,\n Page,\n TargetWithDefaults,\n} from './index.ts';\n\nconst VIEWPORT_PATTERN = /^([0-9]+)x([0-9]+)$/;\n\n/**\n * Maximum number of chunk items sent in a single bulk request.\n * Keeps individual payloads bounded while still protecting against\n * arbitrarily large explicit `chunks` values exceeding server limits.\n */\nconst MAX_BULK_ITEMS_PER_REQUEST = 50;\n\n/**\n * Compute the number of chunks to use based on an estimated snapshot count.\n *\n * Aims for roughly 100 items per chunk, capped at 20. Returns 1 for\n * non-positive or non-finite inputs.\n */\nfunction computeDefaultChunks(estimatedSnapCount: number): number {\n if (!Number.isFinite(estimatedSnapCount) || estimatedSnapCount <= 0) {\n return 1;\n }\n\n return Math.min(20, Math.ceil(estimatedSnapCount / 100));\n}\n\n/**\n * PageSlice is an array of pages with the extra extendsSha property.\n */\ninterface PageSlice extends Array<Page> {\n extendsSha?: string;\n}\n\ninterface Chunk {\n index: number;\n total: number;\n}\n\ninterface ChunkItem {\n type: string;\n targetName: string | undefined;\n payloadString: string;\n payloadHash: string;\n extendsSha?: string;\n}\n\nexport interface CSSBlock {\n id: string;\n conditional: boolean;\n css: string;\n}\n\nexport interface ExecuteParams {\n globalCSS?: string | Array<CSSBlock>;\n\n /** Path to the assets package */\n assetsPackage?: string;\n\n /** Path to the static package */\n staticPackage?: string;\n\n snapPayloads?: Array<unknown>;\n pages?: Array<Page>;\n targetName?: string;\n\n /**\n * Total number of snapshots in the package. When provided for staticPackage\n * requests without explicit chunks, used to automatically determine the\n * optimal number of parallel chunks.\n */\n estimatedSnapsCount?: number;\n}\n\nfunction getPageSlices(pages: Array<Page>, chunks: number): Array<PageSlice> {\n const result: Array<PageSlice> = [];\n\n // First, split the raw pages into chunks\n const pagesPerChunk = Math.ceil(pages.length / chunks);\n for (let i = 0; i < chunks; i += 1) {\n const pageSlice = pages.slice(\n i * pagesPerChunk,\n i * pagesPerChunk + pagesPerChunk,\n );\n\n if (pageSlice.length > 0) {\n result.push(pageSlice);\n }\n }\n return result;\n}\n\nfunction buildChunkItem({\n slice,\n chunk,\n pageSlice,\n browserName,\n viewport,\n maxHeight,\n otherOptions,\n globalCSS,\n staticPackage,\n assetsPackage,\n targetName,\n}: {\n slice?: Array<unknown> | undefined;\n chunk?: Chunk | undefined;\n pageSlice?: PageSlice | undefined;\n browserName: BrowserType;\n viewport: string;\n maxHeight: number | undefined;\n otherOptions: Record<string, unknown>;\n globalCSS: string | Array<CSSBlock> | undefined;\n staticPackage: string | undefined;\n assetsPackage: string | undefined;\n targetName: string | undefined;\n}): ChunkItem {\n const payloadString = JSON.stringify({\n viewport,\n maxHeight,\n ...otherOptions,\n globalCSS,\n snapPayloads: slice,\n chunk,\n staticPackage,\n assetsPackage,\n pages: pageSlice,\n extendsSha: pageSlice ? pageSlice.extendsSha : undefined,\n });\n\n const payloadHash = createHash(payloadString + (pageSlice ? Math.random() : ''));\n\n const type =\n pageSlice && pageSlice.extendsSha ? 'extends-report' : `browser-${browserName}`;\n\n const item: ChunkItem = { type, targetName, payloadString, payloadHash };\n if (pageSlice?.extendsSha) {\n item.extendsSha = pageSlice.extendsSha;\n }\n return item;\n}\n\nasync function sendIndividualSnapRequest(\n item: ChunkItem,\n config: ConfigWithDefaults,\n): Promise<number> {\n const formData: Record<string, string | number | File | undefined> = {\n type: item.type,\n targetName: item.targetName,\n payloadHash: item.payloadHash,\n payload: new File([item.payloadString], 'payload.json', {\n type: 'application/json',\n }),\n };\n\n if (item.extendsSha) {\n formData.extendsSha = item.extendsSha;\n }\n\n // We `await` here inside the loop to avoid POSTing all payloads to the\n // server at the same time (thus reducing load a little).\n const requestResult = await makeHappoAPIRequest(\n {\n path: `/api/snap-requests?payloadHash=${item.payloadHash}`,\n method: 'POST',\n formData,\n },\n config,\n { retryCount: 5 },\n );\n\n if (!requestResult) {\n throw new Error('No requestResult');\n }\n\n if (!('requestId' in requestResult)) {\n throw new Error('No requestId in requestResult');\n }\n\n if (typeof requestResult.requestId !== 'number') {\n throw new TypeError('requestId is not a number');\n }\n\n return requestResult.requestId;\n}\n\nexport default class RemoteBrowserTarget {\n public readonly chunks: number | undefined;\n public readonly browserName: BrowserType;\n public readonly viewport: string;\n public readonly maxHeight: number | undefined;\n public readonly otherOptions: Record<string, unknown>;\n\n constructor(\n browserName: BrowserType,\n {\n viewport = '1024x768',\n chunks,\n maxHeight,\n ...otherOptions\n }: TargetWithDefaults,\n ) {\n if (!browserName) {\n throw new Error(\n `Invalid browser type: \"${browserName}\". Make sure the \"type\" field in your target configuration is set to a valid browser type.`,\n );\n }\n\n const viewportMatch = viewport.match(VIEWPORT_PATTERN);\n if (!viewportMatch) {\n throw new Error(\n `Invalid viewport \"${viewport}\". Here's an example of a valid one: \"1024x768\".`,\n );\n }\n\n this.chunks = chunks;\n this.browserName = browserName;\n this.viewport = viewport;\n this.maxHeight = maxHeight ?? undefined;\n this.otherOptions = otherOptions;\n }\n\n async execute(\n {\n globalCSS,\n assetsPackage,\n staticPackage,\n snapPayloads,\n pages,\n targetName,\n estimatedSnapsCount,\n }: ExecuteParams,\n config: ConfigWithDefaults,\n ): Promise<Array<number>> {\n const buildItemParams = {\n browserName: this.browserName,\n viewport: this.viewport,\n maxHeight: this.maxHeight,\n otherOptions: this.otherOptions,\n globalCSS,\n staticPackage,\n assetsPackage,\n targetName,\n };\n\n // Build all chunk items up front\n const items: Array<ChunkItem> = [];\n\n if (staticPackage) {\n const effectiveChunks =\n this.chunks ?? Math.max(1, computeDefaultChunks(estimatedSnapsCount ?? 0));\n for (let i = 0; i < effectiveChunks; i += 1) {\n items.push(\n buildChunkItem({\n ...buildItemParams,\n chunk:\n effectiveChunks > 1 ? { index: i, total: effectiveChunks } : undefined,\n }),\n );\n }\n } else if (pages) {\n for (const pageSlice of getPageSlices(pages, this.chunks ?? 1)) {\n items.push(buildChunkItem({ ...buildItemParams, pageSlice }));\n }\n } else {\n const effectiveChunks = this.chunks ?? 1;\n const snapsPerChunk = Math.ceil((snapPayloads?.length ?? 0) / effectiveChunks);\n for (let i = 0; i < effectiveChunks; i += 1) {\n const slice = snapPayloads?.slice(\n i * snapsPerChunk,\n i * snapsPerChunk + snapsPerChunk,\n );\n items.push(buildChunkItem({ ...buildItemParams, slice }));\n }\n }\n\n if (items.length === 0) {\n return [];\n }\n\n // Try the bulk endpoint first. If it is unavailable, fall back to individual\n // requests. If it responds with an unexpected payload shape, fail fast to\n // avoid creating duplicate snap-requests.\n //\n // Large item arrays are split into batches of MAX_BULK_ITEMS_PER_REQUEST\n // and sent as sequential bulk requests to keep individual payloads bounded.\n try {\n const requestIds: Array<number | undefined> = Array.from({\n length: items.length,\n });\n\n for (\n let batchStart = 0;\n batchStart < items.length;\n batchStart += MAX_BULK_ITEMS_PER_REQUEST\n ) {\n const batch = items.slice(\n batchStart,\n batchStart + MAX_BULK_ITEMS_PER_REQUEST,\n );\n\n const result = await makeHappoAPIRequest(\n {\n path: '/api/snap-requests/bulk',\n method: 'POST',\n body: { items: batch },\n },\n config,\n { retryCount: 5 },\n );\n\n if (\n result &&\n 'results' in result &&\n Array.isArray(result.results) &&\n result.results.length === batch.length\n ) {\n const bulkResults = result.results as Array<{\n requestId?: number;\n error?: string;\n }>;\n\n for (const [i, r] of bulkResults.entries()) {\n requestIds[batchStart + i] =\n typeof r.requestId === 'number' ? r.requestId : undefined;\n }\n } else {\n // The bulk endpoint responded with a 200 but an unexpected payload\n // shape. Fail fast instead of falling back to avoid potentially\n // creating duplicate snap-requests.\n throw new Error(\n 'Bulk snap-requests endpoint returned an unexpected payload shape; aborting to avoid duplicate snap-requests.',\n );\n }\n }\n\n // Retry any failed items individually (sequentially to reduce load)\n for (const [i, item] of items.entries()) {\n if (requestIds[i] === undefined) {\n requestIds[i] = await sendIndividualSnapRequest(item, config);\n }\n }\n\n return requestIds.map((id, index) => {\n if (id === undefined) {\n throw new Error(\n `Failed to obtain snap request ID for item at index ${index}`,\n );\n }\n\n return id;\n });\n } catch (error) {\n // Fall back to individual requests only when the server explicitly\n // reports that the bulk endpoint is missing or not implemented.\n if (\n !(\n error instanceof ErrorWithStatusCode &&\n (error.statusCode === 404 || error.statusCode === 501)\n )\n ) {\n throw error;\n }\n }\n\n // Fallback: sequential individual requests (for older happo deployments)\n const requestIds: Array<number> = [];\n\n for (const item of items) {\n requestIds.push(await sendIndividualSnapRequest(item, config));\n }\n\n return requestIds;\n }\n}\n", "import { spawn } from 'node:child_process';\nimport fs from 'node:fs';\nimport path from 'node:path';\n\nimport type { StorybookIntegration } from '../config/index.ts';\nimport getStorybookBuildCommandParts from './getStorybookBuildCommandParts.ts';\nimport getStorybookStoryCount from './getStorybookStoryCount.ts';\nimport getStorybookVersionFromPackageJson from './getStorybookVersionFromPackageJson.ts';\nimport type { SkipItems } from './isomorphic/types.ts';\n\nconst { HAPPO_DEBUG } = process.env;\n\nfunction assertSkippedIsSkipItems(skipped: unknown): asserts skipped is SkipItems {\n if (!Array.isArray(skipped)) {\n throw new TypeError(`The \\`skip\\` option didn't provide an array`);\n }\n\n if (skipped.some((item) => !item.component || !item.variant)) {\n throw new Error(\n `Each item provided by the \\`skip\\` option needs a \\`component\\` and a \\`variant\\` property`,\n );\n }\n}\n\nfunction resolveBuildCommandParts() {\n const version = getStorybookVersionFromPackageJson();\n\n if (version < 9) {\n throw new Error(\n `Storybook v${version} is not supported. Please update storybook to v9 or later.`,\n );\n }\n\n return getStorybookBuildCommandParts();\n}\n\nasync function buildStorybook({\n configDir,\n staticDir,\n outputDir,\n}: {\n configDir: string;\n staticDir?: string | undefined;\n outputDir: string;\n}): Promise<void> {\n await fs.promises.rm(outputDir, { recursive: true, force: true });\n\n const buildCommandParts = resolveBuildCommandParts();\n\n if (!buildCommandParts[0]) {\n throw new Error('Failed to resolve build command parts');\n }\n\n const params = [\n ...buildCommandParts,\n '--output-dir',\n outputDir,\n '--config-dir',\n configDir,\n ];\n\n if (staticDir) {\n params.push('--static-dir', staticDir);\n }\n\n let binary = fs.existsSync('yarn.lock') ? 'yarn' : 'npx';\n\n if (buildCommandParts[0].includes('node_modules')) {\n binary = buildCommandParts[0];\n params.shift(); // remove binary from params\n }\n\n if (HAPPO_DEBUG) {\n console.log(`[happo] Using build command \\`${binary} ${params.join(' ')}\\``);\n }\n\n return new Promise((resolve, reject) => {\n const spawned = spawn(binary, params, {\n stdio: 'inherit',\n shell: process.platform == 'win32',\n });\n\n spawned.on('exit', (code) => {\n if (code === 0) {\n try {\n fs.unlinkSync(path.join(outputDir, 'project.json'));\n } catch (error) {\n console.warn(\n `Ignoring error when attempting to remove project.json: ${error}`,\n );\n }\n resolve();\n } else {\n reject(new Error('Failed to build static storybook package'));\n }\n });\n });\n}\n\nexport interface BuildStorybookPackageResult {\n packageDir: string;\n estimatedSnapsCount?: number;\n}\n\nexport default async function buildStorybookPackage({\n configDir = '.storybook',\n staticDir,\n outputDir = '.out',\n usePrebuiltPackage = false,\n skip,\n}: Omit<StorybookIntegration, 'type'>): Promise<BuildStorybookPackageResult> {\n if (!usePrebuiltPackage) {\n await buildStorybook({ configDir, staticDir, outputDir });\n }\n\n const iframePath = path.join(outputDir, 'iframe.html');\n if (!fs.existsSync(iframePath)) {\n throw new Error(\n 'Failed to build static storybook package (missing iframe.html)',\n );\n }\n\n try {\n const skipped =\n typeof skip === 'function' ? await skip() : Array.isArray(skip) ? skip : [];\n\n assertSkippedIsSkipItems(skipped);\n\n const iframeContent = await fs.promises.readFile(iframePath, 'utf8');\n\n await fs.promises.writeFile(\n iframePath,\n iframeContent.replace(\n '<head>',\n `<head>\n <meta name=\"viewport\" content=\"width=device-width, initial-scale=1\">\n <script type=\"text/javascript\">window.__IS_HAPPO_RUN = true;</script>\n <script type=\"text/javascript\">window.happoSkipped = ${JSON.stringify(\n skipped,\n )};</script>\n `,\n ),\n );\n\n const estimatedSnapsCount = await getStorybookStoryCount(outputDir);\n\n const result: BuildStorybookPackageResult = { packageDir: outputDir };\n if (estimatedSnapsCount != null) {\n result.estimatedSnapsCount = estimatedSnapsCount;\n }\n return result;\n } catch (e) {\n console.error(e);\n throw e;\n }\n}\n", "import fs from 'node:fs';\nimport path from 'node:path';\n\nconst { HAPPO_DEBUG } = process.env;\n\nexport default function getStorybookBuildCommandParts(\n packageJsonPath: string = path.join(process.cwd(), 'package.json'),\n): [string, string] {\n try {\n const data = fs.readFileSync(packageJsonPath, 'utf8');\n const packageJson = JSON.parse(data);\n\n if (packageJson.scripts.storybook) {\n if (HAPPO_DEBUG) {\n console.log(\n '[happo] Found `storybook` script in package.json. Will attempt to use binary found at `node_modules/.bin/storybook` instead',\n );\n }\n\n const pathToStorybookCommand = path.join(\n process.cwd(),\n 'node_modules',\n '.bin',\n 'storybook',\n );\n\n if (fs.existsSync(pathToStorybookCommand)) {\n return [pathToStorybookCommand, 'build'];\n }\n }\n } catch (e) {\n if (HAPPO_DEBUG) {\n console.log(\n '[happo] Caught error when resolving Storybook build command parts. Will use default.',\n e,\n );\n }\n }\n\n return ['storybook', 'build'];\n}\n", "import fs from 'node:fs';\nimport path from 'node:path';\n\ninterface StorybookIndexEntry {\n type: string;\n}\n\ninterface StorybookIndex {\n entries?: Record<string, StorybookIndexEntry>;\n stories?: Record<string, StorybookIndexEntry>;\n}\n\n/**\n * Reads the storybook index.json from the given package directory and returns\n * the total number of story entries (excluding docs and other non-story types).\n * Returns undefined if the file cannot be read or parsed.\n */\nexport default async function getStorybookStoryCount(\n packageDir: string,\n): Promise<number | undefined> {\n const indexPath = path.join(packageDir, 'index.json');\n try {\n const content = await fs.promises.readFile(indexPath, 'utf8');\n const data = JSON.parse(content) as StorybookIndex;\n const entries = data.entries ?? data.stories ?? {};\n return Object.values(entries).filter((e) => e.type === 'story').length;\n } catch (error) {\n console.warn('Failed to get estimated snaps count from Storybook:', error);\n return undefined;\n }\n}\n", "import fs from 'node:fs';\nimport path from 'node:path';\n\nexport default function getStorybookVersionFromPackageJson(\n packageJsonPath: string = path.join(process.cwd(), 'package.json'),\n): number {\n const data = fs.readFileSync(packageJsonPath, 'utf8');\n const packageJson = JSON.parse(data);\n\n const combinedDependencies = {\n ...packageJson.dependencies,\n ...packageJson.devDependencies,\n };\n\n const storybookPackage = [\n 'storybook',\n '@storybook/react',\n '@storybook/angular',\n '@storybook/vue',\n ].find((pkg) => combinedDependencies[pkg]);\n\n if (storybookPackage) {\n const storybookVersion = combinedDependencies[storybookPackage];\n const majorVersion = Number.parseInt(storybookVersion.match(/\\d+/)[0], 10);\n return majorVersion;\n } else {\n throw new Error('Storybook is not listed as a dependency in package.json');\n }\n}\n", "import fs from 'node:fs';\nimport path from 'node:path';\nimport { Readable } from 'node:stream';\n\nimport type { Zippable } from 'fflate';\nimport { zip } from 'fflate';\n\nimport createHash from './createHash.ts';\nimport validateArchive from './validateArchive.ts';\n\n// Normalize path separators to forward slashes for cross-platform consistency.\n// path.relative() returns backslashes on Windows; callers may also pass names\n// built with path.join() on Windows.\nfunction normalizeEntryName(name: string): string {\n return name.replaceAll('\\\\', '/');\n}\n\n// We're setting the creation date to the same for all files so that the zip\n// packages created for the same content ends up having the same fingerprint.\n// https://github.com/101arrowz/fflate/issues/219#issuecomment-2333945868\nconst FILE_CREATION_DATE = new Date(2019, 1, 8, 13, 31, 55);\n\n// Type definitions\ninterface FileEntry {\n name: string;\n stream: fs.ReadStream;\n}\n\nexport interface ArchiveContentEntry {\n name: string;\n content: string | Buffer | fs.ReadStream | Readable;\n}\n\ninterface ArchiveResult {\n buffer: Buffer<ArrayBuffer>;\n hash: string;\n}\n\ninterface ArchiveEntry {\n name: string;\n size: number;\n}\n\n/**\n * Resolves all files in a directory and all of its subdirectories\n *\n * @param dirOrFile - The directory or file path to resolve\n * @returns Promise resolving to an array of file entries\n */\nasync function resolveFilesRecursiveForDir(\n dirOrFile: string,\n): Promise<Array<FileEntry>> {\n const resolvedDirOrFile = path.resolve(dirOrFile);\n const isDir = (await fs.promises.lstat(resolvedDirOrFile)).isDirectory();\n\n if (isDir) {\n const fileEntries: Array<FileEntry> = [];\n\n for await (const fileType of fs.promises.glob('**/*', {\n cwd: resolvedDirOrFile,\n withFileTypes: true,\n })) {\n // Check if it's a file (not a directory)\n if (fileType.isFile()) {\n const fullPath = `${fileType.parentPath}/${fileType.name}`;\n\n fileEntries.push({\n name: normalizeEntryName(path.relative(resolvedDirOrFile, fullPath)),\n stream: fs.createReadStream(fullPath),\n });\n }\n }\n\n return fileEntries;\n }\n\n return [\n {\n name: normalizeEntryName(path.relative(process.cwd(), resolvedDirOrFile)),\n stream: fs.createReadStream(resolvedDirOrFile),\n },\n ];\n}\n\n/**\n * Resolves all files in all directories recursively\n *\n * @param dirsAndFiles - Variable number of directory and file paths\n * @returns Promise resolving to a flattened array of file entries\n */\nasync function resolveFilesRecursive(\n ...dirsAndFiles: Array<string>\n): Promise<Array<FileEntry>> {\n const files = await Promise.all(\n dirsAndFiles.map((dirOrFile) => resolveFilesRecursiveForDir(dirOrFile)),\n );\n\n return files.flat();\n}\n\n/**\n * Converts a stream to a Uint8Array\n */\nasync function streamToUint8Array(\n stream: fs.ReadStream | Readable,\n): Promise<Uint8Array> {\n const chunks: Array<Uint8Array> = [];\n for await (const chunk of stream) {\n chunks.push(chunk instanceof Uint8Array ? chunk : new Uint8Array(chunk));\n }\n const totalLength = chunks.reduce((sum, chunk) => sum + chunk.length, 0);\n const result = new Uint8Array(totalLength);\n let offset = 0;\n for (const chunk of chunks) {\n result.set(chunk, offset);\n offset += chunk.length;\n }\n return result;\n}\n\n/**\n * Converts content to Uint8Array\n */\nasync function contentToUint8Array(\n content: string | Buffer | fs.ReadStream | Readable,\n): Promise<Uint8Array> {\n if (typeof content === 'string') {\n return new TextEncoder().encode(content);\n }\n if (Buffer.isBuffer(content)) {\n return new Uint8Array(content);\n }\n return streamToUint8Array(content);\n}\n\n/**\n * Creates a deterministic archive of the given files\n *\n * @param dirsAndFiles - Array of directory and file paths to include\n * @param contentToArchive - Array of content entries to include in the archive\n * @returns Promise resolving to archive result with buffer and hash\n */\nexport default async function deterministicArchive(\n dirsAndFiles: Array<string>,\n contentToArchive: Array<ArchiveContentEntry> = [],\n): Promise<ArchiveResult> {\n const uniqueDirsAndFiles = Array.from(new Set(dirsAndFiles));\n\n // Sort by name to make the output deterministic\n // Use simple string comparison instead of localeCompare for cross-platform determinism\n const filesToArchiveSorted = (\n await resolveFilesRecursive(...uniqueDirsAndFiles)\n ).toSorted((a, b) => (a.name < b.name ? -1 : a.name > b.name ? 1 : 0));\n\n const contentToArchiveSorted = contentToArchive.toSorted((a, b) =>\n a.name < b.name ? -1 : a.name > b.name ? 1 : 0,\n );\n\n const seenFiles = new Set<string>();\n const entries: Array<ArchiveEntry> = [];\n\n // Collect all entries with their data first\n interface EntryData {\n name: string;\n data: Uint8Array;\n }\n\n const entryDataList: Array<EntryData> = [];\n\n // Process files from disk\n for (const file of filesToArchiveSorted) {\n if (!seenFiles.has(file.name)) {\n const data = await streamToUint8Array(file.stream);\n entryDataList.push({ name: file.name, data });\n entries.push({ name: file.name, size: data.length });\n seenFiles.add(file.name);\n }\n }\n\n // Process in-memory content\n // Extract basename to match archiver's behavior with prefix: '' for content entries\n for (const file of contentToArchiveSorted) {\n const normalizedName = normalizeEntryName(file.name);\n if (!seenFiles.has(normalizedName)) {\n const data = await contentToUint8Array(file.content);\n entryDataList.push({ name: normalizedName, data });\n entries.push({ name: normalizedName, size: data.length });\n seenFiles.add(normalizedName);\n }\n }\n\n // Sort all entries by name to ensure deterministic order\n // Use simple string comparison instead of localeCompare for cross-platform determinism\n entryDataList.sort((a, b) => (a.name < b.name ? -1 : a.name > b.name ? 1 : 0));\n\n // Build zipData object in sorted order to ensure deterministic zip creation\n const zipData: Zippable = {};\n for (const entry of entryDataList) {\n zipData[entry.name] = [\n entry.data,\n {\n mtime: FILE_CREATION_DATE,\n level: 6,\n },\n ];\n }\n\n const zipBuffer = await new Promise<Uint8Array>((resolve, reject) => {\n zip(zipData, { level: 6 }, (err, data) => {\n if (err) {\n reject(err);\n } else {\n resolve(data);\n }\n });\n });\n const buffer = Buffer.from(zipBuffer);\n validateArchive(buffer.length, entries);\n const hash = createHash(buffer);\n\n return { buffer, hash };\n}\n", "/**\n * Validates that the archive was created successfully\n * @param totalBytes - The total bytes in the archive\n * @param entries - Array of archive entries\n */\nexport default function validateArchive(\n totalBytes: number,\n entries: Array<{ name: string; size: number }>,\n): void {\n const totalMegaBytes = Math.round(totalBytes / 1024 / 1024);\n\n if (totalMegaBytes < 30) {\n return;\n }\n\n const messageBits = [\n `Package size is ${totalMegaBytes} MB (${totalBytes} bytes), maximum is 60 MB.`,\n \"Here are the largest 20 files in the archive. Consider removing ones that aren't necessary.\",\n ];\n\n const fileSizes = entries.map((entry) => ({\n name: entry.name,\n size: entry.size || 0,\n }));\n\n for (const file of fileSizes.toSorted((a, b) => b.size - a.size).slice(0, 20)) {\n messageBits.push(\n `${file.name}: ${Math.round(file.size / 1024 / 1024)} MB (${file.size} bytes)`,\n );\n }\n\n if (totalMegaBytes > 60) {\n throw new Error(messageBits.join('\\n'));\n }\n\n console.warn(messageBits.join('\\n'));\n}\n", "import { styleText } from 'node:util';\n\ntype PrintFunction = (str: string) => void;\n\nconst red = (str: string) => styleText('red', str);\nconst green = (str: string) => styleText('green', str);\nconst dim = (str: string) => styleText('dim', str);\nconst underline = (str: string) => styleText('underline', str);\n\nexport function logTag(project?: string): string {\n return project ? `[${project}] ` : '';\n}\n\nfunction printDuration(print: PrintFunction, startTime?: number): void {\n if (startTime) {\n print(dim(` (${Date.now() - startTime}ms)`));\n }\n}\n\ninterface LoggerOptions {\n stderrPrint?: PrintFunction;\n print?: PrintFunction;\n}\n\ninterface StartOptions {\n startTime?: number;\n}\n\nexport default class Logger {\n private print: PrintFunction;\n private stderrPrint: PrintFunction;\n private startTime?: number | undefined;\n private startMsg?: string | undefined;\n\n constructor({\n stderrPrint = (str: string) => process.stderr.write(str),\n print = (str: string) => process.stdout.write(str),\n }: LoggerOptions = {}) {\n this.print = print;\n this.stderrPrint = stderrPrint;\n this.startTime = undefined;\n this.startMsg = undefined;\n }\n\n mute(): void {\n this.print = () => null;\n this.stderrPrint = () => null;\n }\n\n divider(): void {\n this.info('-----------------------------------------');\n }\n\n info(msg: string): void {\n this.print(`${msg}`.replaceAll(/https?:\\/\\/[^ ]+/g, underline));\n this.print('\\n');\n }\n\n start(msg?: string, { startTime }: StartOptions = {}): void {\n this.startTime = startTime || Date.now();\n this.startMsg = msg;\n if (msg) {\n this.print(`Starting: ${msg} `);\n this.print('\\n');\n }\n }\n\n success(msg?: string): void {\n this.print(green('\u2713'));\n\n if (this.startMsg) {\n this.print(green(` ${this.startMsg}:`));\n }\n\n if (msg) {\n this.print(green(` ${msg}`));\n }\n printDuration(this.print, this.startTime);\n this.print('\\n');\n\n this.startMsg = undefined;\n }\n\n fail(msg?: string): void {\n this.print(red('\u2717'));\n\n if (this.startMsg) {\n this.print(red(` ${this.startMsg}:`));\n }\n\n if (msg) {\n this.print(red(` ${msg}`));\n }\n printDuration(this.print, this.startTime);\n this.print('\\n');\n\n this.startMsg = undefined;\n }\n\n error(e: Error | string): void {\n let stack: string | undefined;\n if (typeof e === 'object' && e.stack) {\n stack = e.stack;\n if (stack) {\n stack = stack.split(`file://${process.cwd()}/`).join('');\n }\n }\n this.stderrPrint(\n red(stack || (typeof e === 'object' ? e.message : e) || String(e)),\n );\n this.stderrPrint('\\n');\n }\n\n warn(message: string): void {\n this.stderrPrint(red(message));\n this.stderrPrint('\\n');\n }\n}\n", "import { createHash } from 'node:crypto';\n\nimport retry from 'async-retry';\n\nimport type { ConfigWithDefaults } from '../config/index.ts';\nimport { logTag } from '../utils/Logger.ts';\nimport makeHappoAPIRequest from './makeHappoAPIRequest.ts';\n\n// Type definitions\ninterface Logger {\n info: (message: string) => void;\n warn: (message: string) => void;\n}\n\ninterface UploadAssetsOptions {\n hash: string;\n logger: Logger;\n}\n\nexport default async function uploadAssets(\n buffer: Buffer<ArrayBuffer>,\n options: UploadAssetsOptions,\n config: ConfigWithDefaults,\n): Promise<string> {\n const { project } = config;\n const { hash, logger } = options;\n\n // First we need to get the signed URL from Happo.\n const signedUrlRes = await makeHappoAPIRequest(\n {\n path: `/api/snap-requests/assets/${hash}/signed-url`,\n method: 'GET',\n },\n config,\n { retryCount: 3 },\n );\n\n if (!signedUrlRes) {\n throw new Error('Failed to get signed URL');\n }\n\n if ('path' in signedUrlRes) {\n // If the asset has already been uploaded the response will have a path and\n // we can return it now.\n const { path: signedUrlPath } = signedUrlRes;\n\n logger.info(`${logTag(project)}Reusing existing assets at ${signedUrlPath}`);\n return typeof signedUrlPath === 'string' ? signedUrlPath : String(signedUrlPath);\n }\n\n if (!('signedUrl' in signedUrlRes)) {\n throw new Error(\n `Signed URL response does not have path or signedUrl. Response: ${JSON.stringify(signedUrlRes, null, 2)}`,\n );\n }\n\n const { signedUrl } = signedUrlRes;\n\n // Upload the assets to the signed URL using node's built-in fetch with\n // retries\n await retry(\n async (bail: (error: Error) => void) => {\n const res = await fetch(String(signedUrl), {\n method: 'PUT',\n body: buffer,\n headers: {\n 'Content-Type': 'application/zip',\n },\n signal: AbortSignal.timeout(60_000),\n });\n\n if (!res.ok) {\n const error = new Error(\n `Failed to upload assets to S3 signed URL: ${res.status} ${res.statusText}`,\n );\n\n if (res.status < 500 || res.status >= 600) {\n // If it's not a 5xx error, bail immediately instead of retrying\n bail(error);\n return;\n }\n\n throw error;\n }\n\n // Verify the upload succeeded by checking the ETag header. S3 always\n // returns an ETag matching the MD5 of the uploaded content. A firewall\n // or transparent proxy returning a fake 200 will typically not include\n // a correct ETag, catching the case where the payload never reached S3.\n const etag = res.headers.get('etag');\n const expectedEtag = createHash('md5').update(buffer).digest('hex');\n if (!etag || !etag.includes(expectedEtag)) {\n const error = new Error(\n `S3 upload verification failed: expected ETag to include ${expectedEtag}, got ${etag ?? '(none)'}. ` +\n `A firewall may be intercepting the upload.`,\n );\n bail(error);\n return;\n }\n\n return res;\n },\n {\n retries: 3,\n onRetry: (error: Error, attempt: number) => {\n logger.warn(\n `${logTag(project)}PUT request attempt ${attempt} failed: ${error.message}. Retrying...`,\n );\n },\n },\n );\n\n // Finally, we need to tell Happo that we've uploaded the assets.\n const finalizeRes = await makeHappoAPIRequest(\n {\n path: `/api/snap-requests/assets/${hash}/signed-url/finalize`,\n method: 'POST',\n },\n config,\n { retryCount: 3 },\n );\n\n if (!finalizeRes) {\n throw new Error('Failed to finalize assets');\n }\n\n if (!('path' in finalizeRes)) {\n throw new Error('Finalize response is missing path');\n }\n\n const { path: finalizedPath } = finalizeRes;\n\n return typeof finalizedPath === 'string' ? finalizedPath : String(finalizedPath);\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": ["fs", "path", "requestIds", "fs", "path", "fs", "path", "fs", "path", "HAPPO_DEBUG", "fs", "path", "fs", "path", "path", "fs", "createHash", "createHash", "path", "fs"]
|
|
7
|
+
}
|
|
@@ -3,21 +3,21 @@ import {
|
|
|
3
3
|
} from "./chunk-JTRP4JVC.js";
|
|
4
4
|
import {
|
|
5
5
|
startJob
|
|
6
|
-
} from "./chunk-
|
|
6
|
+
} from "./chunk-MGNFR3W2.js";
|
|
7
7
|
import {
|
|
8
8
|
createAsyncComparison
|
|
9
|
-
} from "./chunk-
|
|
9
|
+
} from "./chunk-MCYQSPED.js";
|
|
10
10
|
import {
|
|
11
11
|
postGitHubComment
|
|
12
12
|
} from "./chunk-X4TE2VNY.js";
|
|
13
13
|
import {
|
|
14
14
|
cancelJob
|
|
15
|
-
} from "./chunk-
|
|
15
|
+
} from "./chunk-JMN6VM22.js";
|
|
16
16
|
import {
|
|
17
17
|
makeHappoAPIRequest
|
|
18
|
-
} from "./chunk-
|
|
19
|
-
import "./chunk-
|
|
20
|
-
import "./chunk-
|
|
18
|
+
} from "./chunk-2LNQIG6R.js";
|
|
19
|
+
import "./chunk-SE7XKHF6.js";
|
|
20
|
+
import "./chunk-OJHKEE3W.js";
|
|
21
21
|
|
|
22
22
|
// src/e2e/wrapper.ts
|
|
23
23
|
import { spawn } from "node:child_process";
|
|
@@ -28,7 +28,6 @@ async function postAsyncReport(requestIds, environment, happoConfig) {
|
|
|
28
28
|
{
|
|
29
29
|
path: `/api/async-reports/${afterSha}`,
|
|
30
30
|
method: "POST",
|
|
31
|
-
json: true,
|
|
32
31
|
body: {
|
|
33
32
|
requestIds,
|
|
34
33
|
project: happoConfig.project,
|
|
@@ -69,7 +68,6 @@ async function finalizeAll({
|
|
|
69
68
|
{
|
|
70
69
|
path: `/api/async-reports/${afterSha}/finalize`,
|
|
71
70
|
method: "POST",
|
|
72
|
-
json: true,
|
|
73
71
|
body
|
|
74
72
|
},
|
|
75
73
|
happoConfig,
|
|
@@ -219,4 +217,4 @@ export {
|
|
|
219
217
|
runWithWrapper as default,
|
|
220
218
|
finalizeAll
|
|
221
219
|
};
|
|
222
|
-
//# sourceMappingURL=wrapper-
|
|
220
|
+
//# sourceMappingURL=wrapper-QAPBXQCV.js.map
|
|
@@ -0,0 +1,7 @@
|
|
|
1
|
+
{
|
|
2
|
+
"version": 3,
|
|
3
|
+
"sources": ["../../src/e2e/wrapper.ts"],
|
|
4
|
+
"sourcesContent": ["import { spawn } from 'node:child_process';\nimport http from 'node:http';\n\nimport type { ConfigWithDefaults, E2EIntegration } from '../config/index.ts';\nimport type { EnvironmentResult } from '../environment/index.ts';\nimport cancelJob from '../network/cancelJob.ts';\nimport createAsyncComparison from '../network/createAsyncComparison.ts';\nimport makeHappoAPIRequest from '../network/makeHappoAPIRequest.ts';\nimport postGitHubComment from '../network/postGitHubComment.ts';\nimport startJob, { type StartJobResult } from '../network/startJob.ts';\nimport startServer, { type ServerInfo } from '../network/startServer.ts';\n\nlet allRequestIds: Set<number>;\n\nasync function postAsyncReport(\n requestIds: Array<number>,\n environment: EnvironmentResult,\n happoConfig: ConfigWithDefaults,\n) {\n const { afterSha, nonce, link, message } = environment;\n return await makeHappoAPIRequest(\n {\n path: `/api/async-reports/${afterSha}`,\n method: 'POST',\n body: {\n requestIds,\n project: happoConfig.project,\n nonce,\n link,\n message,\n },\n },\n happoConfig,\n { retryCount: 2 },\n );\n}\n\ntype Example = {\n component: string;\n variant: string;\n target: string;\n};\n\ntype Logger = Pick<Console, 'log' | 'error'>;\n\ninterface FinalizeAllOptions {\n happoConfig: ConfigWithDefaults;\n environment: EnvironmentResult;\n skippedExamplesJSON?: string;\n logger: Logger;\n}\n\nexport async function finalizeAll({\n happoConfig,\n environment,\n skippedExamplesJSON,\n logger,\n}: FinalizeAllOptions): Promise<void> {\n const { afterSha, nonce } = environment;\n\n if (!nonce) {\n throw new Error('[HAPPO] Missing --nonce argument');\n }\n\n const body: {\n project?: string | undefined;\n nonce: string;\n skippedExamples: Array<Example>;\n } = {\n project: happoConfig.project,\n nonce,\n skippedExamples: [],\n };\n\n if (skippedExamplesJSON) {\n try {\n const skippedExamples = JSON.parse(skippedExamplesJSON);\n body.skippedExamples = skippedExamples;\n } catch (e) {\n logger.error('Error when parsing --skippedExamples', skippedExamplesJSON);\n throw e;\n }\n }\n await makeHappoAPIRequest(\n {\n path: `/api/async-reports/${afterSha}/finalize`,\n method: 'POST',\n body,\n },\n happoConfig,\n { retryCount: 3 },\n );\n\n if (environment.beforeSha !== environment.afterSha) {\n const compareResult = await createAsyncComparison(\n happoConfig,\n environment,\n logger,\n );\n\n if (environment.link && environment.githubToken && happoConfig.githubApiUrl) {\n // githubToken and githubApiUrl are set which means that we should post\n // a comment to the PR.\n // https://docs.happo.io/docs/continuous-integration#posting-statuses-without-installing-the-happo-github-app\n await postGitHubComment({\n authToken: environment.githubToken,\n link: environment.link,\n statusImageUrl: compareResult.statusImageUrl,\n compareUrl: compareResult.compareUrl,\n githubApiUrl: happoConfig.githubApiUrl,\n });\n }\n }\n}\n\nasync function finalizeHappoReport(\n happoConfig: ConfigWithDefaults,\n environment: EnvironmentResult,\n job: StartJobResult,\n logger: Logger,\n) {\n if (!allRequestIds.size) {\n logger.log(`[HAPPO] No snapshots were recorded. Ignoring.`);\n return;\n }\n const reportResult = await postAsyncReport(\n [...allRequestIds],\n environment,\n happoConfig,\n );\n\n if (!reportResult) {\n throw new Error('Failed to create async Happo report');\n }\n\n const { nonce } = environment;\n\n if (!nonce) {\n // If there is a nonce, the comparison will happen when the finalize\n // command is called.\n const compareResult = await createAsyncComparison(\n happoConfig,\n environment,\n logger,\n );\n\n if (\n compareResult &&\n environment.link &&\n environment.githubToken &&\n happoConfig.githubApiUrl\n ) {\n // githubToken and githubApiUrl is set which means that we should post\n // a comment to the PR.\n // https://docs.happo.io/docs/continuous-integration#posting-statuses-without-installing-the-happo-github-app\n await postGitHubComment({\n authToken: environment.githubToken,\n link: environment.link,\n statusImageUrl: compareResult.statusImageUrl,\n compareUrl: compareResult.compareUrl,\n githubApiUrl: happoConfig.githubApiUrl,\n });\n }\n }\n logger.log(`[HAPPO] ${job.url}`);\n}\n\nfunction startE2EServer(\n environment: EnvironmentResult,\n happoConfig: ConfigWithDefaults,\n): Promise<ServerInfo> {\n function requestHandler(req: http.IncomingMessage, res: http.ServerResponse) {\n const bodyParts: Array<string> = [];\n req.on('data', (chunk: Buffer) => {\n bodyParts.push(chunk.toString());\n });\n req.on('end', async () => {\n const potentialIds = bodyParts\n .join('')\n .split('\\n')\n .filter(Boolean)\n .map((requestId) => Number.parseInt(requestId, 10));\n\n if (potentialIds.some((id) => Number.isNaN(id))) {\n res.writeHead(400);\n res.end('invalid payload');\n return;\n }\n\n for (const requestId of potentialIds) {\n allRequestIds.add(requestId);\n }\n\n const { nonce } = environment;\n if (nonce && potentialIds.length) {\n // Associate these snapRequests with the async report as soon as possible\n await postAsyncReport(potentialIds, environment, happoConfig);\n }\n res.writeHead(200);\n res.end('');\n });\n }\n return startServer(requestHandler);\n}\n\nfunction assertE2EIntegration(\n integration: NonNullable<ConfigWithDefaults['integration']>,\n): asserts integration is E2EIntegration {\n if (integration.type !== 'cypress' && integration.type !== 'playwright') {\n throw new Error(`Unsupported integration type: ${integration.type}`);\n }\n}\n\n/**\n * Runs a command with the wrapper and returns the exit code.\n *\n * @param dashdashCommandParts The command to run with the wrapper\n * @param happoConfig The Happo config\n * @param environment The environment\n * @param port The port to listen on\n * @param logger The logger\n * @returns The exit code of the command\n */\nexport default async function runWithWrapper(\n dashdashCommandParts: Array<string>,\n happoConfig: ConfigWithDefaults,\n environment: EnvironmentResult,\n logger: Logger,\n configFilePath: string,\n): Promise<number> {\n allRequestIds = new Set<number>();\n const e2eServer = await startE2EServer(environment, happoConfig);\n logger.log(`[HAPPO] Listening on port ${e2eServer.port}`);\n\n const job = await startJob(happoConfig, environment, logger);\n if (!job) {\n throw new Error('Failed to create Happo job');\n }\n try {\n const exitCode = await new Promise<number>((resolve, reject) => {\n const child = spawn(dashdashCommandParts[0]!, dashdashCommandParts.slice(1), {\n stdio: 'inherit',\n env: {\n ...process.env,\n HAPPO_E2E_PORT: e2eServer.port.toString(),\n HAPPO_CONFIG_FILE: configFilePath,\n HAPPO_API_KEY: happoConfig.apiKey,\n HAPPO_API_SECRET: happoConfig.apiSecret,\n },\n shell: process.platform == 'win32',\n });\n\n child.on('error', (e) => {\n return reject(e);\n });\n\n const e2eIntegration = happoConfig.integration;\n assertE2EIntegration(e2eIntegration);\n child.on('close', async (code: number) => {\n if (code === 0 || e2eIntegration.allowFailures) {\n try {\n await finalizeHappoReport(happoConfig, environment, job, logger);\n } catch (e) {\n logger.error('Failed to finalize Happo report', e);\n return reject(e);\n }\n } else {\n logger.error(\n 'Command failed with exit code ${code}. Cancelling Happo job.',\n );\n try {\n await cancelJob(\n 'failure',\n `${e2eIntegration.type} run failed`,\n happoConfig,\n environment,\n logger,\n );\n } catch (e) {\n logger.error('Failed to cancel Happo job', e);\n return reject(e);\n }\n }\n resolve(code);\n });\n });\n return exitCode;\n } finally {\n allRequestIds.clear();\n await e2eServer.close();\n }\n}\n"],
|
|
5
|
+
"mappings": "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",
|
|
6
|
+
"names": []
|
|
7
|
+
}
|
|
@@ -1 +1 @@
|
|
|
1
|
-
{"version":3,"file":"RemoteBrowserTarget.d.ts","sourceRoot":"","sources":["../../src/config/RemoteBrowserTarget.ts"],"names":[],"mappings":"
|
|
1
|
+
{"version":3,"file":"RemoteBrowserTarget.d.ts","sourceRoot":"","sources":["../../src/config/RemoteBrowserTarget.ts"],"names":[],"mappings":"AAGA,OAAO,KAAK,EACV,WAAW,EACX,kBAAkB,EAClB,IAAI,EACJ,kBAAkB,EACnB,MAAM,YAAY,CAAC;AA6CpB,MAAM,WAAW,QAAQ;IACvB,EAAE,EAAE,MAAM,CAAC;IACX,WAAW,EAAE,OAAO,CAAC;IACrB,GAAG,EAAE,MAAM,CAAC;CACb;AAED,MAAM,WAAW,aAAa;IAC5B,SAAS,CAAC,EAAE,MAAM,GAAG,KAAK,CAAC,QAAQ,CAAC,CAAC;IAErC,iCAAiC;IACjC,aAAa,CAAC,EAAE,MAAM,CAAC;IAEvB,iCAAiC;IACjC,aAAa,CAAC,EAAE,MAAM,CAAC;IAEvB,YAAY,CAAC,EAAE,KAAK,CAAC,OAAO,CAAC,CAAC;IAC9B,KAAK,CAAC,EAAE,KAAK,CAAC,IAAI,CAAC,CAAC;IACpB,UAAU,CAAC,EAAE,MAAM,CAAC;IAEpB;;;;OAIG;IACH,mBAAmB,CAAC,EAAE,MAAM,CAAC;CAC9B;AAkHD,MAAM,CAAC,OAAO,OAAO,mBAAmB;IACtC,SAAgB,MAAM,EAAE,MAAM,GAAG,SAAS,CAAC;IAC3C,SAAgB,WAAW,EAAE,WAAW,CAAC;IACzC,SAAgB,QAAQ,EAAE,MAAM,CAAC;IACjC,SAAgB,SAAS,EAAE,MAAM,GAAG,SAAS,CAAC;IAC9C,SAAgB,YAAY,EAAE,MAAM,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;gBAGpD,WAAW,EAAE,WAAW,EACxB,EACE,QAAqB,EACrB,MAAM,EACN,SAAS,EACT,GAAG,YAAY,EAChB,EAAE,kBAAkB;IAsBjB,OAAO,CACX,EACE,SAAS,EACT,aAAa,EACb,aAAa,EACb,YAAY,EACZ,KAAK,EACL,UAAU,EACV,mBAAmB,GACpB,EAAE,aAAa,EAChB,MAAM,EAAE,kBAAkB,GACzB,OAAO,CAAC,KAAK,CAAC,MAAM,CAAC,CAAC;CA6I1B"}
|