UNPKG

20 kBJavaScriptView Raw
1"use strict";
2/**
3 * @license
4 * Copyright Google LLC All Rights Reserved.
5 *
6 * Use of this source code is governed by an MIT-style license that can be
7 * found in the LICENSE file at https://angular.io/license
8 */
9Object.defineProperty(exports, "__esModule", { value: true });
10exports.applyTemplates = exports.template = exports.renameTemplateFiles = exports.pathTemplate = exports.applyPathTemplate = exports.contentTemplate = exports.applyContentTemplate = exports.InvalidPipeException = exports.UnknownPipeException = exports.OptionIsNotDefinedException = exports.TEMPLATE_FILENAME_RE = void 0;
11const core_1 = require("@angular-devkit/core");
12const util_1 = require("util");
13const base_1 = require("./base");
14exports.TEMPLATE_FILENAME_RE = /\.template$/;
15class OptionIsNotDefinedException extends core_1.BaseException {
16 constructor(name) {
17 super(`Option "${name}" is not defined.`);
18 }
19}
20exports.OptionIsNotDefinedException = OptionIsNotDefinedException;
21class UnknownPipeException extends core_1.BaseException {
22 constructor(name) {
23 super(`Pipe "${name}" is not defined.`);
24 }
25}
26exports.UnknownPipeException = UnknownPipeException;
27class InvalidPipeException extends core_1.BaseException {
28 constructor(name) {
29 super(`Pipe "${name}" is invalid.`);
30 }
31}
32exports.InvalidPipeException = InvalidPipeException;
33const decoder = new util_1.TextDecoder('utf-8', { fatal: true });
34function applyContentTemplate(options) {
35 return (entry) => {
36 const { path, content } = entry;
37 try {
38 const decodedContent = decoder.decode(content);
39 return {
40 path,
41 content: Buffer.from((0, core_1.template)(decodedContent, {})(options)),
42 };
43 }
44 catch (e) {
45 if (e.code === 'ERR_ENCODING_INVALID_ENCODED_DATA') {
46 return entry;
47 }
48 throw e;
49 }
50 };
51}
52exports.applyContentTemplate = applyContentTemplate;
53function contentTemplate(options) {
54 return (0, base_1.forEach)(applyContentTemplate(options));
55}
56exports.contentTemplate = contentTemplate;
57function applyPathTemplate(data, options = {
58 interpolationStart: '__',
59 interpolationEnd: '__',
60 pipeSeparator: '@',
61}) {
62 const is = options.interpolationStart;
63 const ie = options.interpolationEnd;
64 const isL = is.length;
65 const ieL = ie.length;
66 return (entry) => {
67 let path = entry.path;
68 const content = entry.content;
69 const original = path;
70 let start = path.indexOf(is);
71 // + 1 to have at least a length 1 name. `____` is not valid.
72 let end = path.indexOf(ie, start + isL + 1);
73 while (start != -1 && end != -1) {
74 const match = path.substring(start + isL, end);
75 let replacement = data[match];
76 if (!options.pipeSeparator) {
77 if (typeof replacement == 'function') {
78 replacement = replacement.call(data, original);
79 }
80 if (replacement === undefined) {
81 throw new OptionIsNotDefinedException(match);
82 }
83 }
84 else {
85 const [name, ...pipes] = match.split(options.pipeSeparator);
86 replacement = data[name];
87 if (typeof replacement == 'function') {
88 replacement = replacement.call(data, original);
89 }
90 if (replacement === undefined) {
91 throw new OptionIsNotDefinedException(name);
92 }
93 replacement = pipes.reduce((acc, pipe) => {
94 if (!pipe) {
95 return acc;
96 }
97 if (!(pipe in data)) {
98 throw new UnknownPipeException(pipe);
99 }
100 if (typeof data[pipe] != 'function') {
101 throw new InvalidPipeException(pipe);
102 }
103 // Coerce to string.
104 return '' + data[pipe](acc);
105 }, '' + replacement);
106 }
107 path = path.substring(0, start) + replacement + path.substring(end + ieL);
108 start = path.indexOf(options.interpolationStart);
109 // See above.
110 end = path.indexOf(options.interpolationEnd, start + isL + 1);
111 }
112 return { path: (0, core_1.normalize)(path), content };
113 };
114}
115exports.applyPathTemplate = applyPathTemplate;
116function pathTemplate(options) {
117 return (0, base_1.forEach)(applyPathTemplate(options));
118}
119exports.pathTemplate = pathTemplate;
120/**
121 * Remove every `.template` suffix from file names.
122 */
123function renameTemplateFiles() {
124 return (0, base_1.forEach)((entry) => {
125 if (entry.path.match(exports.TEMPLATE_FILENAME_RE)) {
126 return {
127 content: entry.content,
128 path: (0, core_1.normalize)(entry.path.replace(exports.TEMPLATE_FILENAME_RE, '')),
129 };
130 }
131 else {
132 return entry;
133 }
134 });
135}
136exports.renameTemplateFiles = renameTemplateFiles;
137function template(options) {
138 return (0, base_1.chain)([
139 contentTemplate(options),
140 // Force cast to PathTemplateData. We need the type for the actual pathTemplate() call,
141 // but in this case we cannot do anything as contentTemplate are more permissive.
142 // Since values are coerced to strings in PathTemplates it will be fine in the end.
143 pathTemplate(options),
144 ]);
145}
146exports.template = template;
147function applyTemplates(options) {
148 return (0, base_1.forEach)((0, base_1.when)((path) => path.endsWith('.template'), (0, base_1.composeFileOperators)([
149 applyContentTemplate(options),
150 // See above for this weird cast.
151 applyPathTemplate(options),
152 (entry) => {
153 return {
154 content: entry.content,
155 path: entry.path.replace(exports.TEMPLATE_FILENAME_RE, ''),
156 };
157 },
158 ])));
159}
160exports.applyTemplates = applyTemplates;
161//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"template.js","sourceRoot":"","sources":["../../../../../../../../packages/angular_devkit/schematics/src/rules/template.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 { BaseException, normalize, template as templateImpl } from '@angular-devkit/core';\nimport { TextDecoder } from 'util';\nimport { FileOperator, Rule } from '../engine/interface';\nimport { FileEntry } from '../tree/interface';\nimport { chain, composeFileOperators, forEach, when } from './base';\n\nexport const TEMPLATE_FILENAME_RE = /\\.template$/;\n\nexport class OptionIsNotDefinedException extends BaseException {\n  constructor(name: string) {\n    super(`Option \"${name}\" is not defined.`);\n  }\n}\n\nexport class UnknownPipeException extends BaseException {\n  constructor(name: string) {\n    super(`Pipe \"${name}\" is not defined.`);\n  }\n}\n\nexport class InvalidPipeException extends BaseException {\n  constructor(name: string) {\n    super(`Pipe \"${name}\" is invalid.`);\n  }\n}\n\nexport type PathTemplateValue = boolean | string | number | undefined;\nexport type PathTemplatePipeFunction = (x: string) => PathTemplateValue;\nexport type PathTemplateData = {\n  [key: string]: PathTemplateValue | PathTemplateData | PathTemplatePipeFunction;\n};\n\nexport interface PathTemplateOptions {\n  // Interpolation start and end strings.\n  interpolationStart: string;\n  // Interpolation start and end strings.\n  interpolationEnd: string;\n\n  // Separator for pipes. Do not specify to remove pipe support.\n  pipeSeparator?: string;\n}\n\nconst decoder = new TextDecoder('utf-8', { fatal: true });\n\nexport function applyContentTemplate<T>(options: T): FileOperator {\n  return (entry: FileEntry) => {\n    const { path, content } = entry;\n\n    try {\n      const decodedContent = decoder.decode(content);\n\n      return {\n        path,\n        content: Buffer.from(templateImpl(decodedContent, {})(options)),\n      };\n    } catch (e) {\n      if ((e as NodeJS.ErrnoException).code === 'ERR_ENCODING_INVALID_ENCODED_DATA') {\n        return entry;\n      }\n\n      throw e;\n    }\n  };\n}\n\nexport function contentTemplate<T>(options: T): Rule {\n  return forEach(applyContentTemplate(options));\n}\n\nexport function applyPathTemplate<T extends PathTemplateData>(\n  data: T,\n  options: PathTemplateOptions = {\n    interpolationStart: '__',\n    interpolationEnd: '__',\n    pipeSeparator: '@',\n  },\n): FileOperator {\n  const is = options.interpolationStart;\n  const ie = options.interpolationEnd;\n  const isL = is.length;\n  const ieL = ie.length;\n\n  return (entry: FileEntry) => {\n    let path = entry.path as string;\n    const content = entry.content;\n    const original = path;\n\n    let start = path.indexOf(is);\n    // + 1 to have at least a length 1 name. `____` is not valid.\n    let end = path.indexOf(ie, start + isL + 1);\n\n    while (start != -1 && end != -1) {\n      const match = path.substring(start + isL, end);\n      let replacement = data[match];\n\n      if (!options.pipeSeparator) {\n        if (typeof replacement == 'function') {\n          replacement = replacement.call(data, original);\n        }\n\n        if (replacement === undefined) {\n          throw new OptionIsNotDefinedException(match);\n        }\n      } else {\n        const [name, ...pipes] = match.split(options.pipeSeparator);\n        replacement = data[name];\n\n        if (typeof replacement == 'function') {\n          replacement = replacement.call(data, original);\n        }\n\n        if (replacement === undefined) {\n          throw new OptionIsNotDefinedException(name);\n        }\n\n        replacement = pipes.reduce((acc: string, pipe: string) => {\n          if (!pipe) {\n            return acc;\n          }\n          if (!(pipe in data)) {\n            throw new UnknownPipeException(pipe);\n          }\n          if (typeof data[pipe] != 'function') {\n            throw new InvalidPipeException(pipe);\n          }\n\n          // Coerce to string.\n          return '' + (data[pipe] as PathTemplatePipeFunction)(acc);\n        }, '' + replacement);\n      }\n\n      path = path.substring(0, start) + replacement + path.substring(end + ieL);\n\n      start = path.indexOf(options.interpolationStart);\n      // See above.\n      end = path.indexOf(options.interpolationEnd, start + isL + 1);\n    }\n\n    return { path: normalize(path), content };\n  };\n}\n\nexport function pathTemplate<T extends PathTemplateData>(options: T): Rule {\n  return forEach(applyPathTemplate(options));\n}\n\n/**\n * Remove every `.template` suffix from file names.\n */\nexport function renameTemplateFiles(): Rule {\n  return forEach((entry) => {\n    if (entry.path.match(TEMPLATE_FILENAME_RE)) {\n      return {\n        content: entry.content,\n        path: normalize(entry.path.replace(TEMPLATE_FILENAME_RE, '')),\n      };\n    } else {\n      return entry;\n    }\n  });\n}\n\nexport function template<T extends object>(options: T): Rule {\n  return chain([\n    contentTemplate(options),\n    // Force cast to PathTemplateData. We need the type for the actual pathTemplate() call,\n    // but in this case we cannot do anything as contentTemplate are more permissive.\n    // Since values are coerced to strings in PathTemplates it will be fine in the end.\n    pathTemplate(options as {} as PathTemplateData),\n  ]);\n}\n\nexport function applyTemplates<T extends object>(options: T): Rule {\n  return forEach(\n    when(\n      (path) => path.endsWith('.template'),\n      composeFileOperators([\n        applyContentTemplate(options),\n        // See above for this weird cast.\n        applyPathTemplate(options as {} as PathTemplateData),\n        (entry) => {\n          return {\n            content: entry.content,\n            path: entry.path.replace(TEMPLATE_FILENAME_RE, ''),\n          } as FileEntry;\n        },\n      ]),\n    ),\n  );\n}\n"]}
\No newline at end of file