@jsenv/core 34.3.0 → 35.0.1

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (74) hide show
  1. package/README.md +2 -9
  2. package/dist/js/ws.js +1 -145
  3. package/dist/{jsenv.js → jsenv_core.js} +1056 -3850
  4. package/package.json +8 -23
  5. package/src/build/build.js +2 -2
  6. package/src/dev/file_service.js +8 -8
  7. package/src/dev/start_dev_server.js +3 -3
  8. package/src/dev/user_agent.js +1 -1
  9. package/src/main.js +0 -23
  10. package/src/plugins/supervisor/jsenv_plugin_supervisor.js +1 -1
  11. package/src/plugins/transpilation/babel/require_babel_plugin.js +1 -1
  12. package/src/plugins/transpilation/js_module_fallback/convert_js_module_to_js_classic.js +1 -1
  13. package/dist/controllable_child_process.mjs +0 -129
  14. package/dist/controllable_worker_thread.mjs +0 -91
  15. package/dist/importmap_node_loader.mjs +0 -49
  16. package/dist/js/execute_using_dynamic_import.js +0 -850
  17. package/dist/js/resolveImport.js +0 -504
  18. package/dist/js/v8_coverage.js +0 -508
  19. package/dist/no_experimental_warnings.cjs +0 -8
  20. package/src/execute/execute.js +0 -111
  21. package/src/execute/run.js +0 -161
  22. package/src/execute/runtimes/browsers/chromium.js +0 -10
  23. package/src/execute/runtimes/browsers/firefox.js +0 -9
  24. package/src/execute/runtimes/browsers/from_playwright.js +0 -574
  25. package/src/execute/runtimes/browsers/middleware_istanbul.js +0 -65
  26. package/src/execute/runtimes/browsers/middleware_js_supervisor.js +0 -100
  27. package/src/execute/runtimes/browsers/webkit.js +0 -26
  28. package/src/execute/runtimes/node/child_exec_options.js +0 -166
  29. package/src/execute/runtimes/node/controllable_child_process.mjs +0 -135
  30. package/src/execute/runtimes/node/controllable_worker_thread.mjs +0 -103
  31. package/src/execute/runtimes/node/exec_options.js +0 -57
  32. package/src/execute/runtimes/node/execute_using_dynamic_import.js +0 -55
  33. package/src/execute/runtimes/node/exit_codes.js +0 -9
  34. package/src/execute/runtimes/node/importmap_node_loader.mjs +0 -51
  35. package/src/execute/runtimes/node/importmap_node_loader_file_url.js +0 -4
  36. package/src/execute/runtimes/node/kill_process_tree.js +0 -76
  37. package/src/execute/runtimes/node/no_experimental_warnings.cjs +0 -12
  38. package/src/execute/runtimes/node/no_experimental_warnings_file_url.js +0 -4
  39. package/src/execute/runtimes/node/node_child_process.js +0 -363
  40. package/src/execute/runtimes/node/node_execution_performance.js +0 -67
  41. package/src/execute/runtimes/node/node_worker_thread.js +0 -295
  42. package/src/execute/runtimes/node/profiler_v8_coverage.js +0 -56
  43. package/src/execute/runtimes/readme.md +0 -13
  44. package/src/execute/web_server_param.js +0 -74
  45. package/src/test/coverage/babel_plugin_instrument.js +0 -48
  46. package/src/test/coverage/coverage_reporter_html_directory.js +0 -32
  47. package/src/test/coverage/coverage_reporter_json_file.js +0 -17
  48. package/src/test/coverage/coverage_reporter_text_log.js +0 -19
  49. package/src/test/coverage/empty_coverage_factory.js +0 -52
  50. package/src/test/coverage/file_by_file_coverage.js +0 -25
  51. package/src/test/coverage/istanbul_coverage_composition.js +0 -28
  52. package/src/test/coverage/istanbul_coverage_map_from_coverage.js +0 -16
  53. package/src/test/coverage/list_files_not_covered.js +0 -15
  54. package/src/test/coverage/missing_coverage.js +0 -41
  55. package/src/test/coverage/report_to_coverage.js +0 -198
  56. package/src/test/coverage/v8_and_istanbul.js +0 -37
  57. package/src/test/coverage/v8_coverage.js +0 -26
  58. package/src/test/coverage/v8_coverage_composition.js +0 -24
  59. package/src/test/coverage/v8_coverage_node_directory.js +0 -85
  60. package/src/test/coverage/v8_coverage_to_istanbul.js +0 -99
  61. package/src/test/execute_steps.js +0 -425
  62. package/src/test/execute_test_plan.js +0 -372
  63. package/src/test/execution_colors.js +0 -10
  64. package/src/test/execution_steps.js +0 -65
  65. package/src/test/gc.js +0 -9
  66. package/src/test/logs_file_execution.js +0 -427
  67. package/src/test/logs_file_execution.test.mjs +0 -41
  68. package/src/test/readme.md +0 -3
  69. /package/src/{basic_fetch.js → helpers/basic_fetch.js} +0 -0
  70. /package/src/{lookup_package_directory.js → helpers/lookup_package_directory.js} +0 -0
  71. /package/src/{ping_server.js → helpers/ping_server.js} +0 -0
  72. /package/src/{require_from_jsenv.js → helpers/require_from_jsenv.js} +0 -0
  73. /package/src/{watch_source_files.js → helpers/watch_source_files.js} +0 -0
  74. /package/src/{web_url_converter.js → helpers/web_url_converter.js} +0 -0
@@ -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
- }
@@ -1,41 +0,0 @@
1
- import { Abort } from "@jsenv/abort"
2
-
3
- import { listRelativeFileUrlToCover } from "./list_files_not_covered.js"
4
- import { relativeUrlToEmptyCoverage } from "./empty_coverage_factory.js"
5
-
6
- export const getMissingFileByFileCoverage = async ({
7
- signal,
8
- rootDirectoryUrl,
9
- coverageConfig,
10
- fileByFileCoverage,
11
- }) => {
12
- const relativeUrlsToCover = await listRelativeFileUrlToCover({
13
- signal,
14
- rootDirectoryUrl,
15
- coverageConfig,
16
- })
17
- const relativeUrlsMissing = relativeUrlsToCover.filter((relativeUrlToCover) =>
18
- Object.keys(fileByFileCoverage).every((key) => {
19
- return key !== `./${relativeUrlToCover}`
20
- }),
21
- )
22
-
23
- const operation = Abort.startOperation()
24
- operation.addAbortSignal(signal)
25
- const missingFileByFileCoverage = {}
26
- await relativeUrlsMissing.reduce(async (previous, relativeUrlMissing) => {
27
- operation.throwIfAborted()
28
- await previous
29
- await operation.withSignal(async (signal) => {
30
- const emptyCoverage = await relativeUrlToEmptyCoverage(
31
- relativeUrlMissing,
32
- {
33
- signal,
34
- rootDirectoryUrl,
35
- },
36
- )
37
- missingFileByFileCoverage[`./${relativeUrlMissing}`] = emptyCoverage
38
- })
39
- }, Promise.resolve())
40
- return missingFileByFileCoverage
41
- }
@@ -1,198 +0,0 @@
1
- import { readFileSync } from "node:fs"
2
- import { Abort } from "@jsenv/abort"
3
-
4
- import { filterV8Coverage } from "./v8_coverage.js"
5
- import { readNodeV8CoverageDirectory } from "./v8_coverage_node_directory.js"
6
- import { composeTwoV8Coverages } from "./v8_coverage_composition.js"
7
- import { composeTwoFileByFileIstanbulCoverages } from "./istanbul_coverage_composition.js"
8
- import { v8CoverageToIstanbul } from "./v8_coverage_to_istanbul.js"
9
- import { composeV8AndIstanbul } from "./v8_and_istanbul.js"
10
- import { normalizeFileByFileCoveragePaths } from "./file_by_file_coverage.js"
11
- import { getMissingFileByFileCoverage } from "./missing_coverage.js"
12
-
13
- export const reportToCoverage = async (
14
- report,
15
- {
16
- signal,
17
- logger,
18
- rootDirectoryUrl,
19
- coverageConfig,
20
- coverageIncludeMissing,
21
- coverageMethodForNodeJs,
22
- coverageV8ConflictWarning,
23
- },
24
- ) => {
25
- // collect v8 and istanbul coverage from executions
26
- let { v8Coverage, fileByFileIstanbulCoverage } = await getCoverageFromReport({
27
- signal,
28
- report,
29
- onMissing: ({ file, executionResult, executionName }) => {
30
- // several reasons not to have coverage here:
31
- // 1. the file we executed did not import an instrumented file.
32
- // - a test file without import
33
- // - a test file importing only file excluded from coverage
34
- // - a coverDescription badly configured so that we don't realize
35
- // a file should be covered
36
-
37
- // 2. the file we wanted to executed timedout
38
- // - infinite loop
39
- // - too extensive operation
40
- // - a badly configured or too low allocatedMs for that execution.
41
-
42
- // 3. the file we wanted to execute contains syntax-error
43
-
44
- // in any scenario we are fine because
45
- // coverDescription will generate empty coverage for files
46
- // that were suppose to be coverage but were not.
47
- if (
48
- executionResult.status === "completed" &&
49
- executionResult.type === "node" &&
50
- coverageMethodForNodeJs !== "NODE_V8_COVERAGE"
51
- ) {
52
- logger.warn(
53
- `"${executionName}" execution of ${file} did not properly write coverage into ${executionResult.coverageFileUrl}`,
54
- )
55
- }
56
- },
57
- })
58
-
59
- if (coverageMethodForNodeJs === "NODE_V8_COVERAGE") {
60
- await readNodeV8CoverageDirectory({
61
- logger,
62
- signal,
63
- onV8Coverage: async (nodeV8Coverage) => {
64
- const nodeV8CoverageLight = await filterV8Coverage(nodeV8Coverage, {
65
- rootDirectoryUrl,
66
- coverageConfig,
67
- })
68
- v8Coverage = v8Coverage
69
- ? composeTwoV8Coverages(v8Coverage, nodeV8CoverageLight)
70
- : nodeV8CoverageLight
71
- },
72
- })
73
- }
74
-
75
- // try to merge v8 with istanbul, if any
76
- let fileByFileCoverage
77
- if (v8Coverage) {
78
- let v8FileByFileCoverage = await v8CoverageToIstanbul(v8Coverage, {
79
- signal,
80
- })
81
-
82
- v8FileByFileCoverage = normalizeFileByFileCoveragePaths(
83
- v8FileByFileCoverage,
84
- rootDirectoryUrl,
85
- )
86
-
87
- if (fileByFileIstanbulCoverage) {
88
- fileByFileIstanbulCoverage = normalizeFileByFileCoveragePaths(
89
- fileByFileIstanbulCoverage,
90
- rootDirectoryUrl,
91
- )
92
- fileByFileCoverage = composeV8AndIstanbul(
93
- v8FileByFileCoverage,
94
- fileByFileIstanbulCoverage,
95
- { coverageV8ConflictWarning },
96
- )
97
- } else {
98
- fileByFileCoverage = v8FileByFileCoverage
99
- }
100
- }
101
- // get istanbul only
102
- else if (fileByFileIstanbulCoverage) {
103
- fileByFileCoverage = normalizeFileByFileCoveragePaths(
104
- fileByFileIstanbulCoverage,
105
- rootDirectoryUrl,
106
- )
107
- }
108
- // no coverage found in execution (or zero file where executed)
109
- else {
110
- fileByFileCoverage = {}
111
- }
112
-
113
- // now add coverage for file not covered
114
- if (coverageIncludeMissing) {
115
- const missingFileByFileCoverage = await getMissingFileByFileCoverage({
116
- signal,
117
- rootDirectoryUrl,
118
- coverageConfig,
119
- fileByFileCoverage,
120
- })
121
- Object.assign(
122
- fileByFileCoverage,
123
- normalizeFileByFileCoveragePaths(
124
- missingFileByFileCoverage,
125
- rootDirectoryUrl,
126
- ),
127
- )
128
- }
129
-
130
- return fileByFileCoverage
131
- }
132
-
133
- const getCoverageFromReport = async ({ signal, report, onMissing }) => {
134
- const operation = Abort.startOperation()
135
- operation.addAbortSignal(signal)
136
-
137
- try {
138
- let v8Coverage
139
- let fileByFileIstanbulCoverage
140
-
141
- // collect v8 and istanbul coverage from executions
142
- await Object.keys(report).reduce(async (previous, file) => {
143
- operation.throwIfAborted()
144
- await previous
145
-
146
- const executionResultForFile = report[file]
147
- await Object.keys(executionResultForFile).reduce(
148
- async (previous, executionName) => {
149
- operation.throwIfAborted()
150
- await previous
151
-
152
- const executionResultForFileOnRuntime =
153
- executionResultForFile[executionName]
154
- const { coverageFileUrl } = executionResultForFileOnRuntime
155
- let executionCoverage
156
- try {
157
- executionCoverage = JSON.parse(
158
- String(readFileSync(new URL(coverageFileUrl))),
159
- )
160
- } catch (e) {
161
- if (e.code === "ENOENT" || e.name === "SyntaxError") {
162
- onMissing({
163
- executionName,
164
- file,
165
- executionResult: executionResultForFileOnRuntime,
166
- })
167
- return
168
- }
169
- throw e
170
- }
171
-
172
- if (isV8Coverage(executionCoverage)) {
173
- v8Coverage = v8Coverage
174
- ? composeTwoV8Coverages(v8Coverage, executionCoverage)
175
- : executionCoverage
176
- } else {
177
- fileByFileIstanbulCoverage = fileByFileIstanbulCoverage
178
- ? composeTwoFileByFileIstanbulCoverages(
179
- fileByFileIstanbulCoverage,
180
- executionCoverage,
181
- )
182
- : executionCoverage
183
- }
184
- },
185
- Promise.resolve(),
186
- )
187
- }, Promise.resolve())
188
-
189
- return {
190
- v8Coverage,
191
- fileByFileIstanbulCoverage,
192
- }
193
- } finally {
194
- await operation.end()
195
- }
196
- }
197
-
198
- const isV8Coverage = (coverage) => Boolean(coverage.result)
@@ -1,37 +0,0 @@
1
- import { createDetailedMessage } from "@jsenv/log"
2
-
3
- export const composeV8AndIstanbul = (
4
- v8FileByFileCoverage,
5
- istanbulFileByFileCoverage,
6
- { coverageV8ConflictWarning },
7
- ) => {
8
- const fileByFileCoverage = {}
9
- const v8Files = Object.keys(v8FileByFileCoverage)
10
- const istanbulFiles = Object.keys(istanbulFileByFileCoverage)
11
-
12
- v8Files.forEach((key) => {
13
- fileByFileCoverage[key] = v8FileByFileCoverage[key]
14
- })
15
- istanbulFiles.forEach((key) => {
16
- const v8Coverage = v8FileByFileCoverage[key]
17
- if (v8Coverage) {
18
- if (coverageV8ConflictWarning) {
19
- console.warn(
20
- createDetailedMessage(
21
- `Coverage conflict on "${key}", found two coverage that cannot be merged together: v8 and istanbul. The istanbul coverage will be ignored.`,
22
- {
23
- details: `This happens when a file is executed on a runtime using v8 coverage (node or chromium) and on runtime using istanbul coverage (firefox or webkit)`,
24
- suggestion:
25
- "You can disable this warning with coverageV8ConflictWarning: false",
26
- },
27
- ),
28
- )
29
- }
30
- fileByFileCoverage[key] = v8Coverage
31
- } else {
32
- fileByFileCoverage[key] = istanbulFileByFileCoverage[key]
33
- }
34
- })
35
-
36
- return fileByFileCoverage
37
- }
@@ -1,26 +0,0 @@
1
- import { URL_META } from "@jsenv/url-meta"
2
-
3
- export const filterV8Coverage = async (
4
- v8Coverage,
5
- { rootDirectoryUrl, coverageConfig },
6
- ) => {
7
- const associations = URL_META.resolveAssociations(
8
- { cover: coverageConfig },
9
- rootDirectoryUrl,
10
- )
11
- const urlShouldBeCovered = (url) => {
12
- const { cover } = URL_META.applyAssociations({
13
- url: new URL(url, rootDirectoryUrl).href,
14
- associations,
15
- })
16
- return cover
17
- }
18
-
19
- const v8CoverageFiltered = {
20
- ...v8Coverage,
21
- result: v8Coverage.result.filter((fileReport) =>
22
- urlShouldBeCovered(fileReport.url),
23
- ),
24
- }
25
- return v8CoverageFiltered
26
- }
@@ -1,24 +0,0 @@
1
- import { requireFromJsenv } from "@jsenv/core/src/require_from_jsenv.js"
2
-
3
- export const composeTwoV8Coverages = (firstV8Coverage, secondV8Coverage) => {
4
- if (secondV8Coverage.result.length === 0) {
5
- return firstV8Coverage
6
- }
7
-
8
- // eslint-disable-next-line import/no-unresolved
9
- const { mergeProcessCovs } = requireFromJsenv("@c88/v8-coverage")
10
- // "mergeProcessCovs" do not preserves source-map-cache during the merge
11
- // so we store sourcemap cache now
12
- const sourceMapCache = {}
13
- const visit = (coverageReport) => {
14
- if (coverageReport["source-map-cache"]) {
15
- Object.assign(sourceMapCache, coverageReport["source-map-cache"])
16
- }
17
- }
18
- visit(firstV8Coverage)
19
- visit(secondV8Coverage)
20
- const v8Coverage = mergeProcessCovs([firstV8Coverage, secondV8Coverage])
21
- v8Coverage["source-map-cache"] = sourceMapCache
22
-
23
- return v8Coverage
24
- }
@@ -1,85 +0,0 @@
1
- import { readFileSync, readdirSync } from "node:fs"
2
- import { assertAndNormalizeDirectoryUrl } from "@jsenv/filesystem"
3
- import { createDetailedMessage } from "@jsenv/log"
4
- import { Abort } from "@jsenv/abort"
5
-
6
- export const readNodeV8CoverageDirectory = async ({
7
- logger,
8
- signal,
9
- onV8Coverage,
10
- maxMsWaitingForNodeToWriteCoverageFile = 2000,
11
- }) => {
12
- const NODE_V8_COVERAGE = process.env.NODE_V8_COVERAGE
13
- const operation = Abort.startOperation()
14
- operation.addAbortSignal(signal)
15
-
16
- let timeSpentTrying = 0
17
- const tryReadDirectory = async () => {
18
- const dirContent = readdirSync(NODE_V8_COVERAGE)
19
- if (dirContent.length > 0) {
20
- return dirContent
21
- }
22
- if (timeSpentTrying < maxMsWaitingForNodeToWriteCoverageFile) {
23
- await new Promise((resolve) => setTimeout(resolve, 200))
24
- timeSpentTrying += 200
25
- logger.debug("retry to read coverage directory")
26
- return tryReadDirectory()
27
- }
28
- logger.warn(`v8 coverage directory is empty at ${NODE_V8_COVERAGE}`)
29
- return dirContent
30
- }
31
-
32
- try {
33
- operation.throwIfAborted()
34
- const dirContent = await tryReadDirectory()
35
-
36
- const coverageDirectoryUrl = assertAndNormalizeDirectoryUrl(
37
- NODE_V8_COVERAGE,
38
- "NODE_V8_COVERAGE",
39
- )
40
-
41
- await dirContent.reduce(async (previous, dirEntry) => {
42
- operation.throwIfAborted()
43
- await previous
44
-
45
- const dirEntryUrl = new URL(dirEntry, coverageDirectoryUrl)
46
- const tryReadJsonFile = async () => {
47
- const fileContent = String(readFileSync(dirEntryUrl))
48
- if (fileContent === "") {
49
- if (timeSpentTrying < maxMsWaitingForNodeToWriteCoverageFile) {
50
- await new Promise((resolve) => setTimeout(resolve, 200))
51
- timeSpentTrying += 200
52
- return tryReadJsonFile()
53
- }
54
- console.warn(`Coverage JSON file is empty at ${dirEntryUrl}`)
55
- return null
56
- }
57
-
58
- try {
59
- const fileAsJson = JSON.parse(fileContent)
60
- return fileAsJson
61
- } catch (e) {
62
- if (timeSpentTrying < maxMsWaitingForNodeToWriteCoverageFile) {
63
- await new Promise((resolve) => setTimeout(resolve, 200))
64
- timeSpentTrying += 200
65
- return tryReadJsonFile()
66
- }
67
- console.warn(
68
- createDetailedMessage(`Error while reading coverage file`, {
69
- "error stack": e.stack,
70
- "file": dirEntryUrl,
71
- }),
72
- )
73
- return null
74
- }
75
- }
76
-
77
- const fileContent = await tryReadJsonFile()
78
- if (fileContent) {
79
- await onV8Coverage(fileContent)
80
- }
81
- }, Promise.resolve())
82
- } finally {
83
- await operation.end()
84
- }
85
- }
@@ -1,99 +0,0 @@
1
- import { urlToFileSystemPath } from "@jsenv/urls"
2
- import { Abort } from "@jsenv/abort"
3
-
4
- import { requireFromJsenv } from "@jsenv/core/src/require_from_jsenv.js"
5
- import { composeTwoFileByFileIstanbulCoverages } from "./istanbul_coverage_composition.js"
6
-
7
- export const v8CoverageToIstanbul = async (v8Coverage, { signal }) => {
8
- const operation = Abort.startOperation()
9
- operation.addAbortSignal(signal)
10
-
11
- try {
12
- const v8ToIstanbul = requireFromJsenv("v8-to-istanbul")
13
- const sourcemapCache = v8Coverage["source-map-cache"]
14
- let istanbulCoverageComposed = null
15
-
16
- await v8Coverage.result.reduce(async (previous, fileV8Coverage) => {
17
- operation.throwIfAborted()
18
- await previous
19
-
20
- const { source } = fileV8Coverage
21
- let sources
22
- // when v8 coverage comes from playwright (chromium) v8Coverage.source is set
23
- if (typeof source === "string") {
24
- sources = { source }
25
- }
26
- // when v8 coverage comes from Node.js, the source can be read from sourcemapCache
27
- else if (sourcemapCache) {
28
- sources = sourcesFromSourceMapCache(fileV8Coverage.url, sourcemapCache)
29
- }
30
- const path = urlToFileSystemPath(fileV8Coverage.url)
31
-
32
- const converter = v8ToIstanbul(
33
- path,
34
- // wrapperLength is undefined we don't need it
35
- // https://github.com/istanbuljs/v8-to-istanbul/blob/2b54bc97c5edf8a37b39a171ec29134ba9bfd532/lib/v8-to-istanbul.js#L27
36
- undefined,
37
- sources,
38
- )
39
- await converter.load()
40
-
41
- converter.applyCoverage(fileV8Coverage.functions)
42
- const istanbulCoverage = converter.toIstanbul()
43
-
44
- istanbulCoverageComposed = istanbulCoverageComposed
45
- ? composeTwoFileByFileIstanbulCoverages(
46
- istanbulCoverageComposed,
47
- istanbulCoverage,
48
- )
49
- : istanbulCoverage
50
- }, Promise.resolve())
51
-
52
- if (!istanbulCoverageComposed) {
53
- return {}
54
- }
55
- istanbulCoverageComposed = markAsConvertedFromV8(istanbulCoverageComposed)
56
- return istanbulCoverageComposed
57
- } finally {
58
- await operation.end()
59
- }
60
- }
61
-
62
- const markAsConvertedFromV8 = (fileByFileCoverage) => {
63
- const fileByFileMarked = {}
64
- Object.keys(fileByFileCoverage).forEach((key) => {
65
- const fileCoverage = fileByFileCoverage[key]
66
- fileByFileMarked[key] = {
67
- ...fileCoverage,
68
- fromV8: true,
69
- }
70
- })
71
- return fileByFileMarked
72
- }
73
-
74
- const sourcesFromSourceMapCache = (url, sourceMapCache) => {
75
- const sourceMapAndLineLengths = sourceMapCache[url]
76
- if (!sourceMapAndLineLengths) {
77
- return {}
78
- }
79
-
80
- const { data, lineLengths } = sourceMapAndLineLengths
81
- // See: https://github.com/nodejs/node/pull/34305
82
- if (!data) {
83
- return undefined
84
- }
85
-
86
- const sources = {
87
- sourcemap: data,
88
- ...(lineLengths ? { source: sourcesFromLineLengths(lineLengths) } : {}),
89
- }
90
- return sources
91
- }
92
-
93
- const sourcesFromLineLengths = (lineLengths) => {
94
- let source = ""
95
- lineLengths.forEach((length) => {
96
- source += `${"".padEnd(length, ".")}\n`
97
- })
98
- return source
99
- }