source: trip-planner-front/node_modules/@angular/material/schematics/ng-update/migrations/theming-api-v12/migration.js

Last change on this file was 6a3a178, checked in by Ema <ema_spirova@…>, 3 years ago

initial commit

  • Property mode set to 100644
File size: 50.2 KB
Line 
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 */
9Object.defineProperty(exports, "__esModule", { value: true });
10exports.migrateFileContent = void 0;
11const config_1 = require("./config");
12/** Possible pairs of comment characters in a Sass file. */
13const commentPairs = new Map([['/*', '*/'], ['//', '\n']]);
14/** Prefix for the placeholder that will be used to escape comments. */
15const commentPlaceholderStart = '__<<ngThemingMigrationEscapedComment';
16/** Suffix for the comment escape placeholder. */
17const commentPlaceholderEnd = '>>__';
18/**
19 * Migrates the content of a file to the new theming API. Note that this migration is using plain
20 * string manipulation, rather than the AST from PostCSS and the schematics string manipulation
21 * APIs, because it allows us to run it inside g3 and to avoid introducing new dependencies.
22 * @param fileContent Content of the file.
23 * @param oldMaterialPrefix Prefix with which the old Material imports should start.
24 * Has to end with a slash. E.g. if `@import '~@angular/material/theming'` should be
25 * matched, the prefix would be `~@angular/material/`.
26 * @param oldCdkPrefix Prefix with which the old CDK imports should start.
27 * Has to end with a slash. E.g. if `@import '~@angular/cdk/overlay'` should be
28 * matched, the prefix would be `~@angular/cdk/`.
29 * @param newMaterialImportPath New import to the Material theming API (e.g. `~@angular/material`).
30 * @param newCdkImportPath New import to the CDK Sass APIs (e.g. `~@angular/cdk`).
31 * @param excludedImports Pattern that can be used to exclude imports from being processed.
32 */
33function migrateFileContent(fileContent, oldMaterialPrefix, oldCdkPrefix, newMaterialImportPath, newCdkImportPath, extraMaterialSymbols = {}, excludedImports) {
34 let { content, placeholders } = escapeComments(fileContent);
35 const materialResults = detectImports(content, oldMaterialPrefix, excludedImports);
36 const cdkResults = detectImports(content, oldCdkPrefix, excludedImports);
37 // Try to migrate the symbols even if there are no imports. This is used
38 // to cover the case where the Components symbols were used transitively.
39 content = migrateCdkSymbols(content, newCdkImportPath, placeholders, cdkResults);
40 content = migrateMaterialSymbols(content, newMaterialImportPath, materialResults, placeholders, extraMaterialSymbols);
41 content = replaceRemovedVariables(content, config_1.removedMaterialVariables);
42 // We can assume that the migration has taken care of any Components symbols that were
43 // imported transitively so we can always drop the old imports. We also assume that imports
44 // to the new entry points have been added already.
45 if (materialResults.imports.length) {
46 content = replaceRemovedVariables(content, config_1.unprefixedRemovedVariables);
47 content = removeStrings(content, materialResults.imports);
48 }
49 if (cdkResults.imports.length) {
50 content = removeStrings(content, cdkResults.imports);
51 }
52 return restoreComments(content, placeholders);
53}
54exports.migrateFileContent = migrateFileContent;
55/**
56 * Counts the number of imports with a specific prefix and extracts their namespaces.
57 * @param content File content in which to look for imports.
58 * @param prefix Prefix that the imports should start with.
59 * @param excludedImports Pattern that can be used to exclude imports from being processed.
60 */
61function detectImports(content, prefix, excludedImports) {
62 if (prefix[prefix.length - 1] !== '/') {
63 // Some of the logic further down makes assumptions about the import depth.
64 throw Error(`Prefix "${prefix}" has to end in a slash.`);
65 }
66 // List of `@use` namespaces from which Angular CDK/Material APIs may be referenced.
67 // Since we know that the library doesn't have any name collisions, we can treat all of these
68 // namespaces as equivalent.
69 const namespaces = [];
70 const imports = [];
71 const pattern = new RegExp(`@(import|use) +['"]${escapeRegExp(prefix)}.*['"].*;?\n`, 'g');
72 let match = null;
73 while (match = pattern.exec(content)) {
74 const [fullImport, type] = match;
75 if (excludedImports === null || excludedImports === void 0 ? void 0 : excludedImports.test(fullImport)) {
76 continue;
77 }
78 if (type === 'use') {
79 const namespace = extractNamespaceFromUseStatement(fullImport);
80 if (namespaces.indexOf(namespace) === -1) {
81 namespaces.push(namespace);
82 }
83 }
84 imports.push(fullImport);
85 }
86 return { imports, namespaces };
87}
88/** Migrates the Material symbols in a file. */
89function migrateMaterialSymbols(content, importPath, detectedImports, commentPlaceholders, extraMaterialSymbols = {}) {
90 const initialContent = content;
91 const namespace = 'mat';
92 // Migrate the mixins.
93 const mixinsToUpdate = Object.assign(Object.assign({}, config_1.materialMixins), extraMaterialSymbols.mixins);
94 content = renameSymbols(content, mixinsToUpdate, detectedImports.namespaces, mixinKeyFormatter, getMixinValueFormatter(namespace));
95 // Migrate the functions.
96 const functionsToUpdate = Object.assign(Object.assign({}, config_1.materialFunctions), extraMaterialSymbols.functions);
97 content = renameSymbols(content, functionsToUpdate, detectedImports.namespaces, functionKeyFormatter, getFunctionValueFormatter(namespace));
98 // Migrate the variables.
99 const variablesToUpdate = Object.assign(Object.assign({}, config_1.materialVariables), extraMaterialSymbols.variables);
100 content = renameSymbols(content, variablesToUpdate, detectedImports.namespaces, variableKeyFormatter, getVariableValueFormatter(namespace));
101 if (content !== initialContent) {
102 // Add an import to the new API only if any of the APIs were being used.
103 content = insertUseStatement(content, importPath, namespace, commentPlaceholders);
104 }
105 return content;
106}
107/** Migrates the CDK symbols in a file. */
108function migrateCdkSymbols(content, importPath, commentPlaceholders, detectedImports) {
109 const initialContent = content;
110 const namespace = 'cdk';
111 // Migrate the mixins.
112 content = renameSymbols(content, config_1.cdkMixins, detectedImports.namespaces, mixinKeyFormatter, getMixinValueFormatter(namespace));
113 // Migrate the variables.
114 content = renameSymbols(content, config_1.cdkVariables, detectedImports.namespaces, variableKeyFormatter, getVariableValueFormatter(namespace));
115 // Previously the CDK symbols were exposed through `material/theming`, but now we have a
116 // dedicated entrypoint for the CDK. Only add an import for it if any of the symbols are used.
117 if (content !== initialContent) {
118 content = insertUseStatement(content, importPath, namespace, commentPlaceholders);
119 }
120 return content;
121}
122/**
123 * Renames all Sass symbols in a file based on a pre-defined mapping.
124 * @param content Content of a file to be migrated.
125 * @param mapping Mapping between symbol names and their replacements.
126 * @param namespaces Names to iterate over and pass to getKeyPattern.
127 * @param getKeyPattern Function used to turn each of the keys into a regex.
128 * @param formatValue Formats the value that will replace any matches of the pattern returned by
129 * `getKeyPattern`.
130 */
131function renameSymbols(content, mapping, namespaces, getKeyPattern, formatValue) {
132 // The null at the end is so that we make one last pass to cover non-namespaced symbols.
133 [...namespaces.slice(), null].forEach(namespace => {
134 Object.keys(mapping).forEach(key => {
135 const pattern = getKeyPattern(namespace, key);
136 // Sanity check since non-global regexes will only replace the first match.
137 if (pattern.flags.indexOf('g') === -1) {
138 throw Error('Replacement pattern must be global.');
139 }
140 content = content.replace(pattern, formatValue(mapping[key]));
141 });
142 });
143 return content;
144}
145/** Inserts an `@use` statement in a string. */
146function insertUseStatement(content, importPath, namespace, commentPlaceholders) {
147 // If the content already has the `@use` import, we don't need to add anything.
148 if (new RegExp(`@use +['"]${importPath}['"]`, 'g').test(content)) {
149 return content;
150 }
151 // Sass will throw an error if an `@use` statement comes after another statement. The safest way
152 // to ensure that we conform to that requirement is by always inserting our imports at the top
153 // of the file. Detecting where the user's content starts is tricky, because there are many
154 // different kinds of syntax we'd have to account for. One approach is to find the first `@import`
155 // and insert before it, but the problem is that Sass allows `@import` to be placed anywhere.
156 let newImportIndex = 0;
157 // One special case is if the file starts with a license header which we want to preserve on top.
158 if (content.trim().startsWith(commentPlaceholderStart)) {
159 const commentStartIndex = content.indexOf(commentPlaceholderStart);
160 newImportIndex = content.indexOf(commentPlaceholderEnd, commentStartIndex + 1) +
161 commentPlaceholderEnd.length;
162 // If the leading comment doesn't end with a newline,
163 // we need to insert the import at the next line.
164 if (!commentPlaceholders[content.slice(commentStartIndex, newImportIndex)].endsWith('\n')) {
165 newImportIndex = Math.max(newImportIndex, content.indexOf('\n', newImportIndex) + 1);
166 }
167 }
168 return content.slice(0, newImportIndex) + `@use '${importPath}' as ${namespace};\n` +
169 content.slice(newImportIndex);
170}
171/** Formats a migration key as a Sass mixin invocation. */
172function mixinKeyFormatter(namespace, name) {
173 // Note that adding a `(` at the end of the pattern would be more accurate, but mixin
174 // invocations don't necessarily have to include the parentheses. We could add `[(;]`,
175 // but then we won't know which character to include in the replacement string.
176 return new RegExp(`@include +${escapeRegExp((namespace ? namespace + '.' : '') + name)}`, 'g');
177}
178/** Returns a function that can be used to format a Sass mixin replacement. */
179function getMixinValueFormatter(namespace) {
180 // Note that adding a `(` at the end of the pattern would be more accurate,
181 // but mixin invocations don't necessarily have to include the parentheses.
182 return name => `@include ${namespace}.${name}`;
183}
184/** Formats a migration key as a Sass function invocation. */
185function functionKeyFormatter(namespace, name) {
186 const functionName = escapeRegExp(`${namespace ? namespace + '.' : ''}${name}(`);
187 return new RegExp(`(?<![-_a-zA-Z0-9])${functionName}`, 'g');
188}
189/** Returns a function that can be used to format a Sass function replacement. */
190function getFunctionValueFormatter(namespace) {
191 return name => `${namespace}.${name}(`;
192}
193/** Formats a migration key as a Sass variable. */
194function variableKeyFormatter(namespace, name) {
195 const variableName = escapeRegExp(`${namespace ? namespace + '.' : ''}$${name}`);
196 return new RegExp(`${variableName}(?![-_a-zA-Z0-9])`, 'g');
197}
198/** Returns a function that can be used to format a Sass variable replacement. */
199function getVariableValueFormatter(namespace) {
200 return name => `${namespace}.$${name}`;
201}
202/** Escapes special regex characters in a string. */
203function escapeRegExp(str) {
204 return str.replace(/([.*+?^=!:${}()|[\]\/\\])/g, '\\$1');
205}
206/** Removes all strings from another string. */
207function removeStrings(content, toRemove) {
208 return toRemove
209 .reduce((accumulator, current) => accumulator.replace(current, ''), content)
210 .replace(/^\s+/, '');
211}
212/** Parses out the namespace from a Sass `@use` statement. */
213function extractNamespaceFromUseStatement(fullImport) {
214 const closeQuoteIndex = Math.max(fullImport.lastIndexOf(`"`), fullImport.lastIndexOf(`'`));
215 if (closeQuoteIndex > -1) {
216 const asExpression = 'as ';
217 const asIndex = fullImport.indexOf(asExpression, closeQuoteIndex);
218 // If we found an ` as ` expression, we consider the rest of the text as the namespace.
219 if (asIndex > -1) {
220 return fullImport.slice(asIndex + asExpression.length).split(';')[0].trim();
221 }
222 // Otherwise the namespace is the name of the file that is being imported.
223 const lastSlashIndex = fullImport.lastIndexOf('/', closeQuoteIndex);
224 if (lastSlashIndex > -1) {
225 const fileName = fullImport.slice(lastSlashIndex + 1, closeQuoteIndex)
226 // Sass allows for leading underscores to be omitted and it technically supports .scss.
227 .replace(/^_|(\.import)?\.scss$|\.import$/g, '');
228 // Sass ignores `/index` and infers the namespace as the next segment in the path.
229 if (fileName === 'index') {
230 const nextSlashIndex = fullImport.lastIndexOf('/', lastSlashIndex - 1);
231 if (nextSlashIndex > -1) {
232 return fullImport.slice(nextSlashIndex + 1, lastSlashIndex);
233 }
234 }
235 else {
236 return fileName;
237 }
238 }
239 }
240 throw Error(`Could not extract namespace from import "${fullImport}".`);
241}
242/**
243 * Replaces variables that have been removed with their values.
244 * @param content Content of the file to be migrated.
245 * @param variables Mapping between variable names and their values.
246 */
247function replaceRemovedVariables(content, variables) {
248 Object.keys(variables).forEach(variableName => {
249 // Note that the pattern uses a negative lookahead to exclude
250 // variable assignments, because they can't be migrated.
251 const regex = new RegExp(`\\$${escapeRegExp(variableName)}(?!\\s+:|:)`, 'g');
252 content = content.replace(regex, variables[variableName]);
253 });
254 return content;
255}
256/**
257 * Replaces all of the comments in a Sass file with placeholders and
258 * returns the list of placeholders so they can be restored later.
259 */
260function escapeComments(content) {
261 const placeholders = {};
262 let commentCounter = 0;
263 let [openIndex, closeIndex] = findComment(content);
264 while (openIndex > -1 && closeIndex > -1) {
265 const placeholder = commentPlaceholderStart + (commentCounter++) + commentPlaceholderEnd;
266 placeholders[placeholder] = content.slice(openIndex, closeIndex);
267 content = content.slice(0, openIndex) + placeholder + content.slice(closeIndex);
268 [openIndex, closeIndex] = findComment(content);
269 }
270 return { content, placeholders };
271}
272/** Finds the start and end index of a comment in a file. */
273function findComment(content) {
274 // Add an extra new line at the end so that we can correctly capture single-line comments
275 // at the end of the file. It doesn't really matter that the end index will be out of bounds,
276 // because `String.prototype.slice` will clamp it to the string length.
277 content += '\n';
278 for (const [open, close] of commentPairs.entries()) {
279 const openIndex = content.indexOf(open);
280 if (openIndex > -1) {
281 const closeIndex = content.indexOf(close, openIndex + 1);
282 return closeIndex > -1 ? [openIndex, closeIndex + close.length] : [-1, -1];
283 }
284 }
285 return [-1, -1];
286}
287/** Restores the comments that have been escaped by `escapeComments`. */
288function restoreComments(content, placeholders) {
289 Object.keys(placeholders).forEach(key => content = content.replace(key, placeholders[key]));
290 return content;
291}
292//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"migration.js","sourceRoot":"","sources":["../../../../../../../../../src/material/schematics/ng-update/migrations/theming-api-v12/migration.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 {\n  materialMixins,\n  materialFunctions,\n  materialVariables,\n  cdkMixins,\n  cdkVariables,\n  removedMaterialVariables,\n  unprefixedRemovedVariables\n} from './config';\n\n/** The result of a search for imports and namespaces in a file. */\ninterface DetectImportResult {\n  imports: string[];\n  namespaces: string[];\n}\n\n/** Addition mixin and function names that can be updated when invoking migration directly. */\ninterface ExtraSymbols {\n  mixins?: Record<string, string>;\n  functions?: Record<string, string>;\n  variables?: Record<string, string>;\n}\n\n/** Possible pairs of comment characters in a Sass file. */\nconst commentPairs = new Map<string, string>([['/*', '*/'], ['//', '\\n']]);\n\n/** Prefix for the placeholder that will be used to escape comments. */\nconst commentPlaceholderStart = '__<<ngThemingMigrationEscapedComment';\n\n/** Suffix for the comment escape placeholder. */\nconst commentPlaceholderEnd = '>>__';\n\n/**\n * Migrates the content of a file to the new theming API. Note that this migration is using plain\n * string manipulation, rather than the AST from PostCSS and the schematics string manipulation\n * APIs, because it allows us to run it inside g3 and to avoid introducing new dependencies.\n * @param fileContent Content of the file.\n * @param oldMaterialPrefix Prefix with which the old Material imports should start.\n *   Has to end with a slash. E.g. if `@import '~@angular/material/theming'` should be\n *   matched, the prefix would be `~@angular/material/`.\n * @param oldCdkPrefix Prefix with which the old CDK imports should start.\n *   Has to end with a slash. E.g. if `@import '~@angular/cdk/overlay'` should be\n *   matched, the prefix would be `~@angular/cdk/`.\n * @param newMaterialImportPath New import to the Material theming API (e.g. `~@angular/material`).\n * @param newCdkImportPath New import to the CDK Sass APIs (e.g. `~@angular/cdk`).\n * @param excludedImports Pattern that can be used to exclude imports from being processed.\n */\nexport function migrateFileContent(fileContent: string,\n                                   oldMaterialPrefix: string,\n                                   oldCdkPrefix: string,\n                                   newMaterialImportPath: string,\n                                   newCdkImportPath: string,\n                                   extraMaterialSymbols: ExtraSymbols = {},\n                                   excludedImports?: RegExp): string {\n  let {content, placeholders} = escapeComments(fileContent);\n  const materialResults = detectImports(content, oldMaterialPrefix, excludedImports);\n  const cdkResults = detectImports(content, oldCdkPrefix, excludedImports);\n\n  // Try to migrate the symbols even if there are no imports. This is used\n  // to cover the case where the Components symbols were used transitively.\n  content = migrateCdkSymbols(content, newCdkImportPath, placeholders, cdkResults);\n  content = migrateMaterialSymbols(\n      content, newMaterialImportPath, materialResults, placeholders, extraMaterialSymbols);\n  content = replaceRemovedVariables(content, removedMaterialVariables);\n\n  // We can assume that the migration has taken care of any Components symbols that were\n  // imported transitively so we can always drop the old imports. We also assume that imports\n  // to the new entry points have been added already.\n  if (materialResults.imports.length) {\n    content = replaceRemovedVariables(content, unprefixedRemovedVariables);\n    content = removeStrings(content, materialResults.imports);\n  }\n\n  if (cdkResults.imports.length) {\n    content = removeStrings(content, cdkResults.imports);\n  }\n\n  return restoreComments(content, placeholders);\n}\n\n/**\n * Counts the number of imports with a specific prefix and extracts their namespaces.\n * @param content File content in which to look for imports.\n * @param prefix Prefix that the imports should start with.\n * @param excludedImports Pattern that can be used to exclude imports from being processed.\n */\nfunction detectImports(content: string, prefix: string,\n                       excludedImports?: RegExp): DetectImportResult {\n  if (prefix[prefix.length - 1] !== '/') {\n    // Some of the logic further down makes assumptions about the import depth.\n    throw Error(`Prefix \"${prefix}\" has to end in a slash.`);\n  }\n\n  // List of `@use` namespaces from which Angular CDK/Material APIs may be referenced.\n  // Since we know that the library doesn't have any name collisions, we can treat all of these\n  // namespaces as equivalent.\n  const namespaces: string[] = [];\n  const imports: string[] = [];\n  const pattern = new RegExp(`@(import|use) +['\"]${escapeRegExp(prefix)}.*['\"].*;?\\n`, 'g');\n  let match: RegExpExecArray | null = null;\n\n  while (match = pattern.exec(content)) {\n    const [fullImport, type] = match;\n\n    if (excludedImports?.test(fullImport)) {\n      continue;\n    }\n\n    if (type === 'use') {\n      const namespace = extractNamespaceFromUseStatement(fullImport);\n\n      if (namespaces.indexOf(namespace) === -1) {\n        namespaces.push(namespace);\n      }\n    }\n\n    imports.push(fullImport);\n  }\n\n  return {imports, namespaces};\n}\n\n/** Migrates the Material symbols in a file. */\nfunction migrateMaterialSymbols(content: string, importPath: string,\n                                detectedImports: DetectImportResult,\n                                commentPlaceholders: Record<string, string>,\n                                extraMaterialSymbols: ExtraSymbols = {}): string {\n  const initialContent = content;\n  const namespace = 'mat';\n\n  // Migrate the mixins.\n  const mixinsToUpdate = {...materialMixins, ...extraMaterialSymbols.mixins};\n  content = renameSymbols(content, mixinsToUpdate, detectedImports.namespaces, mixinKeyFormatter,\n    getMixinValueFormatter(namespace));\n\n  // Migrate the functions.\n  const functionsToUpdate = {...materialFunctions, ...extraMaterialSymbols.functions};\n  content = renameSymbols(content, functionsToUpdate, detectedImports.namespaces,\n    functionKeyFormatter, getFunctionValueFormatter(namespace));\n\n  // Migrate the variables.\n  const variablesToUpdate = {...materialVariables, ...extraMaterialSymbols.variables};\n  content = renameSymbols(content, variablesToUpdate, detectedImports.namespaces,\n    variableKeyFormatter, getVariableValueFormatter(namespace));\n\n  if (content !== initialContent) {\n    // Add an import to the new API only if any of the APIs were being used.\n    content = insertUseStatement(content, importPath, namespace, commentPlaceholders);\n  }\n\n  return content;\n}\n\n/** Migrates the CDK symbols in a file. */\nfunction migrateCdkSymbols(content: string, importPath: string,\n                           commentPlaceholders: Record<string, string>,\n                           detectedImports: DetectImportResult): string {\n  const initialContent = content;\n  const namespace = 'cdk';\n\n  // Migrate the mixins.\n  content = renameSymbols(content, cdkMixins, detectedImports.namespaces, mixinKeyFormatter,\n    getMixinValueFormatter(namespace));\n\n  // Migrate the variables.\n  content = renameSymbols(content, cdkVariables, detectedImports.namespaces, variableKeyFormatter,\n    getVariableValueFormatter(namespace));\n\n  // Previously the CDK symbols were exposed through `material/theming`, but now we have a\n  // dedicated entrypoint for the CDK. Only add an import for it if any of the symbols are used.\n  if (content !== initialContent) {\n    content = insertUseStatement(content, importPath, namespace, commentPlaceholders);\n  }\n\n  return content;\n}\n\n/**\n * Renames all Sass symbols in a file based on a pre-defined mapping.\n * @param content Content of a file to be migrated.\n * @param mapping Mapping between symbol names and their replacements.\n * @param namespaces Names to iterate over and pass to getKeyPattern.\n * @param getKeyPattern Function used to turn each of the keys into a regex.\n * @param formatValue Formats the value that will replace any matches of the pattern returned by\n *  `getKeyPattern`.\n */\nfunction renameSymbols(content: string,\n                       mapping: Record<string, string>,\n                       namespaces: string[],\n                       getKeyPattern: (namespace: string|null, key: string) => RegExp,\n                       formatValue: (key: string) => string): string {\n  // The null at the end is so that we make one last pass to cover non-namespaced symbols.\n  [...namespaces.slice(), null].forEach(namespace => {\n    Object.keys(mapping).forEach(key => {\n      const pattern = getKeyPattern(namespace, key);\n\n      // Sanity check since non-global regexes will only replace the first match.\n      if (pattern.flags.indexOf('g') === -1) {\n        throw Error('Replacement pattern must be global.');\n      }\n\n      content = content.replace(pattern, formatValue(mapping[key]));\n    });\n  });\n\n  return content;\n}\n\n/** Inserts an `@use` statement in a string. */\nfunction insertUseStatement(content: string, importPath: string, namespace: string,\n                            commentPlaceholders: Record<string, string>): string {\n  // If the content already has the `@use` import, we don't need to add anything.\n  if (new RegExp(`@use +['\"]${importPath}['\"]`, 'g').test(content)) {\n    return content;\n  }\n\n  // Sass will throw an error if an `@use` statement comes after another statement. The safest way\n  // to ensure that we conform to that requirement is by always inserting our imports at the top\n  // of the file. Detecting where the user's content starts is tricky, because there are many\n  // different kinds of syntax we'd have to account for. One approach is to find the first `@import`\n  // and insert before it, but the problem is that Sass allows `@import` to be placed anywhere.\n  let newImportIndex = 0;\n\n  // One special case is if the file starts with a license header which we want to preserve on top.\n  if (content.trim().startsWith(commentPlaceholderStart)) {\n    const commentStartIndex = content.indexOf(commentPlaceholderStart);\n    newImportIndex = content.indexOf(commentPlaceholderEnd, commentStartIndex + 1) +\n        commentPlaceholderEnd.length;\n    // If the leading comment doesn't end with a newline,\n    // we need to insert the import at the next line.\n    if (!commentPlaceholders[content.slice(commentStartIndex, newImportIndex)].endsWith('\\n')) {\n      newImportIndex = Math.max(newImportIndex, content.indexOf('\\n', newImportIndex) + 1);\n    }\n  }\n\n  return content.slice(0, newImportIndex) + `@use '${importPath}' as ${namespace};\\n` +\n         content.slice(newImportIndex);\n}\n\n/** Formats a migration key as a Sass mixin invocation. */\nfunction mixinKeyFormatter(namespace: string|null, name: string): RegExp {\n  // Note that adding a `(` at the end of the pattern would be more accurate, but mixin\n  // invocations don't necessarily have to include the parentheses. We could add `[(;]`,\n  // but then we won't know which character to include in the replacement string.\n  return new RegExp(`@include +${escapeRegExp((namespace ? namespace + '.' : '') + name)}`, 'g');\n}\n\n/** Returns a function that can be used to format a Sass mixin replacement. */\nfunction getMixinValueFormatter(namespace: string): (name: string) => string {\n  // Note that adding a `(` at the end of the pattern would be more accurate,\n  // but mixin invocations don't necessarily have to include the parentheses.\n  return name => `@include ${namespace}.${name}`;\n}\n\n/** Formats a migration key as a Sass function invocation. */\nfunction functionKeyFormatter(namespace: string|null, name: string): RegExp {\n  const functionName = escapeRegExp(`${namespace ? namespace + '.' : ''}${name}(`);\n  return new RegExp(`(?<![-_a-zA-Z0-9])${functionName}`, 'g');\n}\n\n/** Returns a function that can be used to format a Sass function replacement. */\nfunction getFunctionValueFormatter(namespace: string): (name: string) => string {\n  return name => `${namespace}.${name}(`;\n}\n\n/** Formats a migration key as a Sass variable. */\nfunction variableKeyFormatter(namespace: string|null, name: string): RegExp {\n  const variableName = escapeRegExp(`${namespace ? namespace + '.' : ''}$${name}`);\n  return new RegExp(`${variableName}(?![-_a-zA-Z0-9])`, 'g');\n}\n\n/** Returns a function that can be used to format a Sass variable replacement. */\nfunction getVariableValueFormatter(namespace: string): (name: string) => string {\n  return name => `${namespace}.$${name}`;\n}\n\n/** Escapes special regex characters in a string. */\nfunction escapeRegExp(str: string): string {\n  return str.replace(/([.*+?^=!:${}()|[\\]\\/\\\\])/g, '\\\\$1');\n}\n\n/** Removes all strings from another string. */\nfunction removeStrings(content: string, toRemove: string[]): string {\n  return toRemove\n    .reduce((accumulator, current) => accumulator.replace(current, ''), content)\n    .replace(/^\\s+/, '');\n}\n\n/** Parses out the namespace from a Sass `@use` statement. */\nfunction extractNamespaceFromUseStatement(fullImport: string): string {\n  const closeQuoteIndex = Math.max(fullImport.lastIndexOf(`\"`), fullImport.lastIndexOf(`'`));\n\n  if (closeQuoteIndex > -1) {\n    const asExpression = 'as ';\n    const asIndex = fullImport.indexOf(asExpression, closeQuoteIndex);\n\n    // If we found an ` as ` expression, we consider the rest of the text as the namespace.\n    if (asIndex > -1) {\n      return fullImport.slice(asIndex + asExpression.length).split(';')[0].trim();\n    }\n\n    // Otherwise the namespace is the name of the file that is being imported.\n    const lastSlashIndex = fullImport.lastIndexOf('/', closeQuoteIndex);\n\n    if (lastSlashIndex > -1) {\n      const fileName = fullImport.slice(lastSlashIndex + 1, closeQuoteIndex)\n        // Sass allows for leading underscores to be omitted and it technically supports .scss.\n        .replace(/^_|(\\.import)?\\.scss$|\\.import$/g, '');\n\n      // Sass ignores `/index` and infers the namespace as the next segment in the path.\n      if (fileName === 'index') {\n        const nextSlashIndex = fullImport.lastIndexOf('/', lastSlashIndex - 1);\n\n        if (nextSlashIndex > -1) {\n          return fullImport.slice(nextSlashIndex + 1, lastSlashIndex);\n        }\n      } else {\n        return fileName;\n      }\n    }\n  }\n\n  throw Error(`Could not extract namespace from import \"${fullImport}\".`);\n}\n\n/**\n * Replaces variables that have been removed with their values.\n * @param content Content of the file to be migrated.\n * @param variables Mapping between variable names and their values.\n */\nfunction replaceRemovedVariables(content: string, variables: Record<string, string>): string {\n  Object.keys(variables).forEach(variableName => {\n    // Note that the pattern uses a negative lookahead to exclude\n    // variable assignments, because they can't be migrated.\n    const regex = new RegExp(`\\\\$${escapeRegExp(variableName)}(?!\\\\s+:|:)`, 'g');\n    content = content.replace(regex, variables[variableName]);\n  });\n\n  return content;\n}\n\n\n/**\n * Replaces all of the comments in a Sass file with placeholders and\n * returns the list of placeholders so they can be restored later.\n */\nfunction escapeComments(content: string): {content: string, placeholders: Record<string, string>} {\n  const placeholders: Record<string, string> = {};\n  let commentCounter = 0;\n  let [openIndex, closeIndex] = findComment(content);\n\n  while (openIndex > -1 && closeIndex > -1) {\n    const placeholder = commentPlaceholderStart + (commentCounter++) + commentPlaceholderEnd;\n    placeholders[placeholder] = content.slice(openIndex, closeIndex);\n    content = content.slice(0, openIndex) + placeholder + content.slice(closeIndex);\n    [openIndex, closeIndex] = findComment(content);\n  }\n\n  return {content, placeholders};\n}\n\n/** Finds the start and end index of a comment in a file. */\nfunction findComment(content: string): [openIndex: number, closeIndex: number] {\n  // Add an extra new line at the end so that we can correctly capture single-line comments\n  // at the end of the file. It doesn't really matter that the end index will be out of bounds,\n  // because `String.prototype.slice` will clamp it to the string length.\n  content += '\\n';\n\n  for (const [open, close] of commentPairs.entries()) {\n    const openIndex = content.indexOf(open);\n\n    if (openIndex > -1) {\n      const closeIndex = content.indexOf(close, openIndex + 1);\n      return closeIndex > -1 ? [openIndex, closeIndex + close.length] : [-1, -1];\n    }\n  }\n\n  return [-1, -1];\n}\n\n/** Restores the comments that have been escaped by `escapeComments`. */\nfunction restoreComments(content: string, placeholders: Record<string, string>): string {\n  Object.keys(placeholders).forEach(key => content = content.replace(key, placeholders[key]));\n  return content;\n}\n"]}
Note: See TracBrowser for help on using the repository browser.