[6a3a178] | 1 | /**
|
---|
| 2 | * @license
|
---|
| 3 | * Copyright Google LLC 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 | /**
|
---|
| 9 | * Create a {@link UrlResolver} with no package prefix.
|
---|
| 10 | */
|
---|
| 11 | export function createUrlResolverWithoutPackagePrefix() {
|
---|
| 12 | return new UrlResolver();
|
---|
| 13 | }
|
---|
| 14 | export function createOfflineCompileUrlResolver() {
|
---|
| 15 | return new UrlResolver('.');
|
---|
| 16 | }
|
---|
| 17 | export const UrlResolver = class UrlResolverImpl {
|
---|
| 18 | constructor(_packagePrefix = null) {
|
---|
| 19 | this._packagePrefix = _packagePrefix;
|
---|
| 20 | }
|
---|
| 21 | /**
|
---|
| 22 | * Resolves the `url` given the `baseUrl`:
|
---|
| 23 | * - when the `url` is null, the `baseUrl` is returned,
|
---|
| 24 | * - if `url` is relative ('path/to/here', './path/to/here'), the resolved url is a combination of
|
---|
| 25 | * `baseUrl` and `url`,
|
---|
| 26 | * - if `url` is absolute (it has a scheme: 'http://', 'https://' or start with '/'), the `url` is
|
---|
| 27 | * returned as is (ignoring the `baseUrl`)
|
---|
| 28 | */
|
---|
| 29 | resolve(baseUrl, url) {
|
---|
| 30 | let resolvedUrl = url;
|
---|
| 31 | if (baseUrl != null && baseUrl.length > 0) {
|
---|
| 32 | resolvedUrl = _resolveUrl(baseUrl, resolvedUrl);
|
---|
| 33 | }
|
---|
| 34 | const resolvedParts = _split(resolvedUrl);
|
---|
| 35 | let prefix = this._packagePrefix;
|
---|
| 36 | if (prefix != null && resolvedParts != null &&
|
---|
| 37 | resolvedParts[_ComponentIndex.Scheme] == 'package') {
|
---|
| 38 | let path = resolvedParts[_ComponentIndex.Path];
|
---|
| 39 | prefix = prefix.replace(/\/+$/, '');
|
---|
| 40 | path = path.replace(/^\/+/, '');
|
---|
| 41 | return `${prefix}/${path}`;
|
---|
| 42 | }
|
---|
| 43 | return resolvedUrl;
|
---|
| 44 | }
|
---|
| 45 | };
|
---|
| 46 | /**
|
---|
| 47 | * Extract the scheme of a URL.
|
---|
| 48 | */
|
---|
| 49 | export function getUrlScheme(url) {
|
---|
| 50 | const match = _split(url);
|
---|
| 51 | return (match && match[_ComponentIndex.Scheme]) || '';
|
---|
| 52 | }
|
---|
| 53 | // The code below is adapted from Traceur:
|
---|
| 54 | // https://github.com/google/traceur-compiler/blob/9511c1dafa972bf0de1202a8a863bad02f0f95a8/src/runtime/url.js
|
---|
| 55 | /**
|
---|
| 56 | * Builds a URI string from already-encoded parts.
|
---|
| 57 | *
|
---|
| 58 | * No encoding is performed. Any component may be omitted as either null or
|
---|
| 59 | * undefined.
|
---|
| 60 | *
|
---|
| 61 | * @param opt_scheme The scheme such as 'http'.
|
---|
| 62 | * @param opt_userInfo The user name before the '@'.
|
---|
| 63 | * @param opt_domain The domain such as 'www.google.com', already
|
---|
| 64 | * URI-encoded.
|
---|
| 65 | * @param opt_port The port number.
|
---|
| 66 | * @param opt_path The path, already URI-encoded. If it is not
|
---|
| 67 | * empty, it must begin with a slash.
|
---|
| 68 | * @param opt_queryData The URI-encoded query data.
|
---|
| 69 | * @param opt_fragment The URI-encoded fragment identifier.
|
---|
| 70 | * @return The fully combined URI.
|
---|
| 71 | */
|
---|
| 72 | function _buildFromEncodedParts(opt_scheme, opt_userInfo, opt_domain, opt_port, opt_path, opt_queryData, opt_fragment) {
|
---|
| 73 | const out = [];
|
---|
| 74 | if (opt_scheme != null) {
|
---|
| 75 | out.push(opt_scheme + ':');
|
---|
| 76 | }
|
---|
| 77 | if (opt_domain != null) {
|
---|
| 78 | out.push('//');
|
---|
| 79 | if (opt_userInfo != null) {
|
---|
| 80 | out.push(opt_userInfo + '@');
|
---|
| 81 | }
|
---|
| 82 | out.push(opt_domain);
|
---|
| 83 | if (opt_port != null) {
|
---|
| 84 | out.push(':' + opt_port);
|
---|
| 85 | }
|
---|
| 86 | }
|
---|
| 87 | if (opt_path != null) {
|
---|
| 88 | out.push(opt_path);
|
---|
| 89 | }
|
---|
| 90 | if (opt_queryData != null) {
|
---|
| 91 | out.push('?' + opt_queryData);
|
---|
| 92 | }
|
---|
| 93 | if (opt_fragment != null) {
|
---|
| 94 | out.push('#' + opt_fragment);
|
---|
| 95 | }
|
---|
| 96 | return out.join('');
|
---|
| 97 | }
|
---|
| 98 | /**
|
---|
| 99 | * A regular expression for breaking a URI into its component parts.
|
---|
| 100 | *
|
---|
| 101 | * {@link https://tools.ietf.org/html/rfc3986#appendix-B} says
|
---|
| 102 | * As the "first-match-wins" algorithm is identical to the "greedy"
|
---|
| 103 | * disambiguation method used by POSIX regular expressions, it is natural and
|
---|
| 104 | * commonplace to use a regular expression for parsing the potential five
|
---|
| 105 | * components of a URI reference.
|
---|
| 106 | *
|
---|
| 107 | * The following line is the regular expression for breaking-down a
|
---|
| 108 | * well-formed URI reference into its components.
|
---|
| 109 | *
|
---|
| 110 | * <pre>
|
---|
| 111 | * ^(([^:/?#]+):)?(//([^/?#]*))?([^?#]*)(\?([^#]*))?(#(.*))?
|
---|
| 112 | * 12 3 4 5 6 7 8 9
|
---|
| 113 | * </pre>
|
---|
| 114 | *
|
---|
| 115 | * The numbers in the second line above are only to assist readability; they
|
---|
| 116 | * indicate the reference points for each subexpression (i.e., each paired
|
---|
| 117 | * parenthesis). We refer to the value matched for subexpression <n> as $<n>.
|
---|
| 118 | * For example, matching the above expression to
|
---|
| 119 | * <pre>
|
---|
| 120 | * http://www.ics.uci.edu/pub/ietf/uri/#Related
|
---|
| 121 | * </pre>
|
---|
| 122 | * results in the following subexpression matches:
|
---|
| 123 | * <pre>
|
---|
| 124 | * $1 = http:
|
---|
| 125 | * $2 = http
|
---|
| 126 | * $3 = //www.ics.uci.edu
|
---|
| 127 | * $4 = www.ics.uci.edu
|
---|
| 128 | * $5 = /pub/ietf/uri/
|
---|
| 129 | * $6 = <undefined>
|
---|
| 130 | * $7 = <undefined>
|
---|
| 131 | * $8 = #Related
|
---|
| 132 | * $9 = Related
|
---|
| 133 | * </pre>
|
---|
| 134 | * where <undefined> indicates that the component is not present, as is the
|
---|
| 135 | * case for the query component in the above example. Therefore, we can
|
---|
| 136 | * determine the value of the five components as
|
---|
| 137 | * <pre>
|
---|
| 138 | * scheme = $2
|
---|
| 139 | * authority = $4
|
---|
| 140 | * path = $5
|
---|
| 141 | * query = $7
|
---|
| 142 | * fragment = $9
|
---|
| 143 | * </pre>
|
---|
| 144 | *
|
---|
| 145 | * The regular expression has been modified slightly to expose the
|
---|
| 146 | * userInfo, domain, and port separately from the authority.
|
---|
| 147 | * The modified version yields
|
---|
| 148 | * <pre>
|
---|
| 149 | * $1 = http scheme
|
---|
| 150 | * $2 = <undefined> userInfo -\
|
---|
| 151 | * $3 = www.ics.uci.edu domain | authority
|
---|
| 152 | * $4 = <undefined> port -/
|
---|
| 153 | * $5 = /pub/ietf/uri/ path
|
---|
| 154 | * $6 = <undefined> query without ?
|
---|
| 155 | * $7 = Related fragment without #
|
---|
| 156 | * </pre>
|
---|
| 157 | * @internal
|
---|
| 158 | */
|
---|
| 159 | const _splitRe = new RegExp('^' +
|
---|
| 160 | '(?:' +
|
---|
| 161 | '([^:/?#.]+)' + // scheme - ignore special characters
|
---|
| 162 | // used by other URL parts such as :,
|
---|
| 163 | // ?, /, #, and .
|
---|
| 164 | ':)?' +
|
---|
| 165 | '(?://' +
|
---|
| 166 | '(?:([^/?#]*)@)?' + // userInfo
|
---|
| 167 | '([\\w\\d\\-\\u0100-\\uffff.%]*)' + // domain - restrict to letters,
|
---|
| 168 | // digits, dashes, dots, percent
|
---|
| 169 | // escapes, and unicode characters.
|
---|
| 170 | '(?::([0-9]+))?' + // port
|
---|
| 171 | ')?' +
|
---|
| 172 | '([^?#]+)?' + // path
|
---|
| 173 | '(?:\\?([^#]*))?' + // query
|
---|
| 174 | '(?:#(.*))?' + // fragment
|
---|
| 175 | '$');
|
---|
| 176 | /**
|
---|
| 177 | * The index of each URI component in the return value of goog.uri.utils.split.
|
---|
| 178 | * @enum {number}
|
---|
| 179 | */
|
---|
| 180 | var _ComponentIndex;
|
---|
| 181 | (function (_ComponentIndex) {
|
---|
| 182 | _ComponentIndex[_ComponentIndex["Scheme"] = 1] = "Scheme";
|
---|
| 183 | _ComponentIndex[_ComponentIndex["UserInfo"] = 2] = "UserInfo";
|
---|
| 184 | _ComponentIndex[_ComponentIndex["Domain"] = 3] = "Domain";
|
---|
| 185 | _ComponentIndex[_ComponentIndex["Port"] = 4] = "Port";
|
---|
| 186 | _ComponentIndex[_ComponentIndex["Path"] = 5] = "Path";
|
---|
| 187 | _ComponentIndex[_ComponentIndex["QueryData"] = 6] = "QueryData";
|
---|
| 188 | _ComponentIndex[_ComponentIndex["Fragment"] = 7] = "Fragment";
|
---|
| 189 | })(_ComponentIndex || (_ComponentIndex = {}));
|
---|
| 190 | /**
|
---|
| 191 | * Splits a URI into its component parts.
|
---|
| 192 | *
|
---|
| 193 | * Each component can be accessed via the component indices; for example:
|
---|
| 194 | * <pre>
|
---|
| 195 | * goog.uri.utils.split(someStr)[goog.uri.utils.CompontentIndex.QUERY_DATA];
|
---|
| 196 | * </pre>
|
---|
| 197 | *
|
---|
| 198 | * @param uri The URI string to examine.
|
---|
| 199 | * @return Each component still URI-encoded.
|
---|
| 200 | * Each component that is present will contain the encoded value, whereas
|
---|
| 201 | * components that are not present will be undefined or empty, depending
|
---|
| 202 | * on the browser's regular expression implementation. Never null, since
|
---|
| 203 | * arbitrary strings may still look like path names.
|
---|
| 204 | */
|
---|
| 205 | function _split(uri) {
|
---|
| 206 | return uri.match(_splitRe);
|
---|
| 207 | }
|
---|
| 208 | /**
|
---|
| 209 | * Removes dot segments in given path component, as described in
|
---|
| 210 | * RFC 3986, section 5.2.4.
|
---|
| 211 | *
|
---|
| 212 | * @param path A non-empty path component.
|
---|
| 213 | * @return Path component with removed dot segments.
|
---|
| 214 | */
|
---|
| 215 | function _removeDotSegments(path) {
|
---|
| 216 | if (path == '/')
|
---|
| 217 | return '/';
|
---|
| 218 | const leadingSlash = path[0] == '/' ? '/' : '';
|
---|
| 219 | const trailingSlash = path[path.length - 1] === '/' ? '/' : '';
|
---|
| 220 | const segments = path.split('/');
|
---|
| 221 | const out = [];
|
---|
| 222 | let up = 0;
|
---|
| 223 | for (let pos = 0; pos < segments.length; pos++) {
|
---|
| 224 | const segment = segments[pos];
|
---|
| 225 | switch (segment) {
|
---|
| 226 | case '':
|
---|
| 227 | case '.':
|
---|
| 228 | break;
|
---|
| 229 | case '..':
|
---|
| 230 | if (out.length > 0) {
|
---|
| 231 | out.pop();
|
---|
| 232 | }
|
---|
| 233 | else {
|
---|
| 234 | up++;
|
---|
| 235 | }
|
---|
| 236 | break;
|
---|
| 237 | default:
|
---|
| 238 | out.push(segment);
|
---|
| 239 | }
|
---|
| 240 | }
|
---|
| 241 | if (leadingSlash == '') {
|
---|
| 242 | while (up-- > 0) {
|
---|
| 243 | out.unshift('..');
|
---|
| 244 | }
|
---|
| 245 | if (out.length === 0)
|
---|
| 246 | out.push('.');
|
---|
| 247 | }
|
---|
| 248 | return leadingSlash + out.join('/') + trailingSlash;
|
---|
| 249 | }
|
---|
| 250 | /**
|
---|
| 251 | * Takes an array of the parts from split and canonicalizes the path part
|
---|
| 252 | * and then joins all the parts.
|
---|
| 253 | */
|
---|
| 254 | function _joinAndCanonicalizePath(parts) {
|
---|
| 255 | let path = parts[_ComponentIndex.Path];
|
---|
| 256 | path = path == null ? '' : _removeDotSegments(path);
|
---|
| 257 | parts[_ComponentIndex.Path] = path;
|
---|
| 258 | return _buildFromEncodedParts(parts[_ComponentIndex.Scheme], parts[_ComponentIndex.UserInfo], parts[_ComponentIndex.Domain], parts[_ComponentIndex.Port], path, parts[_ComponentIndex.QueryData], parts[_ComponentIndex.Fragment]);
|
---|
| 259 | }
|
---|
| 260 | /**
|
---|
| 261 | * Resolves a URL.
|
---|
| 262 | * @param base The URL acting as the base URL.
|
---|
| 263 | * @param to The URL to resolve.
|
---|
| 264 | */
|
---|
| 265 | function _resolveUrl(base, url) {
|
---|
| 266 | const parts = _split(encodeURI(url));
|
---|
| 267 | const baseParts = _split(base);
|
---|
| 268 | if (parts[_ComponentIndex.Scheme] != null) {
|
---|
| 269 | return _joinAndCanonicalizePath(parts);
|
---|
| 270 | }
|
---|
| 271 | else {
|
---|
| 272 | parts[_ComponentIndex.Scheme] = baseParts[_ComponentIndex.Scheme];
|
---|
| 273 | }
|
---|
| 274 | for (let i = _ComponentIndex.Scheme; i <= _ComponentIndex.Port; i++) {
|
---|
| 275 | if (parts[i] == null) {
|
---|
| 276 | parts[i] = baseParts[i];
|
---|
| 277 | }
|
---|
| 278 | }
|
---|
| 279 | if (parts[_ComponentIndex.Path][0] == '/') {
|
---|
| 280 | return _joinAndCanonicalizePath(parts);
|
---|
| 281 | }
|
---|
| 282 | let path = baseParts[_ComponentIndex.Path];
|
---|
| 283 | if (path == null)
|
---|
| 284 | path = '/';
|
---|
| 285 | const index = path.lastIndexOf('/');
|
---|
| 286 | path = path.substring(0, index + 1) + parts[_ComponentIndex.Path];
|
---|
| 287 | parts[_ComponentIndex.Path] = path;
|
---|
| 288 | return _joinAndCanonicalizePath(parts);
|
---|
| 289 | }
|
---|
| 290 | //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"url_resolver.js","sourceRoot":"","sources":["../../../../../../packages/compiler/src/url_resolver.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\n/**\n * Create a {@link UrlResolver} with no package prefix.\n */\nexport function createUrlResolverWithoutPackagePrefix(): UrlResolver {\n  return new UrlResolver();\n}\n\nexport function createOfflineCompileUrlResolver(): UrlResolver {\n  return new UrlResolver('.');\n}\n\n/**\n * Used by the {@link Compiler} when resolving HTML and CSS template URLs.\n *\n * This class can be overridden by the application developer to create custom behavior.\n *\n * See {@link Compiler}\n *\n * ## Example\n *\n * <code-example path=\"compiler/ts/url_resolver/url_resolver.ts\"></code-example>\n *\n * @security  When compiling templates at runtime, you must\n * ensure that the entire template comes from a trusted source.\n * Attacker-controlled data introduced by a template could expose your\n * application to XSS risks. For more detail, see the [Security Guide](https://g.co/ng/security).\n */\nexport interface UrlResolver {\n  resolve(baseUrl: string, url: string): string;\n}\n\nexport interface UrlResolverCtor {\n  new(packagePrefix?: string|null): UrlResolver;\n}\n\nexport const UrlResolver: UrlResolverCtor = class UrlResolverImpl {\n  constructor(private _packagePrefix: string|null = null) {}\n\n  /**\n   * Resolves the `url` given the `baseUrl`:\n   * - when the `url` is null, the `baseUrl` is returned,\n   * - if `url` is relative ('path/to/here', './path/to/here'), the resolved url is a combination of\n   * `baseUrl` and `url`,\n   * - if `url` is absolute (it has a scheme: 'http://', 'https://' or start with '/'), the `url` is\n   * returned as is (ignoring the `baseUrl`)\n   */\n  resolve(baseUrl: string, url: string): string {\n    let resolvedUrl = url;\n    if (baseUrl != null && baseUrl.length > 0) {\n      resolvedUrl = _resolveUrl(baseUrl, resolvedUrl);\n    }\n    const resolvedParts = _split(resolvedUrl);\n    let prefix = this._packagePrefix;\n    if (prefix != null && resolvedParts != null &&\n        resolvedParts[_ComponentIndex.Scheme] == 'package') {\n      let path = resolvedParts[_ComponentIndex.Path];\n      prefix = prefix.replace(/\\/+$/, '');\n      path = path.replace(/^\\/+/, '');\n      return `${prefix}/${path}`;\n    }\n    return resolvedUrl;\n  }\n};\n\n/**\n * Extract the scheme of a URL.\n */\nexport function getUrlScheme(url: string): string {\n  const match = _split(url);\n  return (match && match[_ComponentIndex.Scheme]) || '';\n}\n\n// The code below is adapted from Traceur:\n// https://github.com/google/traceur-compiler/blob/9511c1dafa972bf0de1202a8a863bad02f0f95a8/src/runtime/url.js\n\n/**\n * Builds a URI string from already-encoded parts.\n *\n * No encoding is performed.  Any component may be omitted as either null or\n * undefined.\n *\n * @param opt_scheme The scheme such as 'http'.\n * @param opt_userInfo The user name before the '@'.\n * @param opt_domain The domain such as 'www.google.com', already\n *     URI-encoded.\n * @param opt_port The port number.\n * @param opt_path The path, already URI-encoded.  If it is not\n *     empty, it must begin with a slash.\n * @param opt_queryData The URI-encoded query data.\n * @param opt_fragment The URI-encoded fragment identifier.\n * @return The fully combined URI.\n */\nfunction _buildFromEncodedParts(\n    opt_scheme?: string, opt_userInfo?: string, opt_domain?: string, opt_port?: string,\n    opt_path?: string, opt_queryData?: string, opt_fragment?: string): string {\n  const out: string[] = [];\n\n  if (opt_scheme != null) {\n    out.push(opt_scheme + ':');\n  }\n\n  if (opt_domain != null) {\n    out.push('//');\n\n    if (opt_userInfo != null) {\n      out.push(opt_userInfo + '@');\n    }\n\n    out.push(opt_domain);\n\n    if (opt_port != null) {\n      out.push(':' + opt_port);\n    }\n  }\n\n  if (opt_path != null) {\n    out.push(opt_path);\n  }\n\n  if (opt_queryData != null) {\n    out.push('?' + opt_queryData);\n  }\n\n  if (opt_fragment != null) {\n    out.push('#' + opt_fragment);\n  }\n\n  return out.join('');\n}\n\n/**\n * A regular expression for breaking a URI into its component parts.\n *\n * {@link https://tools.ietf.org/html/rfc3986#appendix-B} says\n * As the \"first-match-wins\" algorithm is identical to the \"greedy\"\n * disambiguation method used by POSIX regular expressions, it is natural and\n * commonplace to use a regular expression for parsing the potential five\n * components of a URI reference.\n *\n * The following line is the regular expression for breaking-down a\n * well-formed URI reference into its components.\n *\n * <pre>\n * ^(([^:/?#]+):)?(//([^/?#]*))?([^?#]*)(\\?([^#]*))?(#(.*))?\n *  12            3  4          5       6  7        8 9\n * </pre>\n *\n * The numbers in the second line above are only to assist readability; they\n * indicate the reference points for each subexpression (i.e., each paired\n * parenthesis). We refer to the value matched for subexpression <n> as $<n>.\n * For example, matching the above expression to\n * <pre>\n *     http://www.ics.uci.edu/pub/ietf/uri/#Related\n * </pre>\n * results in the following subexpression matches:\n * <pre>\n *    $1 = http:\n *    $2 = http\n *    $3 = //www.ics.uci.edu\n *    $4 = www.ics.uci.edu\n *    $5 = /pub/ietf/uri/\n *    $6 = <undefined>\n *    $7 = <undefined>\n *    $8 = #Related\n *    $9 = Related\n * </pre>\n * where <undefined> indicates that the component is not present, as is the\n * case for the query component in the above example. Therefore, we can\n * determine the value of the five components as\n * <pre>\n *    scheme    = $2\n *    authority = $4\n *    path      = $5\n *    query     = $7\n *    fragment  = $9\n * </pre>\n *\n * The regular expression has been modified slightly to expose the\n * userInfo, domain, and port separately from the authority.\n * The modified version yields\n * <pre>\n *    $1 = http              scheme\n *    $2 = <undefined>       userInfo -\\\n *    $3 = www.ics.uci.edu   domain     | authority\n *    $4 = <undefined>       port     -/\n *    $5 = /pub/ietf/uri/    path\n *    $6 = <undefined>       query without ?\n *    $7 = Related           fragment without #\n * </pre>\n * @internal\n */\nconst _splitRe = new RegExp(\n    '^' +\n    '(?:' +\n    '([^:/?#.]+)' +  // scheme - ignore special characters\n                     // used by other URL parts such as :,\n                     // ?, /, #, and .\n    ':)?' +\n    '(?://' +\n    '(?:([^/?#]*)@)?' +                  // userInfo\n    '([\\\\w\\\\d\\\\-\\\\u0100-\\\\uffff.%]*)' +  // domain - restrict to letters,\n                                         // digits, dashes, dots, percent\n                                         // escapes, and unicode characters.\n    '(?::([0-9]+))?' +                   // port\n    ')?' +\n    '([^?#]+)?' +        // path\n    '(?:\\\\?([^#]*))?' +  // query\n    '(?:#(.*))?' +       // fragment\n    '$');\n\n/**\n * The index of each URI component in the return value of goog.uri.utils.split.\n * @enum {number}\n */\nenum _ComponentIndex {\n  Scheme = 1,\n  UserInfo,\n  Domain,\n  Port,\n  Path,\n  QueryData,\n  Fragment\n}\n\n/**\n * Splits a URI into its component parts.\n *\n * Each component can be accessed via the component indices; for example:\n * <pre>\n * goog.uri.utils.split(someStr)[goog.uri.utils.CompontentIndex.QUERY_DATA];\n * </pre>\n *\n * @param uri The URI string to examine.\n * @return Each component still URI-encoded.\n *     Each component that is present will contain the encoded value, whereas\n *     components that are not present will be undefined or empty, depending\n *     on the browser's regular expression implementation.  Never null, since\n *     arbitrary strings may still look like path names.\n */\nfunction _split(uri: string): Array<string|any> {\n  return uri.match(_splitRe)!;\n}\n\n/**\n * Removes dot segments in given path component, as described in\n * RFC 3986, section 5.2.4.\n *\n * @param path A non-empty path component.\n * @return Path component with removed dot segments.\n */\nfunction _removeDotSegments(path: string): string {\n  if (path == '/') return '/';\n\n  const leadingSlash = path[0] == '/' ? '/' : '';\n  const trailingSlash = path[path.length - 1] === '/' ? '/' : '';\n  const segments = path.split('/');\n\n  const out: string[] = [];\n  let up = 0;\n  for (let pos = 0; pos < segments.length; pos++) {\n    const segment = segments[pos];\n    switch (segment) {\n      case '':\n      case '.':\n        break;\n      case '..':\n        if (out.length > 0) {\n          out.pop();\n        } else {\n          up++;\n        }\n        break;\n      default:\n        out.push(segment);\n    }\n  }\n\n  if (leadingSlash == '') {\n    while (up-- > 0) {\n      out.unshift('..');\n    }\n\n    if (out.length === 0) out.push('.');\n  }\n\n  return leadingSlash + out.join('/') + trailingSlash;\n}\n\n/**\n * Takes an array of the parts from split and canonicalizes the path part\n * and then joins all the parts.\n */\nfunction _joinAndCanonicalizePath(parts: any[]): string {\n  let path = parts[_ComponentIndex.Path];\n  path = path == null ? '' : _removeDotSegments(path);\n  parts[_ComponentIndex.Path] = path;\n\n  return _buildFromEncodedParts(\n      parts[_ComponentIndex.Scheme], parts[_ComponentIndex.UserInfo], parts[_ComponentIndex.Domain],\n      parts[_ComponentIndex.Port], path, parts[_ComponentIndex.QueryData],\n      parts[_ComponentIndex.Fragment]);\n}\n\n/**\n * Resolves a URL.\n * @param base The URL acting as the base URL.\n * @param to The URL to resolve.\n */\nfunction _resolveUrl(base: string, url: string): string {\n  const parts = _split(encodeURI(url));\n  const baseParts = _split(base);\n\n  if (parts[_ComponentIndex.Scheme] != null) {\n    return _joinAndCanonicalizePath(parts);\n  } else {\n    parts[_ComponentIndex.Scheme] = baseParts[_ComponentIndex.Scheme];\n  }\n\n  for (let i = _ComponentIndex.Scheme; i <= _ComponentIndex.Port; i++) {\n    if (parts[i] == null) {\n      parts[i] = baseParts[i];\n    }\n  }\n\n  if (parts[_ComponentIndex.Path][0] == '/') {\n    return _joinAndCanonicalizePath(parts);\n  }\n\n  let path = baseParts[_ComponentIndex.Path];\n  if (path == null) path = '/';\n  const index = path.lastIndexOf('/');\n  path = path.substring(0, index + 1) + parts[_ComponentIndex.Path];\n  parts[_ComponentIndex.Path] = path;\n  return _joinAndCanonicalizePath(parts);\n}\n"]} |
---|