1 | ;
|
2 | /**
|
3 | * @license
|
4 | * Copyright Google LLC All Rights Reserved.
|
5 | *
|
6 | * Use of this source code is governed by an MIT-style license that can be
|
7 | * found in the LICENSE file at https://angular.io/license
|
8 | */
|
9 | var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
|
10 | if (k2 === undefined) k2 = k;
|
11 | var desc = Object.getOwnPropertyDescriptor(m, k);
|
12 | if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
|
13 | desc = { enumerable: true, get: function() { return m[k]; } };
|
14 | }
|
15 | Object.defineProperty(o, k2, desc);
|
16 | }) : (function(o, m, k, k2) {
|
17 | if (k2 === undefined) k2 = k;
|
18 | o[k2] = m[k];
|
19 | }));
|
20 | var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
|
21 | Object.defineProperty(o, "default", { enumerable: true, value: v });
|
22 | }) : function(o, v) {
|
23 | o["default"] = v;
|
24 | });
|
25 | var __importStar = (this && this.__importStar) || function (mod) {
|
26 | if (mod && mod.__esModule) return mod;
|
27 | var result = {};
|
28 | if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k);
|
29 | __setModuleDefault(result, mod);
|
30 | return result;
|
31 | };
|
32 | var __importDefault = (this && this.__importDefault) || function (mod) {
|
33 | return (mod && mod.__esModule) ? mod : { "default": mod };
|
34 | };
|
35 | Object.defineProperty(exports, "__esModule", { value: true });
|
36 | exports.WebpackResourceLoader = void 0;
|
37 | const node_assert_1 = __importDefault(require("node:assert"));
|
38 | const node_buffer_1 = require("node:buffer");
|
39 | const path = __importStar(require("node:path"));
|
40 | const vm = __importStar(require("node:vm"));
|
41 | const diagnostics_1 = require("./ivy/diagnostics");
|
42 | const paths_1 = require("./ivy/paths");
|
43 | const inline_resource_1 = require("./loaders/inline-resource");
|
44 | const replace_resources_1 = require("./transformers/replace_resources");
|
45 | class WebpackResourceLoader {
|
46 | constructor(shouldCache) {
|
47 | this._fileDependencies = new Map();
|
48 | this._reverseDependencies = new Map();
|
49 | this.modifiedResources = new Set();
|
50 | this.outputPathCounter = 1;
|
51 | this.inlineDataLoaderPath = inline_resource_1.InlineAngularResourceLoaderPath;
|
52 | if (shouldCache) {
|
53 | this.fileCache = new Map();
|
54 | this.assetCache = new Map();
|
55 | }
|
56 | }
|
57 | update(parentCompilation, changedFiles) {
|
58 | var _a, _b, _c, _d, _e;
|
59 | this._parentCompilation = parentCompilation;
|
60 | // Update resource cache and modified resources
|
61 | this.modifiedResources.clear();
|
62 | if (changedFiles) {
|
63 | for (const changedFile of changedFiles) {
|
64 | const changedFileNormalized = (0, paths_1.normalizePath)(changedFile);
|
65 | (_a = this.assetCache) === null || _a === void 0 ? void 0 : _a.delete(changedFileNormalized);
|
66 | for (const affectedResource of this.getAffectedResources(changedFile)) {
|
67 | const affectedResourceNormalized = (0, paths_1.normalizePath)(affectedResource);
|
68 | (_b = this.fileCache) === null || _b === void 0 ? void 0 : _b.delete(affectedResourceNormalized);
|
69 | this.modifiedResources.add(affectedResource);
|
70 | for (const effectedDependencies of this.getResourceDependencies(affectedResourceNormalized)) {
|
71 | (_c = this.assetCache) === null || _c === void 0 ? void 0 : _c.delete((0, paths_1.normalizePath)(effectedDependencies));
|
72 | }
|
73 | }
|
74 | }
|
75 | }
|
76 | else {
|
77 | (_d = this.fileCache) === null || _d === void 0 ? void 0 : _d.clear();
|
78 | (_e = this.assetCache) === null || _e === void 0 ? void 0 : _e.clear();
|
79 | }
|
80 | // Re-emit all assets for un-effected files
|
81 | if (this.assetCache) {
|
82 | for (const [, { name, source, info }] of this.assetCache) {
|
83 | this._parentCompilation.emitAsset(name, source, info);
|
84 | }
|
85 | }
|
86 | }
|
87 | clearParentCompilation() {
|
88 | this._parentCompilation = undefined;
|
89 | }
|
90 | getModifiedResourceFiles() {
|
91 | return this.modifiedResources;
|
92 | }
|
93 | getResourceDependencies(filePath) {
|
94 | return this._fileDependencies.get(filePath) || [];
|
95 | }
|
96 | getAffectedResources(file) {
|
97 | return this._reverseDependencies.get(file) || [];
|
98 | }
|
99 | setAffectedResources(file, resources) {
|
100 | this._reverseDependencies.set(file, new Set(resources));
|
101 | }
|
102 | // eslint-disable-next-line max-lines-per-function
|
103 | async _compile(filePath, data, fileExtension, resourceType, containingFile) {
|
104 | if (!this._parentCompilation) {
|
105 | throw new Error('WebpackResourceLoader cannot be used without parentCompilation');
|
106 | }
|
107 | const { context, webpack } = this._parentCompilation.compiler;
|
108 | const { EntryPlugin, NormalModule, WebpackError, library, node, sources, util: { createHash }, } = webpack;
|
109 | const getEntry = () => {
|
110 | if (filePath) {
|
111 | return `${filePath}?${replace_resources_1.NG_COMPONENT_RESOURCE_QUERY}`;
|
112 | }
|
113 | else if (resourceType) {
|
114 | return (
|
115 | // app.component.ts-2.css?ngResource!=!@ngtools/webpack/src/loaders/inline-resource.js!app.component.ts
|
116 | `${containingFile}-${this.outputPathCounter}.${fileExtension}` +
|
117 | `?${replace_resources_1.NG_COMPONENT_RESOURCE_QUERY}!=!${this.inlineDataLoaderPath}!${containingFile}`);
|
118 | }
|
119 | else if (data) {
|
120 | // Create a special URL for reading the resource from memory
|
121 | return `angular-resource:${resourceType},${createHash('xxhash64')
|
122 | .update(data)
|
123 | .digest('hex')}`;
|
124 | }
|
125 | throw new Error(`"filePath", "resourceType" or "data" must be specified.`);
|
126 | };
|
127 | const entry = getEntry();
|
128 | // Simple sanity check.
|
129 | if (filePath === null || filePath === void 0 ? void 0 : filePath.match(/\.[jt]s$/)) {
|
130 | throw new Error(`Cannot use a JavaScript or TypeScript file (${filePath}) in a component's styleUrls or templateUrl.`);
|
131 | }
|
132 | const outputFilePath = filePath ||
|
133 | `${containingFile}-angular-inline--${this.outputPathCounter++}.${resourceType === 'template' ? 'html' : 'css'}`;
|
134 | const outputOptions = {
|
135 | filename: outputFilePath,
|
136 | library: {
|
137 | type: 'var',
|
138 | name: 'resource',
|
139 | },
|
140 | };
|
141 | const childCompiler = this._parentCompilation.createChildCompiler('angular-compiler:resource', outputOptions, [
|
142 | new node.NodeTemplatePlugin(outputOptions),
|
143 | new node.NodeTargetPlugin(),
|
144 | new EntryPlugin(context, entry, { name: 'resource' }),
|
145 | new library.EnableLibraryPlugin('var'),
|
146 | ]);
|
147 | childCompiler.hooks.thisCompilation.tap('angular-compiler', (compilation, { normalModuleFactory }) => {
|
148 | // If no data is provided, the resource will be read from the filesystem
|
149 | if (data !== undefined) {
|
150 | normalModuleFactory.hooks.resolveForScheme
|
151 | .for('angular-resource')
|
152 | .tap('angular-compiler', (resourceData) => {
|
153 | if (filePath) {
|
154 | resourceData.path = filePath;
|
155 | resourceData.resource = filePath;
|
156 | }
|
157 | return true;
|
158 | });
|
159 | NormalModule.getCompilationHooks(compilation)
|
160 | .readResourceForScheme.for('angular-resource')
|
161 | .tap('angular-compiler', () => data);
|
162 | compilation[inline_resource_1.InlineAngularResourceSymbol] = data;
|
163 | }
|
164 | compilation.hooks.additionalAssets.tap('angular-compiler', () => {
|
165 | const asset = compilation.assets[outputFilePath];
|
166 | if (!asset) {
|
167 | return;
|
168 | }
|
169 | try {
|
170 | const output = this._evaluate(outputFilePath, asset.source().toString());
|
171 | if (typeof output === 'string') {
|
172 | compilation.assets[outputFilePath] = new sources.RawSource(output);
|
173 | }
|
174 | }
|
175 | catch (error) {
|
176 | (0, node_assert_1.default)(error instanceof Error, 'catch clause variable is not an Error instance');
|
177 | // Use compilation errors, as otherwise webpack will choke
|
178 | (0, diagnostics_1.addError)(compilation, error.message);
|
179 | }
|
180 | });
|
181 | });
|
182 | let finalContent;
|
183 | childCompiler.hooks.compilation.tap('angular-compiler', (childCompilation) => {
|
184 | childCompilation.hooks.processAssets.tap({ name: 'angular-compiler', stage: webpack.Compilation.PROCESS_ASSETS_STAGE_REPORT }, () => {
|
185 | var _a;
|
186 | finalContent = (_a = childCompilation.assets[outputFilePath]) === null || _a === void 0 ? void 0 : _a.source().toString();
|
187 | for (const { files } of childCompilation.chunks) {
|
188 | for (const file of files) {
|
189 | childCompilation.deleteAsset(file);
|
190 | }
|
191 | }
|
192 | });
|
193 | });
|
194 | return new Promise((resolve, reject) => {
|
195 | childCompiler.runAsChild((error, _, childCompilation) => {
|
196 | var _a, _b;
|
197 | if (error) {
|
198 | reject(error);
|
199 | return;
|
200 | }
|
201 | else if (!childCompilation) {
|
202 | reject(new Error('Unknown child compilation error'));
|
203 | return;
|
204 | }
|
205 | // Workaround to attempt to reduce memory usage of child compilations.
|
206 | // This removes the child compilation from the main compilation and manually propagates
|
207 | // all dependencies, warnings, and errors.
|
208 | const parent = childCompiler.parentCompilation;
|
209 | if (parent) {
|
210 | parent.children = parent.children.filter((child) => child !== childCompilation);
|
211 | let fileDependencies;
|
212 | for (const dependency of childCompilation.fileDependencies) {
|
213 | // Skip paths that do not appear to be files (have no extension).
|
214 | // `fileDependencies` can contain directories and not just files which can
|
215 | // cause incorrect cache invalidation on rebuilds.
|
216 | if (!path.extname(dependency)) {
|
217 | continue;
|
218 | }
|
219 | if (data && containingFile && dependency.endsWith(entry)) {
|
220 | // use containing file if the resource was inline
|
221 | parent.fileDependencies.add(containingFile);
|
222 | }
|
223 | else {
|
224 | parent.fileDependencies.add(dependency);
|
225 | }
|
226 | // Save the dependencies for this resource.
|
227 | if (filePath) {
|
228 | const resolvedFile = (0, paths_1.normalizePath)(dependency);
|
229 | const entry = this._reverseDependencies.get(resolvedFile);
|
230 | if (entry) {
|
231 | entry.add(filePath);
|
232 | }
|
233 | else {
|
234 | this._reverseDependencies.set(resolvedFile, new Set([filePath]));
|
235 | }
|
236 | if (fileDependencies) {
|
237 | fileDependencies.add(dependency);
|
238 | }
|
239 | else {
|
240 | fileDependencies = new Set([dependency]);
|
241 | this._fileDependencies.set(filePath, fileDependencies);
|
242 | }
|
243 | }
|
244 | }
|
245 | parent.contextDependencies.addAll(childCompilation.contextDependencies);
|
246 | parent.missingDependencies.addAll(childCompilation.missingDependencies);
|
247 | parent.buildDependencies.addAll(childCompilation.buildDependencies);
|
248 | parent.warnings.push(...childCompilation.warnings);
|
249 | parent.errors.push(...childCompilation.errors);
|
250 | if (this.assetCache) {
|
251 | for (const { info, name, source } of childCompilation.getAssets()) {
|
252 | // Use the originating file as the cache key if present
|
253 | // Otherwise, generate a cache key based on the generated name
|
254 | const cacheKey = (_a = info.sourceFilename) !== null && _a !== void 0 ? _a : `!![GENERATED]:${name}`;
|
255 | this.assetCache.set(cacheKey, { info, name, source });
|
256 | }
|
257 | }
|
258 | }
|
259 | resolve({
|
260 | content: finalContent !== null && finalContent !== void 0 ? finalContent : '',
|
261 | success: ((_b = childCompilation.errors) === null || _b === void 0 ? void 0 : _b.length) === 0,
|
262 | });
|
263 | });
|
264 | });
|
265 | }
|
266 | _evaluate(filename, source) {
|
267 | // Evaluate code
|
268 | var _a;
|
269 | // css-loader requires the btoa function to exist to correctly generate inline sourcemaps
|
270 | const context = {
|
271 | btoa(input) {
|
272 | return node_buffer_1.Buffer.from(input).toString('base64');
|
273 | },
|
274 | };
|
275 | try {
|
276 | vm.runInNewContext(source, context, { filename });
|
277 | }
|
278 | catch (_b) {
|
279 | // Error are propagated through the child compilation.
|
280 | return null;
|
281 | }
|
282 | if (typeof context.resource === 'string') {
|
283 | return context.resource;
|
284 | }
|
285 | else if (typeof ((_a = context.resource) === null || _a === void 0 ? void 0 : _a.default) === 'string') {
|
286 | return context.resource.default;
|
287 | }
|
288 | throw new Error(`The loader "${filename}" didn't return a string.`);
|
289 | }
|
290 | async get(filePath) {
|
291 | var _a;
|
292 | const normalizedFile = (0, paths_1.normalizePath)(filePath);
|
293 | let compilationResult = (_a = this.fileCache) === null || _a === void 0 ? void 0 : _a.get(normalizedFile);
|
294 | if (compilationResult === undefined) {
|
295 | // cache miss so compile resource
|
296 | compilationResult = await this._compile(filePath);
|
297 | // Only cache if compilation was successful
|
298 | if (this.fileCache && compilationResult.success) {
|
299 | this.fileCache.set(normalizedFile, compilationResult);
|
300 | }
|
301 | }
|
302 | return compilationResult.content;
|
303 | }
|
304 | async process(data, fileExtension, resourceType, containingFile) {
|
305 | if (data.trim().length === 0) {
|
306 | return '';
|
307 | }
|
308 | const compilationResult = await this._compile(undefined, data, fileExtension, resourceType, containingFile);
|
309 | return compilationResult.content;
|
310 | }
|
311 | }
|
312 | exports.WebpackResourceLoader = WebpackResourceLoader;
|
313 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resource_loader.js","sourceRoot":"","sources":["../../../../../../../packages/ngtools/webpack/src/resource_loader.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google LLC All Rights Reserved.\n *\n * Use of this source code is governed by an MIT-style license that can be\n * found in the LICENSE file at https://angular.io/license\n */\n\nimport assert from 'node:assert';\nimport { Buffer } from 'node:buffer';\nimport * as path from 'node:path';\nimport * as vm from 'node:vm';\nimport type { Asset, Compilation } from 'webpack';\nimport { addError } from './ivy/diagnostics';\nimport { normalizePath } from './ivy/paths';\nimport {\n  CompilationWithInlineAngularResource,\n  InlineAngularResourceLoaderPath,\n  InlineAngularResourceSymbol,\n} from './loaders/inline-resource';\nimport { NG_COMPONENT_RESOURCE_QUERY } from './transformers/replace_resources';\n\ninterface CompilationOutput {\n  content: string;\n  map?: string;\n  success: boolean;\n}\n\nexport class WebpackResourceLoader {\n  private _parentCompilation?: Compilation;\n  private _fileDependencies = new Map<string, Set<string>>();\n  private _reverseDependencies = new Map<string, Set<string>>();\n\n  private fileCache?: Map<string, CompilationOutput>;\n  private assetCache?: Map<string, Asset>;\n\n  private modifiedResources = new Set<string>();\n  private outputPathCounter = 1;\n\n  private readonly inlineDataLoaderPath = InlineAngularResourceLoaderPath;\n\n  constructor(shouldCache: boolean) {\n    if (shouldCache) {\n      this.fileCache = new Map();\n      this.assetCache = new Map();\n    }\n  }\n\n  update(parentCompilation: Compilation, changedFiles?: Iterable<string>) {\n    this._parentCompilation = parentCompilation;\n\n    // Update resource cache and modified resources\n    this.modifiedResources.clear();\n\n    if (changedFiles) {\n      for (const changedFile of changedFiles) {\n        const changedFileNormalized = normalizePath(changedFile);\n        this.assetCache?.delete(changedFileNormalized);\n\n        for (const affectedResource of this.getAffectedResources(changedFile)) {\n          const affectedResourceNormalized = normalizePath(affectedResource);\n          this.fileCache?.delete(affectedResourceNormalized);\n          this.modifiedResources.add(affectedResource);\n\n          for (const effectedDependencies of this.getResourceDependencies(\n            affectedResourceNormalized,\n          )) {\n            this.assetCache?.delete(normalizePath(effectedDependencies));\n          }\n        }\n      }\n    } else {\n      this.fileCache?.clear();\n      this.assetCache?.clear();\n    }\n\n    // Re-emit all assets for un-effected files\n    if (this.assetCache) {\n      for (const [, { name, source, info }] of this.assetCache) {\n        this._parentCompilation.emitAsset(name, source, info);\n      }\n    }\n  }\n\n  clearParentCompilation() {\n    this._parentCompilation = undefined;\n  }\n\n  getModifiedResourceFiles() {\n    return this.modifiedResources;\n  }\n\n  getResourceDependencies(filePath: string) {\n    return this._fileDependencies.get(filePath) || [];\n  }\n\n  getAffectedResources(file: string) {\n    return this._reverseDependencies.get(file) || [];\n  }\n\n  setAffectedResources(file: string, resources: Iterable<string>) {\n    this._reverseDependencies.set(file, new Set(resources));\n  }\n\n  // eslint-disable-next-line max-lines-per-function\n  private async _compile(\n    filePath?: string,\n    data?: string,\n    fileExtension?: string,\n    resourceType?: 'style' | 'template',\n    containingFile?: string,\n  ): Promise<CompilationOutput> {\n    if (!this._parentCompilation) {\n      throw new Error('WebpackResourceLoader cannot be used without parentCompilation');\n    }\n\n    const { context, webpack } = this._parentCompilation.compiler;\n    const {\n      EntryPlugin,\n      NormalModule,\n      WebpackError,\n      library,\n      node,\n      sources,\n      util: { createHash },\n    } = webpack;\n\n    const getEntry = (): string => {\n      if (filePath) {\n        return `${filePath}?${NG_COMPONENT_RESOURCE_QUERY}`;\n      } else if (resourceType) {\n        return (\n          // app.component.ts-2.css?ngResource!=!@ngtools/webpack/src/loaders/inline-resource.js!app.component.ts\n          `${containingFile}-${this.outputPathCounter}.${fileExtension}` +\n          `?${NG_COMPONENT_RESOURCE_QUERY}!=!${this.inlineDataLoaderPath}!${containingFile}`\n        );\n      } else if (data) {\n        // Create a special URL for reading the resource from memory\n        return `angular-resource:${resourceType},${createHash('xxhash64')\n          .update(data)\n          .digest('hex')}`;\n      }\n\n      throw new Error(`\"filePath\", \"resourceType\" or \"data\" must be specified.`);\n    };\n\n    const entry = getEntry();\n\n    // Simple sanity check.\n    if (filePath?.match(/\\.[jt]s$/)) {\n      throw new Error(\n        `Cannot use a JavaScript or TypeScript file (${filePath}) in a component's styleUrls or templateUrl.`,\n      );\n    }\n\n    const outputFilePath =\n      filePath ||\n      `${containingFile}-angular-inline--${this.outputPathCounter++}.${\n        resourceType === 'template' ? 'html' : 'css'\n      }`;\n    const outputOptions = {\n      filename: outputFilePath,\n      library: {\n        type: 'var',\n        name: 'resource',\n      },\n    };\n\n    const childCompiler = this._parentCompilation.createChildCompiler(\n      'angular-compiler:resource',\n      outputOptions,\n      [\n        new node.NodeTemplatePlugin(outputOptions),\n        new node.NodeTargetPlugin(),\n        new EntryPlugin(context, entry, { name: 'resource' }),\n        new library.EnableLibraryPlugin('var'),\n      ],\n    );\n\n    childCompiler.hooks.thisCompilation.tap(\n      'angular-compiler',\n      (compilation, { normalModuleFactory }) => {\n        // If no data is provided, the resource will be read from the filesystem\n        if (data !== undefined) {\n          normalModuleFactory.hooks.resolveForScheme\n            .for('angular-resource')\n            .tap('angular-compiler', (resourceData) => {\n              if (filePath) {\n                resourceData.path = filePath;\n                resourceData.resource = filePath;\n              }\n\n              return true;\n            });\n          NormalModule.getCompilationHooks(compilation)\n            .readResourceForScheme.for('angular-resource')\n            .tap('angular-compiler', () => data);\n\n          (compilation as CompilationWithInlineAngularResource)[InlineAngularResourceSymbol] = data;\n        }\n\n        compilation.hooks.additionalAssets.tap('angular-compiler', () => {\n          const asset = compilation.assets[outputFilePath];\n          if (!asset) {\n            return;\n          }\n\n          try {\n            const output = this._evaluate(outputFilePath, asset.source().toString());\n\n            if (typeof output === 'string') {\n              compilation.assets[outputFilePath] = new sources.RawSource(output);\n            }\n          } catch (error) {\n            assert(error instanceof Error, 'catch clause variable is not an Error instance');\n            // Use compilation errors, as otherwise webpack will choke\n            addError(compilation, error.message);\n          }\n        });\n      },\n    );\n\n    let finalContent: string | undefined;\n    childCompiler.hooks.compilation.tap('angular-compiler', (childCompilation) => {\n      childCompilation.hooks.processAssets.tap(\n        { name: 'angular-compiler', stage: webpack.Compilation.PROCESS_ASSETS_STAGE_REPORT },\n        () => {\n          finalContent = childCompilation.assets[outputFilePath]?.source().toString();\n\n          for (const { files } of childCompilation.chunks) {\n            for (const file of files) {\n              childCompilation.deleteAsset(file);\n            }\n          }\n        },\n      );\n    });\n\n    return new Promise<CompilationOutput>((resolve, reject) => {\n      childCompiler.runAsChild((error, _, childCompilation) => {\n        if (error) {\n          reject(error);\n\n          return;\n        } else if (!childCompilation) {\n          reject(new Error('Unknown child compilation error'));\n\n          return;\n        }\n\n        // Workaround to attempt to reduce memory usage of child compilations.\n        // This removes the child compilation from the main compilation and manually propagates\n        // all dependencies, warnings, and errors.\n        const parent = childCompiler.parentCompilation;\n        if (parent) {\n          parent.children = parent.children.filter((child) => child !== childCompilation);\n          let fileDependencies: Set<string> | undefined;\n\n          for (const dependency of childCompilation.fileDependencies) {\n            // Skip paths that do not appear to be files (have no extension).\n            // `fileDependencies` can contain directories and not just files which can\n            // cause incorrect cache invalidation on rebuilds.\n            if (!path.extname(dependency)) {\n              continue;\n            }\n\n            if (data && containingFile && dependency.endsWith(entry)) {\n              // use containing file if the resource was inline\n              parent.fileDependencies.add(containingFile);\n            } else {\n              parent.fileDependencies.add(dependency);\n            }\n\n            // Save the dependencies for this resource.\n            if (filePath) {\n              const resolvedFile = normalizePath(dependency);\n              const entry = this._reverseDependencies.get(resolvedFile);\n              if (entry) {\n                entry.add(filePath);\n              } else {\n                this._reverseDependencies.set(resolvedFile, new Set([filePath]));\n              }\n\n              if (fileDependencies) {\n                fileDependencies.add(dependency);\n              } else {\n                fileDependencies = new Set([dependency]);\n                this._fileDependencies.set(filePath, fileDependencies);\n              }\n            }\n          }\n\n          parent.contextDependencies.addAll(childCompilation.contextDependencies);\n          parent.missingDependencies.addAll(childCompilation.missingDependencies);\n          parent.buildDependencies.addAll(childCompilation.buildDependencies);\n\n          parent.warnings.push(...childCompilation.warnings);\n          parent.errors.push(...childCompilation.errors);\n\n          if (this.assetCache) {\n            for (const { info, name, source } of childCompilation.getAssets()) {\n              // Use the originating file as the cache key if present\n              // Otherwise, generate a cache key based on the generated name\n              const cacheKey = info.sourceFilename ?? `!![GENERATED]:${name}`;\n\n              this.assetCache.set(cacheKey, { info, name, source });\n            }\n          }\n        }\n\n        resolve({\n          content: finalContent ?? '',\n          success: childCompilation.errors?.length === 0,\n        });\n      });\n    });\n  }\n\n  private _evaluate(filename: string, source: string): string | null {\n    // Evaluate code\n\n    // css-loader requires the btoa function to exist to correctly generate inline sourcemaps\n    const context: { btoa: (input: string) => string; resource?: string | { default?: string } } = {\n      btoa(input) {\n        return Buffer.from(input).toString('base64');\n      },\n    };\n\n    try {\n      vm.runInNewContext(source, context, { filename });\n    } catch {\n      // Error are propagated through the child compilation.\n      return null;\n    }\n\n    if (typeof context.resource === 'string') {\n      return context.resource;\n    } else if (typeof context.resource?.default === 'string') {\n      return context.resource.default;\n    }\n\n    throw new Error(`The loader \"${filename}\" didn't return a string.`);\n  }\n\n  async get(filePath: string): Promise<string> {\n    const normalizedFile = normalizePath(filePath);\n    let compilationResult = this.fileCache?.get(normalizedFile);\n\n    if (compilationResult === undefined) {\n      // cache miss so compile resource\n      compilationResult = await this._compile(filePath);\n\n      // Only cache if compilation was successful\n      if (this.fileCache && compilationResult.success) {\n        this.fileCache.set(normalizedFile, compilationResult);\n      }\n    }\n\n    return compilationResult.content;\n  }\n\n  async process(\n    data: string,\n    fileExtension: string | undefined,\n    resourceType: 'template' | 'style',\n    containingFile?: string,\n  ): Promise<string> {\n    if (data.trim().length === 0) {\n      return '';\n    }\n\n    const compilationResult = await this._compile(\n      undefined,\n      data,\n      fileExtension,\n      resourceType,\n      containingFile,\n    );\n\n    return compilationResult.content;\n  }\n}\n"]} |
\ | No newline at end of file |