@angular-devkit/build-angular
Version:
Angular Webpack Build Facade
158 lines • 23.3 kB
JavaScript
;
/**
* @license
* Copyright Google LLC All Rights Reserved.
*
* Use of this source code is governed by an MIT-style license that can be
* found in the LICENSE file at https://angular.io/license
*/
var __importDefault = (this && this.__importDefault) || function (mod) {
return (mod && mod.__esModule) ? mod : { "default": mod };
};
Object.defineProperty(exports, "__esModule", { value: true });
const remapping_1 = __importDefault(require("@ampproject/remapping"));
const node_path_1 = require("node:path");
const node_url_1 = require("node:url");
const node_worker_threads_1 = require("node:worker_threads");
const sass_1 = require("sass");
const rebasing_importer_1 = require("./rebasing-importer");
if (!node_worker_threads_1.parentPort || !node_worker_threads_1.workerData) {
throw new Error('Sass worker must be executed as a Worker.');
}
// The importer variables are used to proxy import requests to the main thread
const { workerImporterPort, importerSignal } = node_worker_threads_1.workerData;
node_worker_threads_1.parentPort.on('message', (message) => {
if (!node_worker_threads_1.parentPort) {
throw new Error('"parentPort" is not defined. Sass worker must be executed as a Worker.');
}
const { id, hasImporter, hasLogger, source, options, rebase } = message;
const entryDirectory = (0, node_path_1.dirname)(options.url);
let warnings;
try {
const directoryCache = new Map();
const rebaseSourceMaps = options.sourceMap ? new Map() : undefined;
if (hasImporter) {
// When a custom importer function is present, the importer request must be proxied
// back to the main thread where it can be executed.
// This process must be synchronous from the perspective of dart-sass. The `Atomics`
// functions combined with the shared memory `importSignal` and the Node.js
// `receiveMessageOnPort` function are used to ensure synchronous behavior.
const proxyImporter = {
findFileUrl: (url, options) => {
Atomics.store(importerSignal, 0, 0);
workerImporterPort.postMessage({ id, url, options });
Atomics.wait(importerSignal, 0, 0);
const result = (0, node_worker_threads_1.receiveMessageOnPort)(workerImporterPort)?.message;
return result ? (0, node_url_1.pathToFileURL)(result) : null;
},
};
options.importers = [
rebase
? (0, rebasing_importer_1.sassBindWorkaround)(new rebasing_importer_1.ModuleUrlRebasingImporter(entryDirectory, directoryCache, rebaseSourceMaps, proxyImporter.findFileUrl))
: proxyImporter,
];
}
if (rebase && options.loadPaths?.length) {
options.importers ?? (options.importers = []);
options.importers.push((0, rebasing_importer_1.sassBindWorkaround)(new rebasing_importer_1.LoadPathsUrlRebasingImporter(entryDirectory, directoryCache, rebaseSourceMaps, options.loadPaths)));
options.loadPaths = undefined;
}
let relativeImporter;
if (rebase) {
relativeImporter = (0, rebasing_importer_1.sassBindWorkaround)(new rebasing_importer_1.RelativeUrlRebasingImporter(entryDirectory, directoryCache, rebaseSourceMaps));
}
// The synchronous Sass render function can be up to two times faster than the async variant
const result = (0, sass_1.compileString)(source, {
...options,
// URL is not serializable so to convert to string in the parent and back to URL here.
url: (0, node_url_1.pathToFileURL)(options.url),
// The `importer` option (singular) handles relative imports
importer: relativeImporter,
logger: hasLogger
? {
warn(message, { deprecation, span, stack }) {
warnings ?? (warnings = []);
warnings.push({
message,
deprecation,
stack,
span: span && convertSourceSpan(span),
});
},
}
: undefined,
});
if (result.sourceMap && rebaseSourceMaps?.size) {
// Merge the intermediate rebasing source maps into the final Sass generated source map.
// Casting is required due to small but compatible differences in typings between the packages.
result.sourceMap = (0, remapping_1.default)(result.sourceMap,
// To prevent an infinite lookup loop, skip getting the source when the rebasing source map
// is referencing its original self.
(file, context) => (file !== context.importer ? rebaseSourceMaps.get(file) : null));
}
node_worker_threads_1.parentPort.postMessage({
id,
warnings,
result: {
...result,
// URL is not serializable so to convert to string here and back to URL in the parent.
loadedUrls: result.loadedUrls.map((p) => (0, node_url_1.fileURLToPath)(p)),
},
});
}
catch (error) {
// Needed because V8 will only serialize the message and stack properties of an Error instance.
if (error instanceof sass_1.Exception) {
const { span, message, stack, sassMessage, sassStack } = error;
node_worker_threads_1.parentPort.postMessage({
id,
warnings,
error: {
span: convertSourceSpan(span),
message,
stack,
sassMessage,
sassStack,
},
});
}
else if (error instanceof Error) {
const { message, stack } = error;
node_worker_threads_1.parentPort.postMessage({ id, warnings, error: { message, stack } });
}
else {
node_worker_threads_1.parentPort.postMessage({
id,
warnings,
error: { message: 'An unknown error has occurred.' },
});
}
}
});
/**
* Converts a Sass SourceSpan object into a serializable form.
* The SourceSpan object contains a URL property which must be converted into a string.
* Also, most of the interface's properties are get accessors and are not automatically
* serialized when sent back from the worker.
*
* @param span The Sass SourceSpan object to convert.
* @returns A serializable form of the SourceSpan object.
*/
function convertSourceSpan(span) {
return {
text: span.text,
context: span.context,
end: {
column: span.end.column,
offset: span.end.offset,
line: span.end.line,
},
start: {
column: span.start.column,
offset: span.start.offset,
line: span.start.line,
},
url: span.url ? (0, node_url_1.fileURLToPath)(span.url) : undefined,
};
}
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"worker.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/tools/sass/worker.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 mergeSourceMaps, { RawSourceMap } from '@ampproject/remapping';\nimport { dirname } from 'node:path';\nimport { fileURLToPath, pathToFileURL } from 'node:url';\nimport { MessagePort, parentPort, receiveMessageOnPort, workerData } from 'node:worker_threads';\nimport {\n  Exception,\n  FileImporter,\n  SourceSpan,\n  StringOptionsWithImporter,\n  compileString,\n} from 'sass';\nimport {\n  DirectoryEntry,\n  LoadPathsUrlRebasingImporter,\n  ModuleUrlRebasingImporter,\n  RelativeUrlRebasingImporter,\n  sassBindWorkaround,\n} from './rebasing-importer';\n\n/**\n * A request to render a Sass stylesheet using the supplied options.\n */\ninterface RenderRequestMessage {\n  /**\n   * The unique request identifier that links the render action with a callback and optional\n   * importer on the main thread.\n   */\n  id: number;\n  /**\n   * The contents to compile.\n   */\n  source: string;\n  /**\n   * The Sass options to provide to the `dart-sass` compile function.\n   */\n  options: Omit<StringOptionsWithImporter<'sync'>, 'url'> & { url: string };\n  /**\n   * Indicates the request has a custom importer function on the main thread.\n   */\n  hasImporter: boolean;\n  /**\n   * Indicates the request has a custom logger for warning messages.\n   */\n  hasLogger: boolean;\n  /**\n   * Indicates paths within url() CSS functions should be rebased.\n   */\n  rebase: boolean;\n}\n\nif (!parentPort || !workerData) {\n  throw new Error('Sass worker must be executed as a Worker.');\n}\n\n// The importer variables are used to proxy import requests to the main thread\nconst { workerImporterPort, importerSignal } = workerData as {\n  workerImporterPort: MessagePort;\n  importerSignal: Int32Array;\n};\n\nparentPort.on('message', (message: RenderRequestMessage) => {\n  if (!parentPort) {\n    throw new Error('\"parentPort\" is not defined. Sass worker must be executed as a Worker.');\n  }\n\n  const { id, hasImporter, hasLogger, source, options, rebase } = message;\n  const entryDirectory = dirname(options.url);\n  let warnings:\n    | {\n        message: string;\n        deprecation: boolean;\n        stack?: string;\n        span?: Omit<SourceSpan, 'url'> & { url?: string };\n      }[]\n    | undefined;\n  try {\n    const directoryCache = new Map<string, DirectoryEntry>();\n    const rebaseSourceMaps = options.sourceMap ? new Map<string, RawSourceMap>() : undefined;\n    if (hasImporter) {\n      // When a custom importer function is present, the importer request must be proxied\n      // back to the main thread where it can be executed.\n      // This process must be synchronous from the perspective of dart-sass. The `Atomics`\n      // functions combined with the shared memory `importSignal` and the Node.js\n      // `receiveMessageOnPort` function are used to ensure synchronous behavior.\n      const proxyImporter: FileImporter<'sync'> = {\n        findFileUrl: (url, options) => {\n          Atomics.store(importerSignal, 0, 0);\n          workerImporterPort.postMessage({ id, url, options });\n          Atomics.wait(importerSignal, 0, 0);\n\n          const result = receiveMessageOnPort(workerImporterPort)?.message as string | null;\n\n          return result ? pathToFileURL(result) : null;\n        },\n      };\n      options.importers = [\n        rebase\n          ? sassBindWorkaround(\n              new ModuleUrlRebasingImporter(\n                entryDirectory,\n                directoryCache,\n                rebaseSourceMaps,\n                proxyImporter.findFileUrl,\n              ),\n            )\n          : proxyImporter,\n      ];\n    }\n\n    if (rebase && options.loadPaths?.length) {\n      options.importers ??= [];\n      options.importers.push(\n        sassBindWorkaround(\n          new LoadPathsUrlRebasingImporter(\n            entryDirectory,\n            directoryCache,\n            rebaseSourceMaps,\n            options.loadPaths,\n          ),\n        ),\n      );\n      options.loadPaths = undefined;\n    }\n\n    let relativeImporter;\n    if (rebase) {\n      relativeImporter = sassBindWorkaround(\n        new RelativeUrlRebasingImporter(entryDirectory, directoryCache, rebaseSourceMaps),\n      );\n    }\n\n    // The synchronous Sass render function can be up to two times faster than the async variant\n    const result = compileString(source, {\n      ...options,\n      // URL is not serializable so to convert to string in the parent and back to URL here.\n      url: pathToFileURL(options.url),\n      // The `importer` option (singular) handles relative imports\n      importer: relativeImporter,\n      logger: hasLogger\n        ? {\n            warn(message, { deprecation, span, stack }) {\n              warnings ??= [];\n              warnings.push({\n                message,\n                deprecation,\n                stack,\n                span: span && convertSourceSpan(span),\n              });\n            },\n          }\n        : undefined,\n    });\n\n    if (result.sourceMap && rebaseSourceMaps?.size) {\n      // Merge the intermediate rebasing source maps into the final Sass generated source map.\n      // Casting is required due to small but compatible differences in typings between the packages.\n      result.sourceMap = mergeSourceMaps(\n        result.sourceMap as unknown as RawSourceMap,\n        // To prevent an infinite lookup loop, skip getting the source when the rebasing source map\n        // is referencing its original self.\n        (file, context) => (file !== context.importer ? rebaseSourceMaps.get(file) : null),\n      ) as unknown as typeof result.sourceMap;\n    }\n\n    parentPort.postMessage({\n      id,\n      warnings,\n      result: {\n        ...result,\n        // URL is not serializable so to convert to string here and back to URL in the parent.\n        loadedUrls: result.loadedUrls.map((p) => fileURLToPath(p)),\n      },\n    });\n  } catch (error) {\n    // Needed because V8 will only serialize the message and stack properties of an Error instance.\n    if (error instanceof Exception) {\n      const { span, message, stack, sassMessage, sassStack } = error;\n      parentPort.postMessage({\n        id,\n        warnings,\n        error: {\n          span: convertSourceSpan(span),\n          message,\n          stack,\n          sassMessage,\n          sassStack,\n        },\n      });\n    } else if (error instanceof Error) {\n      const { message, stack } = error;\n      parentPort.postMessage({ id, warnings, error: { message, stack } });\n    } else {\n      parentPort.postMessage({\n        id,\n        warnings,\n        error: { message: 'An unknown error has occurred.' },\n      });\n    }\n  }\n});\n\n/**\n * Converts a Sass SourceSpan object into a serializable form.\n * The SourceSpan object contains a URL property which must be converted into a string.\n * Also, most of the interface's properties are get accessors and are not automatically\n * serialized when sent back from the worker.\n *\n * @param span The Sass SourceSpan object to convert.\n * @returns A serializable form of the SourceSpan object.\n */\nfunction convertSourceSpan(span: SourceSpan): Omit<SourceSpan, 'url'> & { url?: string } {\n  return {\n    text: span.text,\n    context: span.context,\n    end: {\n      column: span.end.column,\n      offset: span.end.offset,\n      line: span.end.line,\n    },\n    start: {\n      column: span.start.column,\n      offset: span.start.offset,\n      line: span.start.line,\n    },\n    url: span.url ? fileURLToPath(span.url) : undefined,\n  };\n}\n"]}