UNPKG

@angular-devkit/build-angular

Version:
192 lines • 22.2 kB
"use strict"; /** * @license * Copyright Google LLC All Rights Reserved. * * Use of this source code is governed by an MIT-style license that can be * found in the LICENSE file at https://angular.io/license */ var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) { if (k2 === undefined) k2 = k; var desc = Object.getOwnPropertyDescriptor(m, k); if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) { desc = { enumerable: true, get: function() { return m[k]; } }; } Object.defineProperty(o, k2, desc); }) : (function(o, m, k, k2) { if (k2 === undefined) k2 = k; o[k2] = m[k]; })); var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) { Object.defineProperty(o, "default", { enumerable: true, value: v }); }) : function(o, v) { o["default"] = v; }); var __importStar = (this && this.__importStar) || function (mod) { if (mod && mod.__esModule) return mod; var result = {}; if (mod != null) for (var k in mod) if (k !== "default" && Object.prototype.hasOwnProperty.call(mod, k)) __createBinding(result, mod, k); __setModuleDefault(result, mod); return result; }; Object.defineProperty(exports, "__esModule", { value: true }); exports.loadProxyConfiguration = void 0; const fast_glob_1 = require("fast-glob"); const node_fs_1 = require("node:fs"); const promises_1 = require("node:fs/promises"); const node_path_1 = require("node:path"); const node_url_1 = require("node:url"); const picomatch_1 = require("picomatch"); const error_1 = require("../../utils/error"); const load_esm_1 = require("../../utils/load-esm"); async function loadProxyConfiguration(root, proxyConfig, normalize = false) { if (!proxyConfig) { return undefined; } const proxyPath = (0, node_path_1.resolve)(root, proxyConfig); if (!(0, node_fs_1.existsSync)(proxyPath)) { throw new Error(`Proxy configuration file ${proxyPath} does not exist.`); } let proxyConfiguration; switch ((0, node_path_1.extname)(proxyPath)) { case '.json': { const content = await (0, promises_1.readFile)(proxyPath, 'utf-8'); const { parse, printParseErrorCode } = await Promise.resolve().then(() => __importStar(require('jsonc-parser'))); const parseErrors = []; proxyConfiguration = parse(content, parseErrors, { allowTrailingComma: true }); if (parseErrors.length > 0) { let errorMessage = `Proxy configuration file ${proxyPath} contains parse errors:`; for (const parseError of parseErrors) { const { line, column } = getJsonErrorLineColumn(parseError.offset, content); errorMessage += `\n[${line}, ${column}] ${printParseErrorCode(parseError.error)}`; } throw new Error(errorMessage); } break; } case '.mjs': // Load the ESM configuration file using the TypeScript dynamic import workaround. // Once TypeScript provides support for keeping the dynamic import this workaround can be // changed to a direct dynamic import. proxyConfiguration = (await (0, load_esm_1.loadEsmModule)((0, node_url_1.pathToFileURL)(proxyPath))) .default; break; case '.cjs': proxyConfiguration = require(proxyPath); break; default: // The file could be either CommonJS or ESM. // CommonJS is tried first then ESM if loading fails. try { proxyConfiguration = require(proxyPath); break; } catch (e) { (0, error_1.assertIsError)(e); if (e.code === 'ERR_REQUIRE_ESM') { // Load the ESM configuration file using the TypeScript dynamic import workaround. // Once TypeScript provides support for keeping the dynamic import this workaround can be // changed to a direct dynamic import. proxyConfiguration = (await (0, load_esm_1.loadEsmModule)((0, node_url_1.pathToFileURL)(proxyPath))) .default; break; } throw e; } } if (normalize) { proxyConfiguration = normalizeProxyConfiguration(proxyConfiguration); } return proxyConfiguration; } exports.loadProxyConfiguration = loadProxyConfiguration; /** * Converts glob patterns to regular expressions to support Vite's proxy option. * Also converts the Webpack supported array form to an object form supported by both. * * @param proxy A proxy configuration object. */ function normalizeProxyConfiguration(proxy) { let normalizedProxy; if (Array.isArray(proxy)) { // Construct an object-form proxy configuration from the array normalizedProxy = {}; for (const proxyEntry of proxy) { if (!('context' in proxyEntry)) { continue; } if (!Array.isArray(proxyEntry.context)) { continue; } // Array-form entries contain a context string array with the path(s) // to use for the configuration entry. const context = proxyEntry.context; delete proxyEntry.context; for (const contextEntry of context) { if (typeof contextEntry !== 'string') { continue; } normalizedProxy[contextEntry] = proxyEntry; } } } else { normalizedProxy = proxy; } // TODO: Consider upstreaming glob support for (const key of Object.keys(normalizedProxy)) { if ((0, fast_glob_1.isDynamicPattern)(key)) { const { output } = (0, picomatch_1.parse)(key); normalizedProxy[`^${output}$`] = normalizedProxy[key]; delete normalizedProxy[key]; } } // Replace `pathRewrite` field with a `rewrite` function for (const proxyEntry of Object.values(normalizedProxy)) { if ('pathRewrite' in proxyEntry && proxyEntry.pathRewrite && typeof proxyEntry.pathRewrite === 'object') { // Preprocess path rewrite entries const pathRewriteEntries = []; for (const [pattern, value] of Object.entries(proxyEntry.pathRewrite)) { pathRewriteEntries.push([new RegExp(pattern), value]); } proxyEntry.rewrite = pathRewriter.bind(undefined, pathRewriteEntries); delete proxyEntry.pathRewrite; } } return normalizedProxy; } function pathRewriter(pathRewriteEntries, path) { for (const [pattern, value] of pathRewriteEntries) { const updated = path.replace(pattern, value); if (path !== updated) { return updated; } } return path; } /** * Calculates the line and column for an error offset in the content of a JSON file. * @param location The offset error location from the beginning of the content. * @param content The full content of the file containing the error. * @returns An object containing the line and column */ function getJsonErrorLineColumn(offset, content) { if (offset === 0) { return { line: 1, column: 1 }; } let line = 0; let position = 0; // eslint-disable-next-line no-constant-condition while (true) { ++line; const nextNewline = content.indexOf('\n', position); if (nextNewline === -1 || nextNewline > offset) { break; } position = nextNewline + 1; } return { line, column: offset - position + 1 }; } //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"load-proxy-config.js","sourceRoot":"","sources":["../../../../../../../../../packages/angular_devkit/build_angular/src/builders/dev-server/load-proxy-config.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\nimport { isDynamicPattern } from 'fast-glob';\nimport { existsSync } from 'node:fs';\nimport { readFile } from 'node:fs/promises';\nimport { extname, resolve } from 'node:path';\nimport { pathToFileURL } from 'node:url';\nimport { parse as parseGlob } from 'picomatch';\nimport { assertIsError } from '../../utils/error';\nimport { loadEsmModule } from '../../utils/load-esm';\n\nexport async function loadProxyConfiguration(\n  root: string,\n  proxyConfig: string | undefined,\n  normalize = false,\n) {\n  if (!proxyConfig) {\n    return undefined;\n  }\n\n  const proxyPath = resolve(root, proxyConfig);\n\n  if (!existsSync(proxyPath)) {\n    throw new Error(`Proxy configuration file ${proxyPath} does not exist.`);\n  }\n\n  let proxyConfiguration;\n  switch (extname(proxyPath)) {\n    case '.json': {\n      const content = await readFile(proxyPath, 'utf-8');\n\n      const { parse, printParseErrorCode } = await import('jsonc-parser');\n      const parseErrors: import('jsonc-parser').ParseError[] = [];\n      proxyConfiguration = parse(content, parseErrors, { allowTrailingComma: true });\n\n      if (parseErrors.length > 0) {\n        let errorMessage = `Proxy configuration file ${proxyPath} contains parse errors:`;\n        for (const parseError of parseErrors) {\n          const { line, column } = getJsonErrorLineColumn(parseError.offset, content);\n          errorMessage += `\\n[${line}, ${column}] ${printParseErrorCode(parseError.error)}`;\n        }\n        throw new Error(errorMessage);\n      }\n\n      break;\n    }\n    case '.mjs':\n      // Load the ESM configuration file using the TypeScript dynamic import workaround.\n      // Once TypeScript provides support for keeping the dynamic import this workaround can be\n      // changed to a direct dynamic import.\n      proxyConfiguration = (await loadEsmModule<{ default: unknown }>(pathToFileURL(proxyPath)))\n        .default;\n      break;\n    case '.cjs':\n      proxyConfiguration = require(proxyPath);\n      break;\n    default:\n      // The file could be either CommonJS or ESM.\n      // CommonJS is tried first then ESM if loading fails.\n      try {\n        proxyConfiguration = require(proxyPath);\n        break;\n      } catch (e) {\n        assertIsError(e);\n        if (e.code === 'ERR_REQUIRE_ESM') {\n          // Load the ESM configuration file using the TypeScript dynamic import workaround.\n          // Once TypeScript provides support for keeping the dynamic import this workaround can be\n          // changed to a direct dynamic import.\n          proxyConfiguration = (await loadEsmModule<{ default: unknown }>(pathToFileURL(proxyPath)))\n            .default;\n          break;\n        }\n\n        throw e;\n      }\n  }\n\n  if (normalize) {\n    proxyConfiguration = normalizeProxyConfiguration(proxyConfiguration);\n  }\n\n  return proxyConfiguration;\n}\n\n/**\n * Converts glob patterns to regular expressions to support Vite's proxy option.\n * Also converts the Webpack supported array form to an object form supported by both.\n *\n * @param proxy A proxy configuration object.\n */\nfunction normalizeProxyConfiguration(\n  proxy: Record<string, object> | object[],\n): Record<string, object> {\n  let normalizedProxy: Record<string, object> | undefined;\n\n  if (Array.isArray(proxy)) {\n    // Construct an object-form proxy configuration from the array\n    normalizedProxy = {};\n    for (const proxyEntry of proxy) {\n      if (!('context' in proxyEntry)) {\n        continue;\n      }\n      if (!Array.isArray(proxyEntry.context)) {\n        continue;\n      }\n\n      // Array-form entries contain a context string array with the path(s)\n      // to use for the configuration entry.\n      const context = proxyEntry.context;\n      delete proxyEntry.context;\n      for (const contextEntry of context) {\n        if (typeof contextEntry !== 'string') {\n          continue;\n        }\n\n        normalizedProxy[contextEntry] = proxyEntry;\n      }\n    }\n  } else {\n    normalizedProxy = proxy;\n  }\n\n  // TODO: Consider upstreaming glob support\n  for (const key of Object.keys(normalizedProxy)) {\n    if (isDynamicPattern(key)) {\n      const { output } = parseGlob(key);\n      normalizedProxy[`^${output}$`] = normalizedProxy[key];\n      delete normalizedProxy[key];\n    }\n  }\n\n  // Replace `pathRewrite` field with a `rewrite` function\n  for (const proxyEntry of Object.values(normalizedProxy)) {\n    if (\n      'pathRewrite' in proxyEntry &&\n      proxyEntry.pathRewrite &&\n      typeof proxyEntry.pathRewrite === 'object'\n    ) {\n      // Preprocess path rewrite entries\n      const pathRewriteEntries: [RegExp, string][] = [];\n      for (const [pattern, value] of Object.entries(\n        proxyEntry.pathRewrite as Record<string, string>,\n      )) {\n        pathRewriteEntries.push([new RegExp(pattern), value]);\n      }\n\n      (proxyEntry as Record<string, unknown>).rewrite = pathRewriter.bind(\n        undefined,\n        pathRewriteEntries,\n      );\n\n      delete proxyEntry.pathRewrite;\n    }\n  }\n\n  return normalizedProxy;\n}\n\nfunction pathRewriter(pathRewriteEntries: [RegExp, string][], path: string): string {\n  for (const [pattern, value] of pathRewriteEntries) {\n    const updated = path.replace(pattern, value);\n    if (path !== updated) {\n      return updated;\n    }\n  }\n\n  return path;\n}\n\n/**\n * Calculates the line and column for an error offset in the content of a JSON file.\n * @param location The offset error location from the beginning of the content.\n * @param content The full content of the file containing the error.\n * @returns An object containing the line and column\n */\nfunction getJsonErrorLineColumn(offset: number, content: string) {\n  if (offset === 0) {\n    return { line: 1, column: 1 };\n  }\n\n  let line = 0;\n  let position = 0;\n  // eslint-disable-next-line no-constant-condition\n  while (true) {\n    ++line;\n\n    const nextNewline = content.indexOf('\\n', position);\n    if (nextNewline === -1 || nextNewline > offset) {\n      break;\n    }\n\n    position = nextNewline + 1;\n  }\n\n  return { line, column: offset - position + 1 };\n}\n"]}