@jsenv/core 34.2.2 → 35.0.0
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/README.md +1 -1
- package/dist/html/explorer.html +5 -4
- package/dist/{jsenv.js → jsenv_core.js} +840 -3914
- package/package.json +7 -21
- package/src/build/build.js +34 -16
- package/src/build/version_mappings_injection.js +20 -27
- package/src/dev/file_service.js +9 -9
- package/src/dev/start_dev_server.js +3 -3
- package/src/dev/user_agent.js +1 -1
- package/src/kitchen/kitchen.js +5 -3
- package/src/main.js +0 -23
- package/src/plugins/autoreload/jsenv_plugin_autoreload_client.js +2 -2
- package/src/plugins/importmap/jsenv_plugin_importmap.js +6 -2
- package/src/plugins/{inline/jsenv_plugin_html_inline_content.js → inline_content_analysis/jsenv_plugin_html_inline_content_analysis.js} +12 -6
- package/src/plugins/{inline/jsenv_plugin_inline.js → inline_content_analysis/jsenv_plugin_inline_content_analysis.js} +8 -10
- package/src/plugins/{inline/jsenv_plugin_js_inline_content.js → inline_content_analysis/jsenv_plugin_js_inline_content_analysis.js} +4 -2
- package/src/plugins/inlining/jsenv_plugin_inlining.js +22 -0
- package/src/plugins/{inline/jsenv_plugin_inline_query_param.js → inlining/jsenv_plugin_inlining_as_data_url.js} +16 -9
- package/src/plugins/inlining/jsenv_plugin_inlining_into_html.js +149 -0
- package/src/plugins/plugins.js +5 -2
- package/src/plugins/ribbon/jsenv_plugin_ribbon.js +11 -10
- package/src/plugins/server_events/jsenv_plugin_server_events_client_injection.js +2 -2
- package/src/plugins/supervisor/html_supervisor_injection.js +23 -25
- package/src/plugins/supervisor/jsenv_plugin_supervisor.js +1 -1
- package/src/plugins/transpilation/babel/require_babel_plugin.js +1 -1
- package/src/plugins/transpilation/import_assertions/jsenv_plugin_import_assertions.js +20 -5
- package/src/plugins/transpilation/js_module_fallback/convert_js_module_to_js_classic.js +1 -1
- package/src/plugins/transpilation/js_module_fallback/jsenv_plugin_js_module_fallback_inside_html.js +2 -2
- package/src/plugins/transpilation/js_module_fallback/jsenv_plugin_js_module_fallback_on_workers.js +3 -3
- package/src/plugins/url_analysis/html/html_urls.js +1 -1
- package/dist/controllable_child_process.mjs +0 -129
- package/dist/controllable_worker_thread.mjs +0 -91
- package/dist/js/execute_using_dynamic_import.js +0 -850
- package/dist/js/v8_coverage.js +0 -508
- package/src/execute/execute.js +0 -109
- package/src/execute/run.js +0 -161
- package/src/execute/runtimes/browsers/chromium.js +0 -10
- package/src/execute/runtimes/browsers/firefox.js +0 -9
- package/src/execute/runtimes/browsers/from_playwright.js +0 -574
- package/src/execute/runtimes/browsers/middleware_istanbul.js +0 -65
- package/src/execute/runtimes/browsers/middleware_js_supervisor.js +0 -100
- package/src/execute/runtimes/browsers/webkit.js +0 -26
- package/src/execute/runtimes/node/child_exec_options.js +0 -166
- package/src/execute/runtimes/node/controllable_child_process.mjs +0 -135
- package/src/execute/runtimes/node/controllable_worker_thread.mjs +0 -103
- package/src/execute/runtimes/node/exec_options.js +0 -44
- package/src/execute/runtimes/node/execute_using_dynamic_import.js +0 -55
- package/src/execute/runtimes/node/exit_codes.js +0 -9
- package/src/execute/runtimes/node/kill_process_tree.js +0 -76
- package/src/execute/runtimes/node/node_child_process.js +0 -348
- package/src/execute/runtimes/node/node_execution_performance.js +0 -67
- package/src/execute/runtimes/node/node_worker_thread.js +0 -282
- package/src/execute/runtimes/node/profiler_v8_coverage.js +0 -56
- package/src/execute/runtimes/readme.md +0 -13
- package/src/execute/web_server_param.js +0 -74
- package/src/test/coverage/babel_plugin_instrument.js +0 -48
- package/src/test/coverage/coverage_reporter_html_directory.js +0 -32
- package/src/test/coverage/coverage_reporter_json_file.js +0 -17
- package/src/test/coverage/coverage_reporter_text_log.js +0 -19
- package/src/test/coverage/empty_coverage_factory.js +0 -52
- package/src/test/coverage/file_by_file_coverage.js +0 -25
- package/src/test/coverage/istanbul_coverage_composition.js +0 -28
- package/src/test/coverage/istanbul_coverage_map_from_coverage.js +0 -16
- package/src/test/coverage/list_files_not_covered.js +0 -15
- package/src/test/coverage/missing_coverage.js +0 -41
- package/src/test/coverage/report_to_coverage.js +0 -198
- package/src/test/coverage/v8_and_istanbul.js +0 -37
- package/src/test/coverage/v8_coverage.js +0 -26
- package/src/test/coverage/v8_coverage_composition.js +0 -24
- package/src/test/coverage/v8_coverage_node_directory.js +0 -85
- package/src/test/coverage/v8_coverage_to_istanbul.js +0 -99
- package/src/test/execute_steps.js +0 -425
- package/src/test/execute_test_plan.js +0 -372
- package/src/test/execution_colors.js +0 -10
- package/src/test/execution_steps.js +0 -65
- package/src/test/gc.js +0 -9
- package/src/test/logs_file_execution.js +0 -427
- package/src/test/logs_file_execution.test.mjs +0 -41
- package/src/test/readme.md +0 -3
- /package/src/{basic_fetch.js → helpers/basic_fetch.js} +0 -0
- /package/src/{lookup_package_directory.js → helpers/lookup_package_directory.js} +0 -0
- /package/src/{ping_server.js → helpers/ping_server.js} +0 -0
- /package/src/{require_from_jsenv.js → helpers/require_from_jsenv.js} +0 -0
- /package/src/{watch_source_files.js → helpers/watch_source_files.js} +0 -0
- /package/src/{web_url_converter.js → helpers/web_url_converter.js} +0 -0
- /package/src/plugins/{inline → inline_content_analysis}/client/inline_content.js +0 -0
- /package/src/plugins/{inline → inline_content_analysis}/jsenv_plugin_data_urls.js +0 -0
|
@@ -1,282 +0,0 @@
|
|
|
1
|
-
// https://github.com/avajs/ava/blob/576f534b345259055c95fa0c2b33bef10847a2af/lib/fork.js#L23
|
|
2
|
-
// https://nodejs.org/api/worker_threads.html
|
|
3
|
-
// https://github.com/avajs/ava/blob/576f534b345259055c95fa0c2b33bef10847a2af/lib/worker/base.js
|
|
4
|
-
import { Worker } from "node:worker_threads"
|
|
5
|
-
import { fileURLToPath } from "node:url"
|
|
6
|
-
import {
|
|
7
|
-
Abort,
|
|
8
|
-
createCallbackListNotifiedOnce,
|
|
9
|
-
raceCallbacks,
|
|
10
|
-
} from "@jsenv/abort"
|
|
11
|
-
import { memoize } from "@jsenv/utils/src/memoize/memoize.js"
|
|
12
|
-
|
|
13
|
-
import { createChildExecOptions } from "./child_exec_options.js"
|
|
14
|
-
import { ExecOptions } from "./exec_options.js"
|
|
15
|
-
import { EXIT_CODES } from "./exit_codes.js"
|
|
16
|
-
|
|
17
|
-
const CONTROLLABLE_WORKER_THREAD_URL = new URL(
|
|
18
|
-
"./controllable_worker_thread.mjs?entry_point",
|
|
19
|
-
import.meta.url,
|
|
20
|
-
).href
|
|
21
|
-
|
|
22
|
-
export const nodeWorkerThread = {
|
|
23
|
-
type: "node",
|
|
24
|
-
name: "node_worker_thread",
|
|
25
|
-
version: process.version.slice(1),
|
|
26
|
-
}
|
|
27
|
-
|
|
28
|
-
nodeWorkerThread.run = async ({
|
|
29
|
-
signal = new AbortController().signal,
|
|
30
|
-
// logger,
|
|
31
|
-
rootDirectoryUrl,
|
|
32
|
-
fileRelativeUrl,
|
|
33
|
-
|
|
34
|
-
keepRunning,
|
|
35
|
-
stopSignal,
|
|
36
|
-
onConsole,
|
|
37
|
-
|
|
38
|
-
collectConsole = false,
|
|
39
|
-
collectPerformance,
|
|
40
|
-
coverageEnabled = false,
|
|
41
|
-
coverageConfig,
|
|
42
|
-
coverageMethodForNodeJs,
|
|
43
|
-
coverageFileUrl,
|
|
44
|
-
|
|
45
|
-
env,
|
|
46
|
-
debugPort,
|
|
47
|
-
debugMode,
|
|
48
|
-
debugModeInheritBreak,
|
|
49
|
-
inheritProcessEnv = true,
|
|
50
|
-
commandLineOptions = [],
|
|
51
|
-
}) => {
|
|
52
|
-
if (env !== undefined && typeof env !== "object") {
|
|
53
|
-
throw new TypeError(`env must be an object, got ${env}`)
|
|
54
|
-
}
|
|
55
|
-
env = {
|
|
56
|
-
...env,
|
|
57
|
-
JSENV: true,
|
|
58
|
-
}
|
|
59
|
-
if (coverageMethodForNodeJs !== "NODE_V8_COVERAGE") {
|
|
60
|
-
env.NODE_V8_COVERAGE = ""
|
|
61
|
-
}
|
|
62
|
-
|
|
63
|
-
const workerThreadExecOptions = await createChildExecOptions({
|
|
64
|
-
signal,
|
|
65
|
-
debugPort,
|
|
66
|
-
debugMode,
|
|
67
|
-
debugModeInheritBreak,
|
|
68
|
-
})
|
|
69
|
-
const execArgvForWorkerThread = ExecOptions.toExecArgv({
|
|
70
|
-
...workerThreadExecOptions,
|
|
71
|
-
...ExecOptions.fromExecArgv(commandLineOptions),
|
|
72
|
-
})
|
|
73
|
-
const envForWorkerThread = {
|
|
74
|
-
...(inheritProcessEnv ? process.env : {}),
|
|
75
|
-
...env,
|
|
76
|
-
}
|
|
77
|
-
|
|
78
|
-
const cleanupCallbackList = createCallbackListNotifiedOnce()
|
|
79
|
-
const cleanup = async (reason) => {
|
|
80
|
-
await cleanupCallbackList.notify({ reason })
|
|
81
|
-
}
|
|
82
|
-
const actionOperation = Abort.startOperation()
|
|
83
|
-
actionOperation.addAbortSignal(signal)
|
|
84
|
-
// https://nodejs.org/api/worker_threads.html#new-workerfilename-options
|
|
85
|
-
const workerThread = new Worker(
|
|
86
|
-
fileURLToPath(CONTROLLABLE_WORKER_THREAD_URL),
|
|
87
|
-
{
|
|
88
|
-
env: envForWorkerThread,
|
|
89
|
-
execArgv: execArgvForWorkerThread,
|
|
90
|
-
// workerData: { options },
|
|
91
|
-
stdin: true,
|
|
92
|
-
stdout: true,
|
|
93
|
-
stderr: true,
|
|
94
|
-
},
|
|
95
|
-
)
|
|
96
|
-
const removeOutputListener = installWorkerThreadOutputListener(
|
|
97
|
-
workerThread,
|
|
98
|
-
({ type, text }) => {
|
|
99
|
-
onConsole({ type, text })
|
|
100
|
-
},
|
|
101
|
-
)
|
|
102
|
-
const workerThreadReadyPromise = new Promise((resolve) => {
|
|
103
|
-
onceWorkerThreadMessage(workerThread, "ready", resolve)
|
|
104
|
-
})
|
|
105
|
-
|
|
106
|
-
const stop = memoize(async () => {
|
|
107
|
-
// read all stdout before terminating
|
|
108
|
-
// (no need for stderr because it's sync)
|
|
109
|
-
if (collectConsole) {
|
|
110
|
-
while (workerThread.stdout.read() !== null) {}
|
|
111
|
-
await new Promise((resolve) => {
|
|
112
|
-
setTimeout(resolve, 50)
|
|
113
|
-
})
|
|
114
|
-
}
|
|
115
|
-
await workerThread.terminate()
|
|
116
|
-
})
|
|
117
|
-
|
|
118
|
-
const winnerPromise = new Promise((resolve) => {
|
|
119
|
-
raceCallbacks(
|
|
120
|
-
{
|
|
121
|
-
aborted: (cb) => {
|
|
122
|
-
return actionOperation.addAbortCallback(cb)
|
|
123
|
-
},
|
|
124
|
-
error: (cb) => {
|
|
125
|
-
return onceWorkerThreadEvent(workerThread, "error", cb)
|
|
126
|
-
},
|
|
127
|
-
exit: (cb) => {
|
|
128
|
-
return onceWorkerThreadEvent(workerThread, "exit", (code, signal) => {
|
|
129
|
-
cb({ code, signal })
|
|
130
|
-
})
|
|
131
|
-
},
|
|
132
|
-
response: (cb) => {
|
|
133
|
-
return onceWorkerThreadMessage(workerThread, "action-result", cb)
|
|
134
|
-
},
|
|
135
|
-
},
|
|
136
|
-
resolve,
|
|
137
|
-
)
|
|
138
|
-
})
|
|
139
|
-
|
|
140
|
-
const result = {
|
|
141
|
-
status: "executing",
|
|
142
|
-
errors: [],
|
|
143
|
-
namespace: null,
|
|
144
|
-
}
|
|
145
|
-
|
|
146
|
-
const writeResult = async () => {
|
|
147
|
-
actionOperation.throwIfAborted()
|
|
148
|
-
await workerThreadReadyPromise
|
|
149
|
-
actionOperation.throwIfAborted()
|
|
150
|
-
await sendToWorkerThread(workerThread, {
|
|
151
|
-
type: "action",
|
|
152
|
-
data: {
|
|
153
|
-
actionType: "execute-using-dynamic-import",
|
|
154
|
-
actionParams: {
|
|
155
|
-
rootDirectoryUrl,
|
|
156
|
-
fileUrl: new URL(fileRelativeUrl, rootDirectoryUrl).href,
|
|
157
|
-
collectPerformance,
|
|
158
|
-
coverageEnabled,
|
|
159
|
-
coverageConfig,
|
|
160
|
-
coverageMethodForNodeJs,
|
|
161
|
-
coverageFileUrl,
|
|
162
|
-
exitAfterAction: true,
|
|
163
|
-
},
|
|
164
|
-
},
|
|
165
|
-
})
|
|
166
|
-
const winner = await winnerPromise
|
|
167
|
-
if (winner.name === "aborted") {
|
|
168
|
-
result.status = "aborted"
|
|
169
|
-
return
|
|
170
|
-
}
|
|
171
|
-
if (winner.name === "error") {
|
|
172
|
-
const error = winner.data
|
|
173
|
-
removeOutputListener()
|
|
174
|
-
result.status = "failed"
|
|
175
|
-
result.errors.push(error)
|
|
176
|
-
return
|
|
177
|
-
}
|
|
178
|
-
if (winner.name === "exit") {
|
|
179
|
-
const { code } = winner.data
|
|
180
|
-
await cleanup("process exit")
|
|
181
|
-
if (code === 12) {
|
|
182
|
-
result.status = "failed"
|
|
183
|
-
result.errors.push(
|
|
184
|
-
new Error(
|
|
185
|
-
`node process exited with 12 (the forked child process wanted to use a non-available port for debug)`,
|
|
186
|
-
),
|
|
187
|
-
)
|
|
188
|
-
return
|
|
189
|
-
}
|
|
190
|
-
if (
|
|
191
|
-
code === null ||
|
|
192
|
-
code === 0 ||
|
|
193
|
-
code === EXIT_CODES.SIGINT ||
|
|
194
|
-
code === EXIT_CODES.SIGTERM ||
|
|
195
|
-
code === EXIT_CODES.SIGABORT
|
|
196
|
-
) {
|
|
197
|
-
result.status = "failed"
|
|
198
|
-
result.errors.push(
|
|
199
|
-
new Error(`node worker thread exited during execution`),
|
|
200
|
-
)
|
|
201
|
-
return
|
|
202
|
-
}
|
|
203
|
-
// process.exit(1) in child process or process.exitCode = 1 + process.exit()
|
|
204
|
-
// means there was an error even if we don't know exactly what.
|
|
205
|
-
result.status = "failed"
|
|
206
|
-
result.errors.push(
|
|
207
|
-
new Error(
|
|
208
|
-
`node worker thread exited with code ${code} during execution`,
|
|
209
|
-
),
|
|
210
|
-
)
|
|
211
|
-
}
|
|
212
|
-
const { status, value } = winner.data
|
|
213
|
-
if (status === "action-failed") {
|
|
214
|
-
result.status = "failed"
|
|
215
|
-
result.errors.push(value)
|
|
216
|
-
return
|
|
217
|
-
}
|
|
218
|
-
const { namespace, performance, coverage } = value
|
|
219
|
-
result.status = "completed"
|
|
220
|
-
result.namespace = namespace
|
|
221
|
-
result.performance = performance
|
|
222
|
-
result.coverage = coverage
|
|
223
|
-
}
|
|
224
|
-
|
|
225
|
-
try {
|
|
226
|
-
await writeResult()
|
|
227
|
-
} catch (e) {
|
|
228
|
-
result.status = "failed"
|
|
229
|
-
result.errors.push(e)
|
|
230
|
-
}
|
|
231
|
-
|
|
232
|
-
if (keepRunning) {
|
|
233
|
-
stopSignal.notify = stop
|
|
234
|
-
} else {
|
|
235
|
-
await stop()
|
|
236
|
-
}
|
|
237
|
-
await actionOperation.end()
|
|
238
|
-
return result
|
|
239
|
-
}
|
|
240
|
-
|
|
241
|
-
const installWorkerThreadOutputListener = (workerThread, callback) => {
|
|
242
|
-
// beware that we may receive ansi output here, should not be a problem but keep that in mind
|
|
243
|
-
const stdoutDataCallback = (chunk) => {
|
|
244
|
-
const text = String(chunk)
|
|
245
|
-
callback({ type: "log", text })
|
|
246
|
-
}
|
|
247
|
-
workerThread.stdout.on("data", stdoutDataCallback)
|
|
248
|
-
const stdErrorDataCallback = (chunk) => {
|
|
249
|
-
const text = String(chunk)
|
|
250
|
-
callback({ type: "error", text })
|
|
251
|
-
}
|
|
252
|
-
workerThread.stderr.on("data", stdErrorDataCallback)
|
|
253
|
-
return () => {
|
|
254
|
-
workerThread.stdout.removeListener("data", stdoutDataCallback)
|
|
255
|
-
workerThread.stderr.removeListener("data", stdErrorDataCallback)
|
|
256
|
-
}
|
|
257
|
-
}
|
|
258
|
-
|
|
259
|
-
const sendToWorkerThread = (worker, { type, data }) => {
|
|
260
|
-
worker.postMessage({ jsenv: true, type, data })
|
|
261
|
-
}
|
|
262
|
-
|
|
263
|
-
const onceWorkerThreadMessage = (workerThread, type, callback) => {
|
|
264
|
-
const onmessage = (message) => {
|
|
265
|
-
if (message && message.jsenv && message.type === type) {
|
|
266
|
-
workerThread.removeListener("message", onmessage)
|
|
267
|
-
// eslint-disable-next-line no-eval
|
|
268
|
-
callback(message.data ? eval(`(${message.data})`) : undefined)
|
|
269
|
-
}
|
|
270
|
-
}
|
|
271
|
-
workerThread.on("message", onmessage)
|
|
272
|
-
return () => {
|
|
273
|
-
workerThread.removeListener("message", onmessage)
|
|
274
|
-
}
|
|
275
|
-
}
|
|
276
|
-
|
|
277
|
-
const onceWorkerThreadEvent = (worker, type, callback) => {
|
|
278
|
-
worker.once(type, callback)
|
|
279
|
-
return () => {
|
|
280
|
-
worker.removeListener(type, callback)
|
|
281
|
-
}
|
|
282
|
-
}
|
|
@@ -1,56 +0,0 @@
|
|
|
1
|
-
/*
|
|
2
|
-
* Calling Profiler.startPreciseCoverage DO NOT propagate to
|
|
3
|
-
* subprocesses (new Worker or child_process.fork())
|
|
4
|
-
* So the best solution remains NODE_V8_COVERAGE
|
|
5
|
-
* This profiler strategy remains useful when:
|
|
6
|
-
* - As fallback when NODE_V8_COVERAGE is not configured
|
|
7
|
-
* - If explicitely enabled with coverageMethodForNodeJs: 'Profiler'
|
|
8
|
-
* - Used by jsenv during automated tests about coverage
|
|
9
|
-
* - Anyone prefering this approach over NODE_V8_COVERAGE and assuming
|
|
10
|
-
* it will not fork subprocess or don't care if coverage is missed for this code
|
|
11
|
-
* - https://v8.dev/blog/javascript-code-coverage#for-embedders
|
|
12
|
-
* - https://github.com/nodejs/node/issues/28283
|
|
13
|
-
* - https://vanilla.aslushnikov.com/?Profiler.startPreciseCoverage
|
|
14
|
-
*/
|
|
15
|
-
|
|
16
|
-
import { Session } from "node:inspector"
|
|
17
|
-
|
|
18
|
-
export const startJsCoverage = async ({
|
|
19
|
-
callCount = true,
|
|
20
|
-
detailed = true,
|
|
21
|
-
} = {}) => {
|
|
22
|
-
const session = new Session()
|
|
23
|
-
session.connect()
|
|
24
|
-
const postSession = (action, options) => {
|
|
25
|
-
const promise = new Promise((resolve, reject) => {
|
|
26
|
-
session.post(action, options, (error, data) => {
|
|
27
|
-
if (error) {
|
|
28
|
-
reject(error)
|
|
29
|
-
} else {
|
|
30
|
-
resolve(data)
|
|
31
|
-
}
|
|
32
|
-
})
|
|
33
|
-
})
|
|
34
|
-
return promise
|
|
35
|
-
}
|
|
36
|
-
|
|
37
|
-
await postSession("Profiler.enable")
|
|
38
|
-
await postSession("Profiler.startPreciseCoverage", { callCount, detailed })
|
|
39
|
-
|
|
40
|
-
const takeJsCoverage = async () => {
|
|
41
|
-
const coverage = await postSession("Profiler.takePreciseCoverage")
|
|
42
|
-
return coverage
|
|
43
|
-
}
|
|
44
|
-
|
|
45
|
-
const stopJsCoverage = async () => {
|
|
46
|
-
const coverage = await takeJsCoverage()
|
|
47
|
-
await postSession("Profiler.stopPreciseCoverage")
|
|
48
|
-
await postSession("Profiler.disable")
|
|
49
|
-
return coverage
|
|
50
|
-
}
|
|
51
|
-
|
|
52
|
-
return {
|
|
53
|
-
takeJsCoverage,
|
|
54
|
-
stopJsCoverage,
|
|
55
|
-
}
|
|
56
|
-
}
|
|
@@ -1,13 +0,0 @@
|
|
|
1
|
-
# runtimes/
|
|
2
|
-
|
|
3
|
-
Code implementing runtimes can be found here.
|
|
4
|
-
|
|
5
|
-
# Description
|
|
6
|
-
|
|
7
|
-
A runtime is an object with a "run" method.
|
|
8
|
-
The run method is roughly doing the following:
|
|
9
|
-
|
|
10
|
-
1. spawn a runtime (browser,Node.js)
|
|
11
|
-
2. set various listeners to monitor file execution
|
|
12
|
-
3. execute the file
|
|
13
|
-
4. return info about file execution (logs and errors for instance)
|
|
@@ -1,74 +0,0 @@
|
|
|
1
|
-
import { assertAndNormalizeDirectoryUrl } from "@jsenv/filesystem"
|
|
2
|
-
|
|
3
|
-
import { pingServer } from "../ping_server.js"
|
|
4
|
-
import { basicFetch } from "../basic_fetch.js"
|
|
5
|
-
|
|
6
|
-
export const assertAndNormalizeWebServer = async (webServer) => {
|
|
7
|
-
if (!webServer) {
|
|
8
|
-
throw new TypeError(
|
|
9
|
-
`webServer is required when running tests on browser(s)`,
|
|
10
|
-
)
|
|
11
|
-
}
|
|
12
|
-
const unexpectedParamNames = Object.keys(webServer).filter((key) => {
|
|
13
|
-
return !["origin", "moduleUrl", "rootDirectoryUrl"].includes(key)
|
|
14
|
-
})
|
|
15
|
-
if (unexpectedParamNames.length > 0) {
|
|
16
|
-
throw new TypeError(
|
|
17
|
-
`${unexpectedParamNames.join(",")}: there is no such param to webServer`,
|
|
18
|
-
)
|
|
19
|
-
}
|
|
20
|
-
|
|
21
|
-
let aServerIsListening = await pingServer(webServer.origin)
|
|
22
|
-
if (!aServerIsListening) {
|
|
23
|
-
if (!webServer.moduleUrl) {
|
|
24
|
-
throw new TypeError(
|
|
25
|
-
`webServer.moduleUrl is required as there is no server listening "${webServer.origin}"`,
|
|
26
|
-
)
|
|
27
|
-
}
|
|
28
|
-
try {
|
|
29
|
-
process.env.IMPORTED_BY_TEST_PLAN = "1"
|
|
30
|
-
await import(webServer.moduleUrl)
|
|
31
|
-
delete process.env.IMPORTED_BY_TEST_PLAN
|
|
32
|
-
} catch (e) {
|
|
33
|
-
if (e.code === "ERR_MODULE_NOT_FOUND") {
|
|
34
|
-
throw new Error(
|
|
35
|
-
`webServer.moduleUrl does not lead to a file at "${webServer.moduleUrl}"`,
|
|
36
|
-
)
|
|
37
|
-
}
|
|
38
|
-
throw e
|
|
39
|
-
}
|
|
40
|
-
aServerIsListening = await pingServer(webServer.origin)
|
|
41
|
-
if (!aServerIsListening) {
|
|
42
|
-
throw new Error(
|
|
43
|
-
`webServer.moduleUrl did not start a server listening at "${webServer.origin}", check file at "${webServer.moduleUrl}"`,
|
|
44
|
-
)
|
|
45
|
-
}
|
|
46
|
-
}
|
|
47
|
-
const { headers } = await basicFetch(webServer.origin, {
|
|
48
|
-
method: "GET",
|
|
49
|
-
rejectUnauthorized: false,
|
|
50
|
-
headers: {
|
|
51
|
-
"x-server-inspect": "1",
|
|
52
|
-
},
|
|
53
|
-
})
|
|
54
|
-
if (String(headers["server"]).includes("jsenv_dev_server")) {
|
|
55
|
-
webServer.isJsenvDevServer = true
|
|
56
|
-
const { json } = await basicFetch(`${webServer.origin}/__params__.json`, {
|
|
57
|
-
rejectUnauthorized: false,
|
|
58
|
-
})
|
|
59
|
-
if (webServer.rootDirectoryUrl === undefined) {
|
|
60
|
-
const jsenvDevServerParams = await json()
|
|
61
|
-
webServer.rootDirectoryUrl = jsenvDevServerParams.sourceDirectoryUrl
|
|
62
|
-
} else {
|
|
63
|
-
webServer.rootDirectoryUrl = assertAndNormalizeDirectoryUrl(
|
|
64
|
-
webServer.rootDirectoryUrl,
|
|
65
|
-
"webServer.rootDirectoryUrl",
|
|
66
|
-
)
|
|
67
|
-
}
|
|
68
|
-
} else {
|
|
69
|
-
webServer.rootDirectoryUrl = assertAndNormalizeDirectoryUrl(
|
|
70
|
-
webServer.rootDirectoryUrl,
|
|
71
|
-
"webServer.rootDirectoryUrl",
|
|
72
|
-
)
|
|
73
|
-
}
|
|
74
|
-
}
|
|
@@ -1,48 +0,0 @@
|
|
|
1
|
-
import { requireFromJsenv } from "@jsenv/core/src/require_from_jsenv.js"
|
|
2
|
-
|
|
3
|
-
// https://github.com/istanbuljs/babel-plugin-istanbul/blob/321740f7b25d803f881466ea819d870f7ed6a254/src/index.js
|
|
4
|
-
|
|
5
|
-
export const babelPluginInstrument = (api, { useInlineSourceMaps = false }) => {
|
|
6
|
-
const { programVisitor } = requireFromJsenv("istanbul-lib-instrument")
|
|
7
|
-
const { types } = api
|
|
8
|
-
|
|
9
|
-
return {
|
|
10
|
-
name: "transform-instrument",
|
|
11
|
-
visitor: {
|
|
12
|
-
Program: {
|
|
13
|
-
enter(path) {
|
|
14
|
-
const { file } = this
|
|
15
|
-
const { opts } = file
|
|
16
|
-
let inputSourceMap
|
|
17
|
-
if (useInlineSourceMaps) {
|
|
18
|
-
// https://github.com/istanbuljs/babel-plugin-istanbul/commit/a9e15643d249a2985e4387e4308022053b2cd0ad#diff-1fdf421c05c1140f6d71444ea2b27638R65
|
|
19
|
-
inputSourceMap =
|
|
20
|
-
opts.inputSourceMap || file.inputMap
|
|
21
|
-
? file.inputMap.sourcemap
|
|
22
|
-
: null
|
|
23
|
-
} else {
|
|
24
|
-
inputSourceMap = opts.inputSourceMap
|
|
25
|
-
}
|
|
26
|
-
this.__dv__ = programVisitor(
|
|
27
|
-
types,
|
|
28
|
-
opts.filenameRelative || opts.filename,
|
|
29
|
-
{
|
|
30
|
-
coverageVariable: "__coverage__",
|
|
31
|
-
inputSourceMap,
|
|
32
|
-
},
|
|
33
|
-
)
|
|
34
|
-
this.__dv__.enter(path)
|
|
35
|
-
},
|
|
36
|
-
|
|
37
|
-
exit(path) {
|
|
38
|
-
if (!this.__dv__) {
|
|
39
|
-
return
|
|
40
|
-
}
|
|
41
|
-
const object = this.__dv__.exit(path)
|
|
42
|
-
// object got two properties: fileCoverage and sourceMappingURL
|
|
43
|
-
this.file.metadata.coverage = object.fileCoverage
|
|
44
|
-
},
|
|
45
|
-
},
|
|
46
|
-
},
|
|
47
|
-
}
|
|
48
|
-
}
|
|
@@ -1,32 +0,0 @@
|
|
|
1
|
-
import { fileURLToPath } from "node:url"
|
|
2
|
-
import { readFileSync } from "node:fs"
|
|
3
|
-
|
|
4
|
-
import { requireFromJsenv } from "@jsenv/core/src/require_from_jsenv.js"
|
|
5
|
-
import { istanbulCoverageMapFromCoverage } from "./istanbul_coverage_map_from_coverage.js"
|
|
6
|
-
|
|
7
|
-
export const generateCoverageHtmlDirectory = async (
|
|
8
|
-
coverage,
|
|
9
|
-
{
|
|
10
|
-
rootDirectoryUrl,
|
|
11
|
-
coverageHtmlDirectoryRelativeUrl,
|
|
12
|
-
coverageReportSkipEmpty,
|
|
13
|
-
coverageReportSkipFull,
|
|
14
|
-
},
|
|
15
|
-
) => {
|
|
16
|
-
const libReport = requireFromJsenv("istanbul-lib-report")
|
|
17
|
-
const reports = requireFromJsenv("istanbul-reports")
|
|
18
|
-
|
|
19
|
-
const context = libReport.createContext({
|
|
20
|
-
dir: fileURLToPath(rootDirectoryUrl),
|
|
21
|
-
coverageMap: istanbulCoverageMapFromCoverage(coverage),
|
|
22
|
-
sourceFinder: (path) =>
|
|
23
|
-
readFileSync(new URL(path, rootDirectoryUrl), "utf8"),
|
|
24
|
-
})
|
|
25
|
-
|
|
26
|
-
const report = reports.create("html", {
|
|
27
|
-
skipEmpty: coverageReportSkipEmpty,
|
|
28
|
-
skipFull: coverageReportSkipFull,
|
|
29
|
-
subdir: coverageHtmlDirectoryRelativeUrl,
|
|
30
|
-
})
|
|
31
|
-
report.execute(context)
|
|
32
|
-
}
|
|
@@ -1,17 +0,0 @@
|
|
|
1
|
-
import { writeFile } from "@jsenv/filesystem"
|
|
2
|
-
import { urlToFileSystemPath } from "@jsenv/urls"
|
|
3
|
-
import { byteAsFileSize } from "@jsenv/log"
|
|
4
|
-
|
|
5
|
-
export const generateCoverageJsonFile = async ({
|
|
6
|
-
coverage,
|
|
7
|
-
coverageJsonFileUrl,
|
|
8
|
-
logger,
|
|
9
|
-
}) => {
|
|
10
|
-
const coverageAsText = JSON.stringify(coverage, null, " ")
|
|
11
|
-
logger.info(
|
|
12
|
-
`-> ${urlToFileSystemPath(coverageJsonFileUrl)} (${byteAsFileSize(
|
|
13
|
-
Buffer.byteLength(coverageAsText),
|
|
14
|
-
)})`,
|
|
15
|
-
)
|
|
16
|
-
await writeFile(coverageJsonFileUrl, coverageAsText)
|
|
17
|
-
}
|
|
@@ -1,19 +0,0 @@
|
|
|
1
|
-
import { requireFromJsenv } from "@jsenv/core/src/require_from_jsenv.js"
|
|
2
|
-
import { istanbulCoverageMapFromCoverage } from "./istanbul_coverage_map_from_coverage.js"
|
|
3
|
-
|
|
4
|
-
export const generateCoverageTextLog = (
|
|
5
|
-
coverage,
|
|
6
|
-
{ coverageReportSkipEmpty, coverageReportSkipFull },
|
|
7
|
-
) => {
|
|
8
|
-
const libReport = requireFromJsenv("istanbul-lib-report")
|
|
9
|
-
const reports = requireFromJsenv("istanbul-reports")
|
|
10
|
-
|
|
11
|
-
const context = libReport.createContext({
|
|
12
|
-
coverageMap: istanbulCoverageMapFromCoverage(coverage),
|
|
13
|
-
})
|
|
14
|
-
const report = reports.create("text", {
|
|
15
|
-
skipEmpty: coverageReportSkipEmpty,
|
|
16
|
-
skipFull: coverageReportSkipFull,
|
|
17
|
-
})
|
|
18
|
-
report.execute(context)
|
|
19
|
-
}
|
|
@@ -1,52 +0,0 @@
|
|
|
1
|
-
import { readFile } from "@jsenv/filesystem"
|
|
2
|
-
import { resolveUrl } from "@jsenv/urls"
|
|
3
|
-
import { Abort } from "@jsenv/abort"
|
|
4
|
-
import { applyBabelPlugins } from "@jsenv/ast"
|
|
5
|
-
|
|
6
|
-
import { requireFromJsenv } from "@jsenv/core/src/require_from_jsenv.js"
|
|
7
|
-
import { babelPluginInstrument } from "./babel_plugin_instrument.js"
|
|
8
|
-
|
|
9
|
-
export const relativeUrlToEmptyCoverage = async (
|
|
10
|
-
relativeUrl,
|
|
11
|
-
{ signal, rootDirectoryUrl },
|
|
12
|
-
) => {
|
|
13
|
-
const operation = Abort.startOperation()
|
|
14
|
-
operation.addAbortSignal(signal)
|
|
15
|
-
|
|
16
|
-
try {
|
|
17
|
-
const fileUrl = resolveUrl(relativeUrl, rootDirectoryUrl)
|
|
18
|
-
const content = await readFile(fileUrl, { as: "string" })
|
|
19
|
-
|
|
20
|
-
operation.throwIfAborted()
|
|
21
|
-
const { metadata } = await applyBabelPlugins({
|
|
22
|
-
babelPlugins: [babelPluginInstrument],
|
|
23
|
-
urlInfo: {
|
|
24
|
-
originalUrl: fileUrl,
|
|
25
|
-
content,
|
|
26
|
-
},
|
|
27
|
-
})
|
|
28
|
-
const { coverage } = metadata
|
|
29
|
-
if (!coverage) {
|
|
30
|
-
throw new Error(`missing coverage for file`)
|
|
31
|
-
}
|
|
32
|
-
// https://github.com/gotwarlost/istanbul/blob/bc84c315271a5dd4d39bcefc5925cfb61a3d174a/lib/command/common/run-with-cover.js#L229
|
|
33
|
-
Object.keys(coverage.s).forEach(function (key) {
|
|
34
|
-
coverage.s[key] = 0
|
|
35
|
-
})
|
|
36
|
-
return coverage
|
|
37
|
-
} catch (e) {
|
|
38
|
-
if (e && e.code === "PARSE_ERROR") {
|
|
39
|
-
// return an empty coverage for that file when
|
|
40
|
-
// it contains a syntax error
|
|
41
|
-
return createEmptyCoverage(relativeUrl)
|
|
42
|
-
}
|
|
43
|
-
throw e
|
|
44
|
-
} finally {
|
|
45
|
-
await operation.end()
|
|
46
|
-
}
|
|
47
|
-
}
|
|
48
|
-
|
|
49
|
-
const createEmptyCoverage = (relativeUrl) => {
|
|
50
|
-
const { createFileCoverage } = requireFromJsenv("istanbul-lib-coverage")
|
|
51
|
-
return createFileCoverage(relativeUrl).toJSON()
|
|
52
|
-
}
|
|
@@ -1,25 +0,0 @@
|
|
|
1
|
-
import {
|
|
2
|
-
urlToRelativeUrl,
|
|
3
|
-
fileSystemPathToUrl,
|
|
4
|
-
isFileSystemPath,
|
|
5
|
-
} from "@jsenv/urls"
|
|
6
|
-
|
|
7
|
-
export const normalizeFileByFileCoveragePaths = (
|
|
8
|
-
fileByFileCoverage,
|
|
9
|
-
rootDirectoryUrl,
|
|
10
|
-
) => {
|
|
11
|
-
const fileByFileNormalized = {}
|
|
12
|
-
Object.keys(fileByFileCoverage).forEach((key) => {
|
|
13
|
-
const fileCoverage = fileByFileCoverage[key]
|
|
14
|
-
const { path } = fileCoverage
|
|
15
|
-
const url = isFileSystemPath(path)
|
|
16
|
-
? fileSystemPathToUrl(path)
|
|
17
|
-
: new URL(path, rootDirectoryUrl).href
|
|
18
|
-
const relativeUrl = urlToRelativeUrl(url, rootDirectoryUrl)
|
|
19
|
-
fileByFileNormalized[`./${relativeUrl}`] = {
|
|
20
|
-
...fileCoverage,
|
|
21
|
-
path: `./${relativeUrl}`,
|
|
22
|
-
}
|
|
23
|
-
})
|
|
24
|
-
return fileByFileNormalized
|
|
25
|
-
}
|
|
@@ -1,28 +0,0 @@
|
|
|
1
|
-
import { requireFromJsenv } from "@jsenv/core/src/require_from_jsenv.js"
|
|
2
|
-
|
|
3
|
-
export const composeTwoFileByFileIstanbulCoverages = (
|
|
4
|
-
firstFileByFileIstanbulCoverage,
|
|
5
|
-
secondFileByFileIstanbulCoverage,
|
|
6
|
-
) => {
|
|
7
|
-
const fileByFileIstanbulCoverage = {}
|
|
8
|
-
Object.keys(firstFileByFileIstanbulCoverage).forEach((key) => {
|
|
9
|
-
fileByFileIstanbulCoverage[key] = firstFileByFileIstanbulCoverage[key]
|
|
10
|
-
})
|
|
11
|
-
Object.keys(secondFileByFileIstanbulCoverage).forEach((key) => {
|
|
12
|
-
const firstCoverage = firstFileByFileIstanbulCoverage[key]
|
|
13
|
-
const secondCoverage = secondFileByFileIstanbulCoverage[key]
|
|
14
|
-
fileByFileIstanbulCoverage[key] = firstCoverage
|
|
15
|
-
? merge(firstCoverage, secondCoverage)
|
|
16
|
-
: secondCoverage
|
|
17
|
-
})
|
|
18
|
-
|
|
19
|
-
return fileByFileIstanbulCoverage
|
|
20
|
-
}
|
|
21
|
-
|
|
22
|
-
const merge = (firstIstanbulCoverage, secondIstanbulCoverage) => {
|
|
23
|
-
const { createFileCoverage } = requireFromJsenv("istanbul-lib-coverage")
|
|
24
|
-
const istanbulFileCoverageObject = createFileCoverage(firstIstanbulCoverage)
|
|
25
|
-
istanbulFileCoverageObject.merge(secondIstanbulCoverage)
|
|
26
|
-
const istanbulCoverage = istanbulFileCoverageObject.toJSON()
|
|
27
|
-
return istanbulCoverage
|
|
28
|
-
}
|
|
@@ -1,16 +0,0 @@
|
|
|
1
|
-
import { requireFromJsenv } from "@jsenv/core/src/require_from_jsenv.js"
|
|
2
|
-
|
|
3
|
-
export const istanbulCoverageMapFromCoverage = (coverage) => {
|
|
4
|
-
const { createCoverageMap } = requireFromJsenv("istanbul-lib-coverage")
|
|
5
|
-
|
|
6
|
-
const coverageAdjusted = {}
|
|
7
|
-
Object.keys(coverage).forEach((key) => {
|
|
8
|
-
coverageAdjusted[key.slice(2)] = {
|
|
9
|
-
...coverage[key],
|
|
10
|
-
path: key.slice(2),
|
|
11
|
-
}
|
|
12
|
-
})
|
|
13
|
-
|
|
14
|
-
const coverageMap = createCoverageMap(coverageAdjusted)
|
|
15
|
-
return coverageMap
|
|
16
|
-
}
|
|
@@ -1,15 +0,0 @@
|
|
|
1
|
-
import { collectFiles } from "@jsenv/filesystem"
|
|
2
|
-
|
|
3
|
-
export const listRelativeFileUrlToCover = async ({
|
|
4
|
-
signal,
|
|
5
|
-
rootDirectoryUrl,
|
|
6
|
-
coverageConfig,
|
|
7
|
-
}) => {
|
|
8
|
-
const matchingFileResultArray = await collectFiles({
|
|
9
|
-
signal,
|
|
10
|
-
directoryUrl: rootDirectoryUrl,
|
|
11
|
-
associations: { cover: coverageConfig },
|
|
12
|
-
predicate: ({ cover }) => cover,
|
|
13
|
-
})
|
|
14
|
-
return matchingFileResultArray.map(({ relativeUrl }) => relativeUrl)
|
|
15
|
-
}
|