@angular-devkit/build-angular 16.2.0-next.2 → 16.2.0-next.4
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/package.json +14 -14
- package/src/builders/dev-server/vite-server.js +4 -2
- package/src/tools/babel/plugins/adjust-typescript-enums.js +21 -31
- package/src/tools/esbuild/angular/compilation/angular-compilation.js +2 -1
- package/src/tools/esbuild/application-code-bundle.js +2 -1
- package/src/tools/esbuild/stylesheets/sass-language.js +58 -28
- package/src/tools/sass/lexer.d.ts +29 -0
- package/src/tools/sass/lexer.js +248 -0
- package/src/tools/sass/rebasing-importer.d.ts +5 -2
- package/src/tools/sass/rebasing-importer.js +47 -153
- package/src/tools/sass/sass-service.d.ts +5 -0
- package/src/tools/sass/sass-service.js +1 -1
- package/src/utils/index-file/inline-critical-css.js +10 -2
|
@@ -48,16 +48,19 @@ exports.SassStylesheetLanguage = Object.freeze({
|
|
|
48
48
|
fileFilter: /\.s[ac]ss$/,
|
|
49
49
|
process(data, file, format, options, build) {
|
|
50
50
|
const syntax = format === 'sass' ? 'indented' : 'scss';
|
|
51
|
-
const resolveUrl = async (url,
|
|
51
|
+
const resolveUrl = async (url, options) => {
|
|
52
52
|
let result = await build.resolve(url, {
|
|
53
53
|
kind: 'import-rule',
|
|
54
|
-
//
|
|
55
|
-
|
|
56
|
-
resolveDir: build.initialOptions.absWorkingDir,
|
|
54
|
+
// Use the provided resolve directory from the custom Sass service if available
|
|
55
|
+
resolveDir: options.resolveDir ?? build.initialOptions.absWorkingDir,
|
|
57
56
|
});
|
|
58
|
-
//
|
|
59
|
-
if (
|
|
60
|
-
|
|
57
|
+
// If a resolve directory is provided, no additional speculative resolutions are required
|
|
58
|
+
if (options.resolveDir) {
|
|
59
|
+
return result;
|
|
60
|
+
}
|
|
61
|
+
// Workaround to support Yarn PnP and pnpm without access to the importer file from Sass
|
|
62
|
+
if (!result.path && options.previousResolvedModules?.size) {
|
|
63
|
+
for (const previous of options.previousResolvedModules) {
|
|
61
64
|
result = await build.resolve(url, {
|
|
62
65
|
kind: 'import-rule',
|
|
63
66
|
resolveDir: previous,
|
|
@@ -72,12 +75,42 @@ exports.SassStylesheetLanguage = Object.freeze({
|
|
|
72
75
|
return compileString(data, file, syntax, options, resolveUrl);
|
|
73
76
|
},
|
|
74
77
|
});
|
|
78
|
+
function parsePackageName(url) {
|
|
79
|
+
const parts = url.split('/');
|
|
80
|
+
const hasScope = parts.length >= 2 && parts[0].startsWith('@');
|
|
81
|
+
const [nameOrScope, nameOrFirstPath, ...pathPart] = parts;
|
|
82
|
+
const packageName = hasScope ? `${nameOrScope}/${nameOrFirstPath}` : nameOrScope;
|
|
83
|
+
return {
|
|
84
|
+
packageName,
|
|
85
|
+
get pathSegments() {
|
|
86
|
+
return !hasScope && nameOrFirstPath ? [nameOrFirstPath, ...pathPart] : pathPart;
|
|
87
|
+
},
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
class Cache extends Map {
|
|
91
|
+
async getOrCreate(key, creator) {
|
|
92
|
+
let value = this.get(key);
|
|
93
|
+
if (value === undefined) {
|
|
94
|
+
value = await creator();
|
|
95
|
+
this.set(key, value);
|
|
96
|
+
}
|
|
97
|
+
return value;
|
|
98
|
+
}
|
|
99
|
+
}
|
|
75
100
|
async function compileString(data, filePath, syntax, options, resolveUrl) {
|
|
76
101
|
// Lazily load Sass when a Sass file is found
|
|
77
102
|
if (sassWorkerPool === undefined) {
|
|
78
103
|
const sassService = await Promise.resolve().then(() => __importStar(require('../../sass/sass-service')));
|
|
79
104
|
sassWorkerPool = new sassService.SassWorkerImplementation(true);
|
|
80
105
|
}
|
|
106
|
+
// Cache is currently local to individual compile requests.
|
|
107
|
+
// Caching follows Sass behavior where a given url will always resolve to the same value
|
|
108
|
+
// regardless of its importer's path.
|
|
109
|
+
// A null value indicates that the cached resolution attempt failed to find a location and
|
|
110
|
+
// later stage resolution should be attempted. This avoids potentially expensive repeat
|
|
111
|
+
// failing resolution attempts.
|
|
112
|
+
const resolutionCache = new Cache();
|
|
113
|
+
const packageRootCache = new Cache();
|
|
81
114
|
const warnings = [];
|
|
82
115
|
try {
|
|
83
116
|
const { css, sourceMap, loadedUrls } = await sassWorkerPool.compileStringAsync(data, {
|
|
@@ -90,33 +123,30 @@ async function compileString(data, filePath, syntax, options, resolveUrl) {
|
|
|
90
123
|
quietDeps: true,
|
|
91
124
|
importers: [
|
|
92
125
|
{
|
|
93
|
-
findFileUrl:
|
|
94
|
-
|
|
126
|
+
findFileUrl: (url, options) => resolutionCache.getOrCreate(url, async () => {
|
|
127
|
+
const result = await resolveUrl(url, options);
|
|
95
128
|
if (result.path) {
|
|
96
129
|
return (0, node_url_1.pathToFileURL)(result.path);
|
|
97
130
|
}
|
|
98
131
|
// Check for package deep imports
|
|
99
|
-
const
|
|
100
|
-
|
|
101
|
-
|
|
102
|
-
const
|
|
103
|
-
|
|
104
|
-
|
|
105
|
-
return
|
|
106
|
-
}
|
|
107
|
-
//
|
|
108
|
-
//
|
|
109
|
-
|
|
110
|
-
|
|
111
|
-
|
|
112
|
-
|
|
113
|
-
packageResult = await resolveUrl(packageName + '/package.json', previousResolvedModules);
|
|
114
|
-
if (packageResult.path) {
|
|
115
|
-
return (0, node_url_1.pathToFileURL)((0, node_path_1.join)((0, node_path_1.dirname)(packageResult.path), !hasScope && nameOrFirstPath ? nameOrFirstPath : '', ...pathPart));
|
|
132
|
+
const { packageName, pathSegments } = parsePackageName(url);
|
|
133
|
+
// Caching package root locations is particularly beneficial for `@material/*` packages
|
|
134
|
+
// which extensively use deep imports.
|
|
135
|
+
const packageRoot = await packageRootCache.getOrCreate(packageName, async () => {
|
|
136
|
+
// Use the required presence of a package root `package.json` file to resolve the location
|
|
137
|
+
const packageResult = await resolveUrl(packageName + '/package.json', options);
|
|
138
|
+
return packageResult.path ? (0, node_path_1.dirname)(packageResult.path) : null;
|
|
139
|
+
});
|
|
140
|
+
// Package not found could be because of an error or the specifier is intended to be found
|
|
141
|
+
// via a later stage of the resolution process (`loadPaths`, etc.).
|
|
142
|
+
// Errors are reported after the full completion of the resolution process. Exceptions for
|
|
143
|
+
// not found packages should not be raised here.
|
|
144
|
+
if (packageRoot) {
|
|
145
|
+
return (0, node_url_1.pathToFileURL)((0, node_path_1.join)(packageRoot, ...pathSegments));
|
|
116
146
|
}
|
|
117
147
|
// Not found
|
|
118
148
|
return null;
|
|
119
|
-
},
|
|
149
|
+
}),
|
|
120
150
|
},
|
|
121
151
|
],
|
|
122
152
|
logger: {
|
|
@@ -166,4 +196,4 @@ function sourceMapToUrlComment(sourceMap, root) {
|
|
|
166
196
|
const urlSourceMap = Buffer.from(JSON.stringify(sourceMap), 'utf-8').toString('base64');
|
|
167
197
|
return `/*# sourceMappingURL=data:application/json;charset=utf-8;base64,${urlSourceMap} */`;
|
|
168
198
|
}
|
|
169
|
-
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sass-language.js","sourceRoot":"","sources":["../../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/stylesheets/sass-language.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 type { OnLoadResult, PartialMessage, ResolveResult } from 'esbuild';\nimport { dirname, join, relative } from 'node:path';\nimport { fileURLToPath, pathToFileURL } from 'node:url';\nimport type { CompileResult, Exception, Syntax } from 'sass';\nimport type {\n  FileImporterWithRequestContextOptions,\n  SassWorkerImplementation,\n} from '../../sass/sass-service';\nimport { StylesheetLanguage, StylesheetPluginOptions } from './stylesheet-plugin-factory';\n\nlet sassWorkerPool: SassWorkerImplementation | undefined;\n\nfunction isSassException(error: unknown): error is Exception {\n  return !!error && typeof error === 'object' && 'sassMessage' in error;\n}\n\nexport function shutdownSassWorkerPool(): void {\n  sassWorkerPool?.close();\n  sassWorkerPool = undefined;\n}\n\nexport const SassStylesheetLanguage = Object.freeze<StylesheetLanguage>({\n  name: 'sass',\n  componentFilter: /^s[ac]ss;/,\n  fileFilter: /\\.s[ac]ss$/,\n  process(data, file, format, options, build) {\n    const syntax = format === 'sass' ? 'indented' : 'scss';\n    const resolveUrl = async (url: string, previousResolvedModules?: Set<string>) => {\n      let result = await build.resolve(url, {\n        kind: 'import-rule',\n        // This should ideally be the directory of the importer file from Sass\n        // but that is not currently available from the Sass importer API.\n        resolveDir: build.initialOptions.absWorkingDir,\n      });\n\n      // Workaround to support Yarn PnP without access to the importer file from Sass\n      if (!result.path && previousResolvedModules?.size) {\n        for (const previous of previousResolvedModules) {\n          result = await build.resolve(url, {\n            kind: 'import-rule',\n            resolveDir: previous,\n          });\n          if (result.path) {\n            break;\n          }\n        }\n      }\n\n      return result;\n    };\n\n    return compileString(data, file, syntax, options, resolveUrl);\n  },\n});\n\nasync function compileString(\n  data: string,\n  filePath: string,\n  syntax: Syntax,\n  options: StylesheetPluginOptions,\n  resolveUrl: (url: string, previousResolvedModules?: Set<string>) => Promise<ResolveResult>,\n): Promise<OnLoadResult> {\n  // Lazily load Sass when a Sass file is found\n  if (sassWorkerPool === undefined) {\n    const sassService = await import('../../sass/sass-service');\n    sassWorkerPool = new sassService.SassWorkerImplementation(true);\n  }\n\n  const warnings: PartialMessage[] = [];\n  try {\n    const { css, sourceMap, loadedUrls } = await sassWorkerPool.compileStringAsync(data, {\n      url: pathToFileURL(filePath),\n      style: 'expanded',\n      syntax,\n      loadPaths: options.includePaths,\n      sourceMap: options.sourcemap,\n      sourceMapIncludeSources: options.sourcemap,\n      quietDeps: true,\n      importers: [\n        {\n          findFileUrl: async (\n            url,\n            { previousResolvedModules }: FileImporterWithRequestContextOptions,\n          ): Promise<URL | null> => {\n            let result = await resolveUrl(url);\n            if (result.path) {\n              return pathToFileURL(result.path);\n            }\n\n            // Check for package deep imports\n            const parts = url.split('/');\n            const hasScope = parts.length >= 2 && parts[0].startsWith('@');\n            const [nameOrScope, nameOrFirstPath, ...pathPart] = parts;\n            const packageName = hasScope ? `${nameOrScope}/${nameOrFirstPath}` : nameOrScope;\n\n            let packageResult = await resolveUrl(packageName + '/package.json');\n\n            if (packageResult.path) {\n              return pathToFileURL(\n                join(\n                  dirname(packageResult.path),\n                  !hasScope && nameOrFirstPath ? nameOrFirstPath : '',\n                  ...pathPart,\n                ),\n              );\n            }\n\n            // Check with Yarn PnP workaround using previous resolved modules.\n            // This is done last to avoid a performance penalty for common cases.\n\n            result = await resolveUrl(url, previousResolvedModules);\n            if (result.path) {\n              return pathToFileURL(result.path);\n            }\n\n            packageResult = await resolveUrl(\n              packageName + '/package.json',\n              previousResolvedModules,\n            );\n\n            if (packageResult.path) {\n              return pathToFileURL(\n                join(\n                  dirname(packageResult.path),\n                  !hasScope && nameOrFirstPath ? nameOrFirstPath : '',\n                  ...pathPart,\n                ),\n              );\n            }\n\n            // Not found\n            return null;\n          },\n        },\n      ],\n      logger: {\n        warn: (text, { deprecation, span }) => {\n          warnings.push({\n            text: deprecation ? 'Deprecation' : text,\n            location: span && {\n              file: span.url && fileURLToPath(span.url),\n              lineText: span.context,\n              // Sass line numbers are 0-based while esbuild's are 1-based\n              line: span.start.line + 1,\n              column: span.start.column,\n            },\n            notes: deprecation ? [{ text }] : undefined,\n          });\n        },\n      },\n    });\n\n    return {\n      loader: 'css',\n      contents: sourceMap ? `${css}\\n${sourceMapToUrlComment(sourceMap, dirname(filePath))}` : css,\n      watchFiles: loadedUrls.map((url) => fileURLToPath(url)),\n      warnings,\n    };\n  } catch (error) {\n    if (isSassException(error)) {\n      const file = error.span.url ? fileURLToPath(error.span.url) : undefined;\n\n      return {\n        loader: 'css',\n        errors: [\n          {\n            text: error.message,\n          },\n        ],\n        warnings,\n        watchFiles: file ? [file] : undefined,\n      };\n    }\n\n    throw error;\n  }\n}\n\nfunction sourceMapToUrlComment(\n  sourceMap: Exclude<CompileResult['sourceMap'], undefined>,\n  root: string,\n): string {\n  // Remove `file` protocol from all sourcemap sources and adjust to be relative to the input file.\n  // This allows esbuild to correctly process the paths.\n  sourceMap.sources = sourceMap.sources.map((source) => relative(root, fileURLToPath(source)));\n\n  const urlSourceMap = Buffer.from(JSON.stringify(sourceMap), 'utf-8').toString('base64');\n\n  return `/*# sourceMappingURL=data:application/json;charset=utf-8;base64,${urlSourceMap} */`;\n}\n"]}
|
|
199
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"sass-language.js","sourceRoot":"","sources":["../../../../../../../../../../packages/angular_devkit/build_angular/src/tools/esbuild/stylesheets/sass-language.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 type { OnLoadResult, PartialMessage, ResolveResult } from 'esbuild';\nimport { dirname, join, relative } from 'node:path';\nimport { fileURLToPath, pathToFileURL } from 'node:url';\nimport type { CompileResult, Exception, Syntax } from 'sass';\nimport type {\n  FileImporterWithRequestContextOptions,\n  SassWorkerImplementation,\n} from '../../sass/sass-service';\nimport { StylesheetLanguage, StylesheetPluginOptions } from './stylesheet-plugin-factory';\n\nlet sassWorkerPool: SassWorkerImplementation | undefined;\n\nfunction isSassException(error: unknown): error is Exception {\n  return !!error && typeof error === 'object' && 'sassMessage' in error;\n}\n\nexport function shutdownSassWorkerPool(): void {\n  sassWorkerPool?.close();\n  sassWorkerPool = undefined;\n}\n\nexport const SassStylesheetLanguage = Object.freeze<StylesheetLanguage>({\n  name: 'sass',\n  componentFilter: /^s[ac]ss;/,\n  fileFilter: /\\.s[ac]ss$/,\n  process(data, file, format, options, build) {\n    const syntax = format === 'sass' ? 'indented' : 'scss';\n    const resolveUrl = async (url: string, options: FileImporterWithRequestContextOptions) => {\n      let result = await build.resolve(url, {\n        kind: 'import-rule',\n        // Use the provided resolve directory from the custom Sass service if available\n        resolveDir: options.resolveDir ?? build.initialOptions.absWorkingDir,\n      });\n\n      // If a resolve directory is provided, no additional speculative resolutions are required\n      if (options.resolveDir) {\n        return result;\n      }\n\n      // Workaround to support Yarn PnP and pnpm without access to the importer file from Sass\n      if (!result.path && options.previousResolvedModules?.size) {\n        for (const previous of options.previousResolvedModules) {\n          result = await build.resolve(url, {\n            kind: 'import-rule',\n            resolveDir: previous,\n          });\n          if (result.path) {\n            break;\n          }\n        }\n      }\n\n      return result;\n    };\n\n    return compileString(data, file, syntax, options, resolveUrl);\n  },\n});\n\nfunction parsePackageName(url: string): { packageName: string; readonly pathSegments: string[] } {\n  const parts = url.split('/');\n  const hasScope = parts.length >= 2 && parts[0].startsWith('@');\n  const [nameOrScope, nameOrFirstPath, ...pathPart] = parts;\n  const packageName = hasScope ? `${nameOrScope}/${nameOrFirstPath}` : nameOrScope;\n\n  return {\n    packageName,\n    get pathSegments() {\n      return !hasScope && nameOrFirstPath ? [nameOrFirstPath, ...pathPart] : pathPart;\n    },\n  };\n}\n\nclass Cache<K, V> extends Map<K, V> {\n  async getOrCreate(key: K, creator: () => V | Promise<V>): Promise<V> {\n    let value = this.get(key);\n\n    if (value === undefined) {\n      value = await creator();\n      this.set(key, value);\n    }\n\n    return value;\n  }\n}\n\nasync function compileString(\n  data: string,\n  filePath: string,\n  syntax: Syntax,\n  options: StylesheetPluginOptions,\n  resolveUrl: (\n    url: string,\n    options: FileImporterWithRequestContextOptions,\n  ) => Promise<ResolveResult>,\n): Promise<OnLoadResult> {\n  // Lazily load Sass when a Sass file is found\n  if (sassWorkerPool === undefined) {\n    const sassService = await import('../../sass/sass-service');\n    sassWorkerPool = new sassService.SassWorkerImplementation(true);\n  }\n\n  // Cache is currently local to individual compile requests.\n  // Caching follows Sass behavior where a given url will always resolve to the same value\n  // regardless of its importer's path.\n  // A null value indicates that the cached resolution attempt failed to find a location and\n  // later stage resolution should be attempted. This avoids potentially expensive repeat\n  // failing resolution attempts.\n  const resolutionCache = new Cache<string, URL | null>();\n  const packageRootCache = new Cache<string, string | null>();\n\n  const warnings: PartialMessage[] = [];\n  try {\n    const { css, sourceMap, loadedUrls } = await sassWorkerPool.compileStringAsync(data, {\n      url: pathToFileURL(filePath),\n      style: 'expanded',\n      syntax,\n      loadPaths: options.includePaths,\n      sourceMap: options.sourcemap,\n      sourceMapIncludeSources: options.sourcemap,\n      quietDeps: true,\n      importers: [\n        {\n          findFileUrl: (url, options: FileImporterWithRequestContextOptions) =>\n            resolutionCache.getOrCreate(url, async () => {\n              const result = await resolveUrl(url, options);\n              if (result.path) {\n                return pathToFileURL(result.path);\n              }\n\n              // Check for package deep imports\n              const { packageName, pathSegments } = parsePackageName(url);\n\n              // Caching package root locations is particularly beneficial for `@material/*` packages\n              // which extensively use deep imports.\n              const packageRoot = await packageRootCache.getOrCreate(packageName, async () => {\n                // Use the required presence of a package root `package.json` file to resolve the location\n                const packageResult = await resolveUrl(packageName + '/package.json', options);\n\n                return packageResult.path ? dirname(packageResult.path) : null;\n              });\n\n              // Package not found could be because of an error or the specifier is intended to be found\n              // via a later stage of the resolution process (`loadPaths`, etc.).\n              // Errors are reported after the full completion of the resolution process. Exceptions for\n              // not found packages should not be raised here.\n              if (packageRoot) {\n                return pathToFileURL(join(packageRoot, ...pathSegments));\n              }\n\n              // Not found\n              return null;\n            }),\n        },\n      ],\n      logger: {\n        warn: (text, { deprecation, span }) => {\n          warnings.push({\n            text: deprecation ? 'Deprecation' : text,\n            location: span && {\n              file: span.url && fileURLToPath(span.url),\n              lineText: span.context,\n              // Sass line numbers are 0-based while esbuild's are 1-based\n              line: span.start.line + 1,\n              column: span.start.column,\n            },\n            notes: deprecation ? [{ text }] : undefined,\n          });\n        },\n      },\n    });\n\n    return {\n      loader: 'css',\n      contents: sourceMap ? `${css}\\n${sourceMapToUrlComment(sourceMap, dirname(filePath))}` : css,\n      watchFiles: loadedUrls.map((url) => fileURLToPath(url)),\n      warnings,\n    };\n  } catch (error) {\n    if (isSassException(error)) {\n      const file = error.span.url ? fileURLToPath(error.span.url) : undefined;\n\n      return {\n        loader: 'css',\n        errors: [\n          {\n            text: error.message,\n          },\n        ],\n        warnings,\n        watchFiles: file ? [file] : undefined,\n      };\n    }\n\n    throw error;\n  }\n}\n\nfunction sourceMapToUrlComment(\n  sourceMap: Exclude<CompileResult['sourceMap'], undefined>,\n  root: string,\n): string {\n  // Remove `file` protocol from all sourcemap sources and adjust to be relative to the input file.\n  // This allows esbuild to correctly process the paths.\n  sourceMap.sources = sourceMap.sources.map((source) => relative(root, fileURLToPath(source)));\n\n  const urlSourceMap = Buffer.from(JSON.stringify(sourceMap), 'utf-8').toString('base64');\n\n  return `/*# sourceMappingURL=data:application/json;charset=utf-8;base64,${urlSourceMap} */`;\n}\n"]}
|
|
@@ -0,0 +1,29 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* @license
|
|
3
|
+
* Copyright Google LLC All Rights Reserved.
|
|
4
|
+
*
|
|
5
|
+
* Use of this source code is governed by an MIT-style license that can be
|
|
6
|
+
* found in the LICENSE file at https://angular.io/license
|
|
7
|
+
*/
|
|
8
|
+
/**
|
|
9
|
+
* Scans a CSS or Sass file and locates all valid url function values as defined by the
|
|
10
|
+
* syntax specification.
|
|
11
|
+
* @param contents A string containing a CSS or Sass file to scan.
|
|
12
|
+
* @returns An iterable that yields each CSS url function value found.
|
|
13
|
+
*/
|
|
14
|
+
export declare function findUrls(contents: string): Iterable<{
|
|
15
|
+
start: number;
|
|
16
|
+
end: number;
|
|
17
|
+
value: string;
|
|
18
|
+
}>;
|
|
19
|
+
/**
|
|
20
|
+
* Scans a CSS or Sass file and locates all valid import/use directive values as defined by the
|
|
21
|
+
* syntax specification.
|
|
22
|
+
* @param contents A string containing a CSS or Sass file to scan.
|
|
23
|
+
* @returns An iterable that yields each CSS directive value found.
|
|
24
|
+
*/
|
|
25
|
+
export declare function findImports(contents: string): Iterable<{
|
|
26
|
+
start: number;
|
|
27
|
+
end: number;
|
|
28
|
+
specifier: string;
|
|
29
|
+
}>;
|
|
@@ -0,0 +1,248 @@
|
|
|
1
|
+
"use strict";
|
|
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
|
+
Object.defineProperty(exports, "__esModule", { value: true });
|
|
10
|
+
exports.findImports = exports.findUrls = void 0;
|
|
11
|
+
// TODO: Combine everything into a single pass lexer
|
|
12
|
+
/**
|
|
13
|
+
* Determines if a unicode code point is a CSS whitespace character.
|
|
14
|
+
* @param code The unicode code point to test.
|
|
15
|
+
* @returns true, if the code point is CSS whitespace; false, otherwise.
|
|
16
|
+
*/
|
|
17
|
+
function isWhitespace(code) {
|
|
18
|
+
// Based on https://www.w3.org/TR/css-syntax-3/#whitespace
|
|
19
|
+
switch (code) {
|
|
20
|
+
case 0x0009: // tab
|
|
21
|
+
case 0x0020: // space
|
|
22
|
+
case 0x000a: // line feed
|
|
23
|
+
case 0x000c: // form feed
|
|
24
|
+
case 0x000d: // carriage return
|
|
25
|
+
return true;
|
|
26
|
+
default:
|
|
27
|
+
return false;
|
|
28
|
+
}
|
|
29
|
+
}
|
|
30
|
+
/**
|
|
31
|
+
* Scans a CSS or Sass file and locates all valid url function values as defined by the
|
|
32
|
+
* syntax specification.
|
|
33
|
+
* @param contents A string containing a CSS or Sass file to scan.
|
|
34
|
+
* @returns An iterable that yields each CSS url function value found.
|
|
35
|
+
*/
|
|
36
|
+
function* findUrls(contents) {
|
|
37
|
+
let pos = 0;
|
|
38
|
+
let width = 1;
|
|
39
|
+
let current = -1;
|
|
40
|
+
const next = () => {
|
|
41
|
+
pos += width;
|
|
42
|
+
current = contents.codePointAt(pos) ?? -1;
|
|
43
|
+
width = current > 0xffff ? 2 : 1;
|
|
44
|
+
return current;
|
|
45
|
+
};
|
|
46
|
+
// Based on https://www.w3.org/TR/css-syntax-3/#consume-ident-like-token
|
|
47
|
+
while ((pos = contents.indexOf('url(', pos)) !== -1) {
|
|
48
|
+
// Set to position of the (
|
|
49
|
+
pos += 3;
|
|
50
|
+
width = 1;
|
|
51
|
+
// Consume all leading whitespace
|
|
52
|
+
while (isWhitespace(next())) {
|
|
53
|
+
/* empty */
|
|
54
|
+
}
|
|
55
|
+
// Initialize URL state
|
|
56
|
+
const url = { start: pos, end: -1, value: '' };
|
|
57
|
+
let complete = false;
|
|
58
|
+
// If " or ', then consume the value as a string
|
|
59
|
+
if (current === 0x0022 || current === 0x0027) {
|
|
60
|
+
const ending = current;
|
|
61
|
+
// Based on https://www.w3.org/TR/css-syntax-3/#consume-string-token
|
|
62
|
+
while (!complete) {
|
|
63
|
+
switch (next()) {
|
|
64
|
+
case -1: // EOF
|
|
65
|
+
return;
|
|
66
|
+
case 0x000a: // line feed
|
|
67
|
+
case 0x000c: // form feed
|
|
68
|
+
case 0x000d: // carriage return
|
|
69
|
+
// Invalid
|
|
70
|
+
complete = true;
|
|
71
|
+
break;
|
|
72
|
+
case 0x005c: // \ -- character escape
|
|
73
|
+
// If not EOF or newline, add the character after the escape
|
|
74
|
+
switch (next()) {
|
|
75
|
+
case -1:
|
|
76
|
+
return;
|
|
77
|
+
case 0x000a: // line feed
|
|
78
|
+
case 0x000c: // form feed
|
|
79
|
+
case 0x000d: // carriage return
|
|
80
|
+
// Skip when inside a string
|
|
81
|
+
break;
|
|
82
|
+
default:
|
|
83
|
+
// TODO: Handle hex escape codes
|
|
84
|
+
url.value += String.fromCodePoint(current);
|
|
85
|
+
break;
|
|
86
|
+
}
|
|
87
|
+
break;
|
|
88
|
+
case ending:
|
|
89
|
+
// Full string position should include the quotes for replacement
|
|
90
|
+
url.end = pos + 1;
|
|
91
|
+
complete = true;
|
|
92
|
+
yield url;
|
|
93
|
+
break;
|
|
94
|
+
default:
|
|
95
|
+
url.value += String.fromCodePoint(current);
|
|
96
|
+
break;
|
|
97
|
+
}
|
|
98
|
+
}
|
|
99
|
+
next();
|
|
100
|
+
continue;
|
|
101
|
+
}
|
|
102
|
+
// Based on https://www.w3.org/TR/css-syntax-3/#consume-url-token
|
|
103
|
+
while (!complete) {
|
|
104
|
+
switch (current) {
|
|
105
|
+
case -1: // EOF
|
|
106
|
+
return;
|
|
107
|
+
case 0x0022: // "
|
|
108
|
+
case 0x0027: // '
|
|
109
|
+
case 0x0028: // (
|
|
110
|
+
// Invalid
|
|
111
|
+
complete = true;
|
|
112
|
+
break;
|
|
113
|
+
case 0x0029: // )
|
|
114
|
+
// URL is valid and complete
|
|
115
|
+
url.end = pos;
|
|
116
|
+
complete = true;
|
|
117
|
+
break;
|
|
118
|
+
case 0x005c: // \ -- character escape
|
|
119
|
+
// If not EOF or newline, add the character after the escape
|
|
120
|
+
switch (next()) {
|
|
121
|
+
case -1: // EOF
|
|
122
|
+
return;
|
|
123
|
+
case 0x000a: // line feed
|
|
124
|
+
case 0x000c: // form feed
|
|
125
|
+
case 0x000d: // carriage return
|
|
126
|
+
// Invalid
|
|
127
|
+
complete = true;
|
|
128
|
+
break;
|
|
129
|
+
default:
|
|
130
|
+
// TODO: Handle hex escape codes
|
|
131
|
+
url.value += String.fromCodePoint(current);
|
|
132
|
+
break;
|
|
133
|
+
}
|
|
134
|
+
break;
|
|
135
|
+
default:
|
|
136
|
+
if (isWhitespace(current)) {
|
|
137
|
+
while (isWhitespace(next())) {
|
|
138
|
+
/* empty */
|
|
139
|
+
}
|
|
140
|
+
// Unescaped whitespace is only valid before the closing )
|
|
141
|
+
if (current === 0x0029) {
|
|
142
|
+
// URL is valid
|
|
143
|
+
url.end = pos;
|
|
144
|
+
}
|
|
145
|
+
complete = true;
|
|
146
|
+
}
|
|
147
|
+
else {
|
|
148
|
+
// Add the character to the url value
|
|
149
|
+
url.value += String.fromCodePoint(current);
|
|
150
|
+
}
|
|
151
|
+
break;
|
|
152
|
+
}
|
|
153
|
+
next();
|
|
154
|
+
}
|
|
155
|
+
// An end position indicates a URL was found
|
|
156
|
+
if (url.end !== -1) {
|
|
157
|
+
yield url;
|
|
158
|
+
}
|
|
159
|
+
}
|
|
160
|
+
}
|
|
161
|
+
exports.findUrls = findUrls;
|
|
162
|
+
/**
|
|
163
|
+
* Scans a CSS or Sass file and locates all valid import/use directive values as defined by the
|
|
164
|
+
* syntax specification.
|
|
165
|
+
* @param contents A string containing a CSS or Sass file to scan.
|
|
166
|
+
* @returns An iterable that yields each CSS directive value found.
|
|
167
|
+
*/
|
|
168
|
+
function* findImports(contents) {
|
|
169
|
+
yield* find(contents, '@import ');
|
|
170
|
+
yield* find(contents, '@use ');
|
|
171
|
+
}
|
|
172
|
+
exports.findImports = findImports;
|
|
173
|
+
/**
|
|
174
|
+
* Scans a CSS or Sass file and locates all valid function/directive values as defined by the
|
|
175
|
+
* syntax specification.
|
|
176
|
+
* @param contents A string containing a CSS or Sass file to scan.
|
|
177
|
+
* @param prefix The prefix to start a valid segment.
|
|
178
|
+
* @returns An iterable that yields each CSS url function value found.
|
|
179
|
+
*/
|
|
180
|
+
function* find(contents, prefix) {
|
|
181
|
+
let pos = 0;
|
|
182
|
+
let width = 1;
|
|
183
|
+
let current = -1;
|
|
184
|
+
const next = () => {
|
|
185
|
+
pos += width;
|
|
186
|
+
current = contents.codePointAt(pos) ?? -1;
|
|
187
|
+
width = current > 0xffff ? 2 : 1;
|
|
188
|
+
return current;
|
|
189
|
+
};
|
|
190
|
+
// Based on https://www.w3.org/TR/css-syntax-3/#consume-ident-like-token
|
|
191
|
+
while ((pos = contents.indexOf(prefix, pos)) !== -1) {
|
|
192
|
+
// Set to position of the last character in prefix
|
|
193
|
+
pos += prefix.length - 1;
|
|
194
|
+
width = 1;
|
|
195
|
+
// Consume all leading whitespace
|
|
196
|
+
while (isWhitespace(next())) {
|
|
197
|
+
/* empty */
|
|
198
|
+
}
|
|
199
|
+
// Initialize URL state
|
|
200
|
+
const url = { start: pos, end: -1, specifier: '' };
|
|
201
|
+
let complete = false;
|
|
202
|
+
// If " or ', then consume the value as a string
|
|
203
|
+
if (current === 0x0022 || current === 0x0027) {
|
|
204
|
+
const ending = current;
|
|
205
|
+
// Based on https://www.w3.org/TR/css-syntax-3/#consume-string-token
|
|
206
|
+
while (!complete) {
|
|
207
|
+
switch (next()) {
|
|
208
|
+
case -1: // EOF
|
|
209
|
+
return;
|
|
210
|
+
case 0x000a: // line feed
|
|
211
|
+
case 0x000c: // form feed
|
|
212
|
+
case 0x000d: // carriage return
|
|
213
|
+
// Invalid
|
|
214
|
+
complete = true;
|
|
215
|
+
break;
|
|
216
|
+
case 0x005c: // \ -- character escape
|
|
217
|
+
// If not EOF or newline, add the character after the escape
|
|
218
|
+
switch (next()) {
|
|
219
|
+
case -1:
|
|
220
|
+
return;
|
|
221
|
+
case 0x000a: // line feed
|
|
222
|
+
case 0x000c: // form feed
|
|
223
|
+
case 0x000d: // carriage return
|
|
224
|
+
// Skip when inside a string
|
|
225
|
+
break;
|
|
226
|
+
default:
|
|
227
|
+
// TODO: Handle hex escape codes
|
|
228
|
+
url.specifier += String.fromCodePoint(current);
|
|
229
|
+
break;
|
|
230
|
+
}
|
|
231
|
+
break;
|
|
232
|
+
case ending:
|
|
233
|
+
// Full string position should include the quotes for replacement
|
|
234
|
+
url.end = pos + 1;
|
|
235
|
+
complete = true;
|
|
236
|
+
yield url;
|
|
237
|
+
break;
|
|
238
|
+
default:
|
|
239
|
+
url.specifier += String.fromCodePoint(current);
|
|
240
|
+
break;
|
|
241
|
+
}
|
|
242
|
+
}
|
|
243
|
+
next();
|
|
244
|
+
continue;
|
|
245
|
+
}
|
|
246
|
+
}
|
|
247
|
+
}
|
|
248
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"lexer.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/tools/sass/lexer.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\n// TODO: Combine everything into a single pass lexer\n\n/**\n * Determines if a unicode code point is a CSS whitespace character.\n * @param code The unicode code point to test.\n * @returns true, if the code point is CSS whitespace; false, otherwise.\n */\nfunction isWhitespace(code: number): boolean {\n  // Based on https://www.w3.org/TR/css-syntax-3/#whitespace\n  switch (code) {\n    case 0x0009: // tab\n    case 0x0020: // space\n    case 0x000a: // line feed\n    case 0x000c: // form feed\n    case 0x000d: // carriage return\n      return true;\n    default:\n      return false;\n  }\n}\n\n/**\n * Scans a CSS or Sass file and locates all valid url function values as defined by the\n * syntax specification.\n * @param contents A string containing a CSS or Sass file to scan.\n * @returns An iterable that yields each CSS url function value found.\n */\nexport function* findUrls(\n  contents: string,\n): Iterable<{ start: number; end: number; value: string }> {\n  let pos = 0;\n  let width = 1;\n  let current = -1;\n  const next = () => {\n    pos += width;\n    current = contents.codePointAt(pos) ?? -1;\n    width = current > 0xffff ? 2 : 1;\n\n    return current;\n  };\n\n  // Based on https://www.w3.org/TR/css-syntax-3/#consume-ident-like-token\n  while ((pos = contents.indexOf('url(', pos)) !== -1) {\n    // Set to position of the (\n    pos += 3;\n    width = 1;\n\n    // Consume all leading whitespace\n    while (isWhitespace(next())) {\n      /* empty */\n    }\n\n    // Initialize URL state\n    const url = { start: pos, end: -1, value: '' };\n    let complete = false;\n\n    // If \" or ', then consume the value as a string\n    if (current === 0x0022 || current === 0x0027) {\n      const ending = current;\n      // Based on https://www.w3.org/TR/css-syntax-3/#consume-string-token\n      while (!complete) {\n        switch (next()) {\n          case -1: // EOF\n            return;\n          case 0x000a: // line feed\n          case 0x000c: // form feed\n          case 0x000d: // carriage return\n            // Invalid\n            complete = true;\n            break;\n          case 0x005c: // \\ -- character escape\n            // If not EOF or newline, add the character after the escape\n            switch (next()) {\n              case -1:\n                return;\n              case 0x000a: // line feed\n              case 0x000c: // form feed\n              case 0x000d: // carriage return\n                // Skip when inside a string\n                break;\n              default:\n                // TODO: Handle hex escape codes\n                url.value += String.fromCodePoint(current);\n                break;\n            }\n            break;\n          case ending:\n            // Full string position should include the quotes for replacement\n            url.end = pos + 1;\n            complete = true;\n            yield url;\n            break;\n          default:\n            url.value += String.fromCodePoint(current);\n            break;\n        }\n      }\n\n      next();\n      continue;\n    }\n\n    // Based on https://www.w3.org/TR/css-syntax-3/#consume-url-token\n    while (!complete) {\n      switch (current) {\n        case -1: // EOF\n          return;\n        case 0x0022: // \"\n        case 0x0027: // '\n        case 0x0028: // (\n          // Invalid\n          complete = true;\n          break;\n        case 0x0029: // )\n          // URL is valid and complete\n          url.end = pos;\n          complete = true;\n          break;\n        case 0x005c: // \\ -- character escape\n          // If not EOF or newline, add the character after the escape\n          switch (next()) {\n            case -1: // EOF\n              return;\n            case 0x000a: // line feed\n            case 0x000c: // form feed\n            case 0x000d: // carriage return\n              // Invalid\n              complete = true;\n              break;\n            default:\n              // TODO: Handle hex escape codes\n              url.value += String.fromCodePoint(current);\n              break;\n          }\n          break;\n        default:\n          if (isWhitespace(current)) {\n            while (isWhitespace(next())) {\n              /* empty */\n            }\n            // Unescaped whitespace is only valid before the closing )\n            if (current === 0x0029) {\n              // URL is valid\n              url.end = pos;\n            }\n            complete = true;\n          } else {\n            // Add the character to the url value\n            url.value += String.fromCodePoint(current);\n          }\n          break;\n      }\n      next();\n    }\n\n    // An end position indicates a URL was found\n    if (url.end !== -1) {\n      yield url;\n    }\n  }\n}\n\n/**\n * Scans a CSS or Sass file and locates all valid import/use directive values as defined by the\n * syntax specification.\n * @param contents A string containing a CSS or Sass file to scan.\n * @returns An iterable that yields each CSS directive value found.\n */\nexport function* findImports(\n  contents: string,\n): Iterable<{ start: number; end: number; specifier: string }> {\n  yield* find(contents, '@import ');\n  yield* find(contents, '@use ');\n}\n\n/**\n * Scans a CSS or Sass file and locates all valid function/directive values as defined by the\n * syntax specification.\n * @param contents A string containing a CSS or Sass file to scan.\n * @param prefix The prefix to start a valid segment.\n * @returns An iterable that yields each CSS url function value found.\n */\nfunction* find(\n  contents: string,\n  prefix: string,\n): Iterable<{ start: number; end: number; specifier: string }> {\n  let pos = 0;\n  let width = 1;\n  let current = -1;\n  const next = () => {\n    pos += width;\n    current = contents.codePointAt(pos) ?? -1;\n    width = current > 0xffff ? 2 : 1;\n\n    return current;\n  };\n\n  // Based on https://www.w3.org/TR/css-syntax-3/#consume-ident-like-token\n  while ((pos = contents.indexOf(prefix, pos)) !== -1) {\n    // Set to position of the last character in prefix\n    pos += prefix.length - 1;\n    width = 1;\n\n    // Consume all leading whitespace\n    while (isWhitespace(next())) {\n      /* empty */\n    }\n\n    // Initialize URL state\n    const url = { start: pos, end: -1, specifier: '' };\n    let complete = false;\n\n    // If \" or ', then consume the value as a string\n    if (current === 0x0022 || current === 0x0027) {\n      const ending = current;\n      // Based on https://www.w3.org/TR/css-syntax-3/#consume-string-token\n      while (!complete) {\n        switch (next()) {\n          case -1: // EOF\n            return;\n          case 0x000a: // line feed\n          case 0x000c: // form feed\n          case 0x000d: // carriage return\n            // Invalid\n            complete = true;\n            break;\n          case 0x005c: // \\ -- character escape\n            // If not EOF or newline, add the character after the escape\n            switch (next()) {\n              case -1:\n                return;\n              case 0x000a: // line feed\n              case 0x000c: // form feed\n              case 0x000d: // carriage return\n                // Skip when inside a string\n                break;\n              default:\n                // TODO: Handle hex escape codes\n                url.specifier += String.fromCodePoint(current);\n                break;\n            }\n            break;\n          case ending:\n            // Full string position should include the quotes for replacement\n            url.end = pos + 1;\n            complete = true;\n            yield url;\n            break;\n          default:\n            url.specifier += String.fromCodePoint(current);\n            break;\n        }\n      }\n\n      next();\n      continue;\n    }\n  }\n}\n"]}
|
|
@@ -6,7 +6,7 @@
|
|
|
6
6
|
* found in the LICENSE file at https://angular.io/license
|
|
7
7
|
*/
|
|
8
8
|
import { RawSourceMap } from '@ampproject/remapping';
|
|
9
|
-
import type {
|
|
9
|
+
import type { Importer, ImporterResult } from 'sass';
|
|
10
10
|
/**
|
|
11
11
|
* A preprocessed cache entry for the files and directories within a previously searched
|
|
12
12
|
* directory when performing Sass import resolution.
|
|
@@ -76,7 +76,10 @@ export declare class RelativeUrlRebasingImporter extends UrlRebasingImporter {
|
|
|
76
76
|
*/
|
|
77
77
|
export declare class ModuleUrlRebasingImporter extends RelativeUrlRebasingImporter {
|
|
78
78
|
private finder;
|
|
79
|
-
constructor(entryDirectory: string, directoryCache: Map<string, DirectoryEntry>, rebaseSourceMaps: Map<string, RawSourceMap> | undefined, finder:
|
|
79
|
+
constructor(entryDirectory: string, directoryCache: Map<string, DirectoryEntry>, rebaseSourceMaps: Map<string, RawSourceMap> | undefined, finder: (specifier: string, options: {
|
|
80
|
+
fromImport: boolean;
|
|
81
|
+
resolveDir?: string;
|
|
82
|
+
}) => URL | null);
|
|
80
83
|
canonicalize(url: string, options: {
|
|
81
84
|
fromImport: boolean;
|
|
82
85
|
}): URL | null;
|