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 | ;
|
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 |