UNPKG

25.4 kBJavaScriptView Raw
1/**
2 * @license
3 * Copyright Google Inc. 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(function (factory) {
9 if (typeof module === "object" && typeof module.exports === "object") {
10 var v = factory(require, exports);
11 if (v !== undefined) module.exports = v;
12 }
13 else if (typeof define === "function" && define.amd) {
14 define("@angular/compiler-cli/src/ngtsc/resource_loader", ["require", "exports", "tslib", "typescript", "@angular/compiler-cli/src/ngtsc/file_system", "@angular/compiler-cli/src/ngtsc/util/src/typescript"], factory);
15 }
16})(function (require, exports) {
17 "use strict";
18 Object.defineProperty(exports, "__esModule", { value: true });
19 var tslib_1 = require("tslib");
20 var ts = require("typescript");
21 var file_system_1 = require("@angular/compiler-cli/src/ngtsc/file_system");
22 var typescript_1 = require("@angular/compiler-cli/src/ngtsc/util/src/typescript");
23 var CSS_PREPROCESSOR_EXT = /(\.scss|\.less|\.styl)$/;
24 /**
25 * `ResourceLoader` which delegates to a `CompilerHost` resource loading method.
26 */
27 var HostResourceLoader = /** @class */ (function () {
28 function HostResourceLoader(host, options) {
29 this.host = host;
30 this.options = options;
31 this.cache = new Map();
32 this.fetching = new Map();
33 this.canPreload = !!this.host.readResource;
34 this.rootDirs = typescript_1.getRootDirs(host, options);
35 }
36 /**
37 * Resolve the url of a resource relative to the file that contains the reference to it.
38 * The return value of this method can be used in the `load()` and `preload()` methods.
39 *
40 * Uses the provided CompilerHost if it supports mapping resources to filenames.
41 * Otherwise, uses a fallback mechanism that searches the module resolution candidates.
42 *
43 * @param url The, possibly relative, url of the resource.
44 * @param fromFile The path to the file that contains the URL of the resource.
45 * @returns A resolved url of resource.
46 * @throws An error if the resource cannot be resolved.
47 */
48 HostResourceLoader.prototype.resolve = function (url, fromFile) {
49 var resolvedUrl = null;
50 if (this.host.resourceNameToFileName) {
51 resolvedUrl = this.host.resourceNameToFileName(url, fromFile);
52 }
53 else {
54 resolvedUrl = this.fallbackResolve(url, fromFile);
55 }
56 if (resolvedUrl === null) {
57 throw new Error("HostResourceResolver: could not resolve " + url + " in context of " + fromFile + ")");
58 }
59 return resolvedUrl;
60 };
61 /**
62 * Preload the specified resource, asynchronously.
63 *
64 * Once the resource is loaded, its value is cached so it can be accessed synchronously via the
65 * `load()` method.
66 *
67 * @param resolvedUrl The url (resolved by a call to `resolve()`) of the resource to preload.
68 * @returns A Promise that is resolved once the resource has been loaded or `undefined` if the
69 * file has already been loaded.
70 * @throws An Error if pre-loading is not available.
71 */
72 HostResourceLoader.prototype.preload = function (resolvedUrl) {
73 var _this = this;
74 if (!this.host.readResource) {
75 throw new Error('HostResourceLoader: the CompilerHost provided does not support pre-loading resources.');
76 }
77 if (this.cache.has(resolvedUrl)) {
78 return undefined;
79 }
80 else if (this.fetching.has(resolvedUrl)) {
81 return this.fetching.get(resolvedUrl);
82 }
83 var result = this.host.readResource(resolvedUrl);
84 if (typeof result === 'string') {
85 this.cache.set(resolvedUrl, result);
86 return undefined;
87 }
88 else {
89 var fetchCompletion = result.then(function (str) {
90 _this.fetching.delete(resolvedUrl);
91 _this.cache.set(resolvedUrl, str);
92 });
93 this.fetching.set(resolvedUrl, fetchCompletion);
94 return fetchCompletion;
95 }
96 };
97 /**
98 * Load the resource at the given url, synchronously.
99 *
100 * The contents of the resource may have been cached by a previous call to `preload()`.
101 *
102 * @param resolvedUrl The url (resolved by a call to `resolve()`) of the resource to load.
103 * @returns The contents of the resource.
104 */
105 HostResourceLoader.prototype.load = function (resolvedUrl) {
106 if (this.cache.has(resolvedUrl)) {
107 return this.cache.get(resolvedUrl);
108 }
109 var result = this.host.readResource ? this.host.readResource(resolvedUrl) :
110 this.host.readFile(resolvedUrl);
111 if (typeof result !== 'string') {
112 throw new Error("HostResourceLoader: loader(" + resolvedUrl + ") returned a Promise");
113 }
114 this.cache.set(resolvedUrl, result);
115 return result;
116 };
117 /**
118 * Attempt to resolve `url` in the context of `fromFile`, while respecting the rootDirs
119 * option from the tsconfig. First, normalize the file name.
120 */
121 HostResourceLoader.prototype.fallbackResolve = function (url, fromFile) {
122 var e_1, _a;
123 var candidateLocations;
124 if (url.startsWith('/')) {
125 // This path is not really an absolute path, but instead the leading '/' means that it's
126 // rooted in the project rootDirs. So look for it according to the rootDirs.
127 candidateLocations = this.getRootedCandidateLocations(url);
128 }
129 else {
130 // This path is a "relative" path and can be resolved as such. To make this easier on the
131 // downstream resolver, the './' prefix is added if missing to distinguish these paths from
132 // absolute node_modules paths.
133 if (!url.startsWith('.')) {
134 url = "./" + url;
135 }
136 candidateLocations = this.getResolvedCandidateLocations(url, fromFile);
137 }
138 try {
139 for (var candidateLocations_1 = tslib_1.__values(candidateLocations), candidateLocations_1_1 = candidateLocations_1.next(); !candidateLocations_1_1.done; candidateLocations_1_1 = candidateLocations_1.next()) {
140 var candidate = candidateLocations_1_1.value;
141 if (this.host.fileExists(candidate)) {
142 return candidate;
143 }
144 else if (CSS_PREPROCESSOR_EXT.test(candidate)) {
145 /**
146 * If the user specified styleUrl points to *.scss, but the Sass compiler was run before
147 * Angular, then the resource may have been generated as *.css. Simply try the resolution
148 * again.
149 */
150 var cssFallbackUrl = candidate.replace(CSS_PREPROCESSOR_EXT, '.css');
151 if (this.host.fileExists(cssFallbackUrl)) {
152 return cssFallbackUrl;
153 }
154 }
155 }
156 }
157 catch (e_1_1) { e_1 = { error: e_1_1 }; }
158 finally {
159 try {
160 if (candidateLocations_1_1 && !candidateLocations_1_1.done && (_a = candidateLocations_1.return)) _a.call(candidateLocations_1);
161 }
162 finally { if (e_1) throw e_1.error; }
163 }
164 return null;
165 };
166 HostResourceLoader.prototype.getRootedCandidateLocations = function (url) {
167 // The path already starts with '/', so add a '.' to make it relative.
168 var segment = ('.' + url);
169 return this.rootDirs.map(function (rootDir) { return file_system_1.join(rootDir, segment); });
170 };
171 /**
172 * TypeScript provides utilities to resolve module names, but not resource files (which aren't
173 * a part of the ts.Program). However, TypeScript's module resolution can be used creatively
174 * to locate where resource files should be expected to exist. Since module resolution returns
175 * a list of file names that were considered, the loader can enumerate the possible locations
176 * for the file by setting up a module resolution for it that will fail.
177 */
178 HostResourceLoader.prototype.getResolvedCandidateLocations = function (url, fromFile) {
179 // clang-format off
180 var failedLookup = ts.resolveModuleName(url + '.$ngresource$', fromFile, this.options, this.host);
181 // clang-format on
182 if (failedLookup.failedLookupLocations === undefined) {
183 throw new Error("Internal error: expected to find failedLookupLocations during resolution of resource '" + url + "' in context of " + fromFile);
184 }
185 return failedLookup.failedLookupLocations
186 .filter(function (candidate) { return candidate.endsWith('.$ngresource$.ts'); })
187 .map(function (candidate) { return candidate.replace(/\.\$ngresource\$\.ts$/, ''); });
188 };
189 return HostResourceLoader;
190 }());
191 exports.HostResourceLoader = HostResourceLoader;
192});
193//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"resource_loader.js","sourceRoot":"","sources":["../../../../../../../packages/compiler-cli/src/ngtsc/resource_loader.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * @license\n * Copyright Google Inc. 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 * as ts from 'typescript';\n\nimport {CompilerHost} from '../transformers/api';\n\nimport {ResourceLoader} from './annotations';\nimport {AbsoluteFsPath, PathSegment, join} from './file_system';\nimport {getRootDirs} from './util/src/typescript';\n\nconst CSS_PREPROCESSOR_EXT = /(\\.scss|\\.less|\\.styl)$/;\n\n/**\n * `ResourceLoader` which delegates to a `CompilerHost` resource loading method.\n */\nexport class HostResourceLoader implements ResourceLoader {\n  private cache = new Map<string, string>();\n  private fetching = new Map<string, Promise<void>>();\n\n  private rootDirs: AbsoluteFsPath[];\n\n  canPreload = !!this.host.readResource;\n\n  constructor(private host: CompilerHost, private options: ts.CompilerOptions) {\n    this.rootDirs = getRootDirs(host, options);\n  }\n\n  /**\n   * Resolve the url of a resource relative to the file that contains the reference to it.\n   * The return value of this method can be used in the `load()` and `preload()` methods.\n   *\n   * Uses the provided CompilerHost if it supports mapping resources to filenames.\n   * Otherwise, uses a fallback mechanism that searches the module resolution candidates.\n   *\n   * @param url The, possibly relative, url of the resource.\n   * @param fromFile The path to the file that contains the URL of the resource.\n   * @returns A resolved url of resource.\n   * @throws An error if the resource cannot be resolved.\n   */\n  resolve(url: string, fromFile: string): string {\n    let resolvedUrl: string|null = null;\n    if (this.host.resourceNameToFileName) {\n      resolvedUrl = this.host.resourceNameToFileName(url, fromFile);\n    } else {\n      resolvedUrl = this.fallbackResolve(url, fromFile);\n    }\n    if (resolvedUrl === null) {\n      throw new Error(`HostResourceResolver: could not resolve ${url} in context of ${fromFile})`);\n    }\n    return resolvedUrl;\n  }\n\n  /**\n   * Preload the specified resource, asynchronously.\n   *\n   * Once the resource is loaded, its value is cached so it can be accessed synchronously via the\n   * `load()` method.\n   *\n   * @param resolvedUrl The url (resolved by a call to `resolve()`) of the resource to preload.\n   * @returns A Promise that is resolved once the resource has been loaded or `undefined` if the\n   * file has already been loaded.\n   * @throws An Error if pre-loading is not available.\n   */\n  preload(resolvedUrl: string): Promise<void>|undefined {\n    if (!this.host.readResource) {\n      throw new Error(\n          'HostResourceLoader: the CompilerHost provided does not support pre-loading resources.');\n    }\n    if (this.cache.has(resolvedUrl)) {\n      return undefined;\n    } else if (this.fetching.has(resolvedUrl)) {\n      return this.fetching.get(resolvedUrl);\n    }\n\n    const result = this.host.readResource(resolvedUrl);\n    if (typeof result === 'string') {\n      this.cache.set(resolvedUrl, result);\n      return undefined;\n    } else {\n      const fetchCompletion = result.then(str => {\n        this.fetching.delete(resolvedUrl);\n        this.cache.set(resolvedUrl, str);\n      });\n      this.fetching.set(resolvedUrl, fetchCompletion);\n      return fetchCompletion;\n    }\n  }\n\n  /**\n   * Load the resource at the given url, synchronously.\n   *\n   * The contents of the resource may have been cached by a previous call to `preload()`.\n   *\n   * @param resolvedUrl The url (resolved by a call to `resolve()`) of the resource to load.\n   * @returns The contents of the resource.\n   */\n  load(resolvedUrl: string): string {\n    if (this.cache.has(resolvedUrl)) {\n      return this.cache.get(resolvedUrl) !;\n    }\n\n    const result = this.host.readResource ? this.host.readResource(resolvedUrl) :\n                                            this.host.readFile(resolvedUrl);\n    if (typeof result !== 'string') {\n      throw new Error(`HostResourceLoader: loader(${resolvedUrl}) returned a Promise`);\n    }\n    this.cache.set(resolvedUrl, result);\n    return result;\n  }\n\n  /**\n   * Attempt to resolve `url` in the context of `fromFile`, while respecting the rootDirs\n   * option from the tsconfig. First, normalize the file name.\n   */\n  private fallbackResolve(url: string, fromFile: string): string|null {\n    let candidateLocations: string[];\n    if (url.startsWith('/')) {\n      // This path is not really an absolute path, but instead the leading '/' means that it's\n      // rooted in the project rootDirs. So look for it according to the rootDirs.\n      candidateLocations = this.getRootedCandidateLocations(url);\n    } else {\n      // This path is a \"relative\" path and can be resolved as such. To make this easier on the\n      // downstream resolver, the './' prefix is added if missing to distinguish these paths from\n      // absolute node_modules paths.\n      if (!url.startsWith('.')) {\n        url = `./${url}`;\n      }\n      candidateLocations = this.getResolvedCandidateLocations(url, fromFile);\n    }\n\n    for (const candidate of candidateLocations) {\n      if (this.host.fileExists(candidate)) {\n        return candidate;\n      } else if (CSS_PREPROCESSOR_EXT.test(candidate)) {\n        /**\n         * If the user specified styleUrl points to *.scss, but the Sass compiler was run before\n         * Angular, then the resource may have been generated as *.css. Simply try the resolution\n         * again.\n         */\n        const cssFallbackUrl = candidate.replace(CSS_PREPROCESSOR_EXT, '.css');\n        if (this.host.fileExists(cssFallbackUrl)) {\n          return cssFallbackUrl;\n        }\n      }\n    }\n    return null;\n  }\n\n  private getRootedCandidateLocations(url: string): AbsoluteFsPath[] {\n    // The path already starts with '/', so add a '.' to make it relative.\n    const segment: PathSegment = ('.' + url) as PathSegment;\n    return this.rootDirs.map(rootDir => join(rootDir, segment));\n  }\n\n  /**\n   * TypeScript provides utilities to resolve module names, but not resource files (which aren't\n   * a part of the ts.Program). However, TypeScript's module resolution can be used creatively\n   * to locate where resource files should be expected to exist. Since module resolution returns\n   * a list of file names that were considered, the loader can enumerate the possible locations\n   * for the file by setting up a module resolution for it that will fail.\n   */\n  private getResolvedCandidateLocations(url: string, fromFile: string): string[] {\n    // `failedLookupLocations` is in the name of the type ts.ResolvedModuleWithFailedLookupLocations\n    // but is marked @internal in TypeScript. See\n    // https://github.com/Microsoft/TypeScript/issues/28770.\n    type ResolvedModuleWithFailedLookupLocations =\n        ts.ResolvedModuleWithFailedLookupLocations & {failedLookupLocations: ReadonlyArray<string>};\n\n    // clang-format off\n    const failedLookup = ts.resolveModuleName(url + '.$ngresource$', fromFile, this.options, this.host) as ResolvedModuleWithFailedLookupLocations;\n    // clang-format on\n    if (failedLookup.failedLookupLocations === undefined) {\n      throw new Error(\n          `Internal error: expected to find failedLookupLocations during resolution of resource '${url}' in context of ${fromFile}`);\n    }\n\n    return failedLookup.failedLookupLocations\n        .filter(candidate => candidate.endsWith('.$ngresource$.ts'))\n        .map(candidate => candidate.replace(/\\.\\$ngresource\\$\\.ts$/, ''));\n  }\n}\n"]}
\No newline at end of file