-
Notifications
You must be signed in to change notification settings - Fork 201
/
resolve-specifier-node.ts
122 lines (107 loc) · 5.13 KB
/
resolve-specifier-node.ts
1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
/**
* @license
* Copyright (c) 2018 The Polymer Project Authors. All rights reserved.
* This code may only be used under the BSD style license found at
* http://polymer.github.io/LICENSE.txt
* The complete set of authors may be found at
* http://polymer.github.io/AUTHORS.txt
* The complete set of contributors may be found at
* http://polymer.github.io/CONTRIBUTORS.txt
* Code distributed by Google as part of the polymer project is also
* subject to an additional IP rights grant found at
* http://polymer.github.io/PATENTS.txt
*/
import * as whatwgUrl from 'whatwg-url';
import nodeResolve = require('resolve');
import {dirname, relative, join} from 'path';
import * as isWindows from 'is-windows';
import {FileRelativeUrl} from '../model/model';
const pathIsInside = require('path-is-inside');
const isPathSpecifier = (s: string) => /^\.{0,2}\//.test(s);
/**
* Resolves module specifiers using node module resolution.
*
* Full URLs - those parsable by the WHATWG URL spec - are returned as-is.
* Absolute and relative paths are resolved, even though they are valid
* HTML-spec module specifiers, because node resolution supports directories
* and omitting extensions. If a specifier doesn't resolve, it's returned as-is.
*
* @param componentInfo An object describing a "component-style" URL layout. In
* this layout, cross-package URLs reach out of the package directory to
* sibling packages, rather than into the component directory. When given,
* this parameter causes relative paths to be returns for this style.
*/
export const resolve =
(specifier: string, documentPath: string, componentInfo?: {
packageName: string,
rootDir: string,
componentDir: string
}): FileRelativeUrl => {
if (whatwgUrl.parseURL(specifier) !== null) {
return specifier as FileRelativeUrl;
}
const importerFilepath = documentPath;
const dependencyFilepath = nodeResolve.sync(specifier, {
basedir: dirname(importerFilepath),
// It's invalid to load a .json or .node file as a module on the web,
// but this is what Node's resolution algorithm does
// (https://nodejs.org/api/modules.html#modules_all_together), so we
// also do it here for completeness. Without including these
// extensions the user will probably get a 404. With them, they'll
// probably get an invalid MIME type error (which is hopefully more
// useful).
extensions: ['.js', '.json', '.node'],
// Some packages use a non-standard alternative to the "main" field
// in their package.json to differentiate their ES module version.
packageFilter:
(packageJson:
{main?: string, module?: string, 'jsnext:main'?: string}) => {
packageJson.main = packageJson.module ||
packageJson['jsnext:main'] || packageJson.main;
return packageJson;
},
});
let relativeSpecifierUrl =
relative(dirname(importerFilepath), dependencyFilepath) as
FileRelativeUrl;
if (componentInfo !== undefined) {
// Special handling for servers like Polyserve which, when serving a
// package "foo", will map the URL "/components/foo" to the root package
// directory, so that "foo" can make correct relative path references to
// its dependencies.
//
// Note that Polyserve will only set componentInfo if the particular
// request was for a URL path in the components/ directory.
const {packageName, rootDir, componentDir} = componentInfo;
const importerInRootPackage =
!pathIsInside(importerFilepath, componentDir);
const dependencyInRootPackage =
!pathIsInside(dependencyFilepath, componentDir);
if (importerInRootPackage && !dependencyInRootPackage) {
// A module from the root package, served from a components/ URL, is
// importing a module from a different package. In this case we need
// to fix up our relative path specifier, because on disk the
// dependency resolves to e.g. "./node_modules/foo", but in URL space
// it must resolve to "../foo".
//
// Note that the case where both the importer and the dependency are
// in the root package does not need to be fixed up, since the
// relative path works out the same.
const rootRelativeImporterPath = relative(rootDir, importerFilepath);
const effectiveImporterFilepath =
join(componentDir, packageName, rootRelativeImporterPath);
relativeSpecifierUrl = relative(
dirname(effectiveImporterFilepath),
dependencyFilepath) as FileRelativeUrl;
}
}
if (isWindows()) {
// normalize path separators to URL format
relativeSpecifierUrl =
relativeSpecifierUrl.replace(/\\/g, '/') as FileRelativeUrl;
}
if (!isPathSpecifier(relativeSpecifierUrl)) {
relativeSpecifierUrl = './' + relativeSpecifierUrl as FileRelativeUrl;
}
return relativeSpecifierUrl;
};