import * as path from "path"; export interface MappingEntry { readonly pattern: string; readonly paths: ReadonlyArray; } export interface Paths { readonly [key: string]: ReadonlyArray; } /** * Converts an absolute baseUrl and paths to an array of absolute mapping entries. * The array is sorted by longest prefix. * Having an array with entries allows us to keep a sorting order rather than * sort by keys each time we use the mappings. * @param absoluteBaseUrl * @param paths * @param addMatchAll */ export function getAbsoluteMappingEntries( absoluteBaseUrl: string, paths: Paths, addMatchAll: boolean ): ReadonlyArray { // Resolve all paths to absolute form once here, and sort them by // longest prefix once here, this saves time on each request later. // We need to put them in an array to preserve the sorting order. const sortedKeys = sortByLongestPrefix(Object.keys(paths)); const absolutePaths: Array = []; for (const key of sortedKeys) { absolutePaths.push({ pattern: key, paths: paths[key].map((pathToResolve) => path.join(absoluteBaseUrl, pathToResolve) ), }); } // If there is no match-all path specified in the paths section of tsconfig, then try to match // all paths relative to baseUrl, this is how typescript works. if (!paths["*"] && addMatchAll) { absolutePaths.push({ pattern: "*", paths: [`${absoluteBaseUrl.replace(/\/$/, "")}/*`], }); } return absolutePaths; } /** * Sort path patterns. * If a module name can be matched with multiple patterns then pattern with the longest prefix will be picked. */ function sortByLongestPrefix(arr: Array): Array { return arr .concat() .sort((a: string, b: string) => getPrefixLength(b) - getPrefixLength(a)); } function getPrefixLength(pattern: string): number { const prefixLength = pattern.indexOf("*"); return pattern.substr(0, prefixLength).length; }