source: trip-planner-front/node_modules/@angular/compiler-cli/ngcc/src/writing/package_json_updater.js@ 6a3a178

Last change on this file since 6a3a178 was 6a3a178, checked in by Ema <ema_spirova@…>, 3 years ago

initial commit

  • Property mode set to 100644
File size: 25.6 KB
Line 
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(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/ngcc/src/writing/package_json_updater", ["require", "exports", "tslib", "@angular/compiler-cli/src/ngtsc/file_system"], factory);
15 }
16})(function (require, exports) {
17 "use strict";
18 Object.defineProperty(exports, "__esModule", { value: true });
19 exports.applyChange = exports.DirectPackageJsonUpdater = exports.PackageJsonUpdate = void 0;
20 var tslib_1 = require("tslib");
21 var file_system_1 = require("@angular/compiler-cli/src/ngtsc/file_system");
22 /**
23 * A utility class providing a fluent API for recording multiple changes to a `package.json` file
24 * (and optionally its in-memory parsed representation).
25 *
26 * NOTE: This class should generally not be instantiated directly; instances are implicitly created
27 * via `PackageJsonUpdater#createUpdate()`.
28 */
29 var PackageJsonUpdate = /** @class */ (function () {
30 function PackageJsonUpdate(writeChangesImpl) {
31 this.writeChangesImpl = writeChangesImpl;
32 this.changes = [];
33 this.applied = false;
34 }
35 /**
36 * Record a change to a `package.json` property.
37 *
38 * If the ancestor objects do not yet exist in the `package.json` file, they will be created. The
39 * positioning of the property can also be specified. (If the property already exists, it will be
40 * moved accordingly.)
41 *
42 * NOTE: Property positioning is only guaranteed to be respected in the serialized `package.json`
43 * file. Positioning will not be taken into account when updating in-memory representations.
44 *
45 * NOTE 2: Property positioning only affects the last property in `propertyPath`. Ancestor
46 * objects' positioning will not be affected.
47 *
48 * @param propertyPath The path of a (possibly nested) property to add/update.
49 * @param value The new value to set the property to.
50 * @param position The desired position for the added/updated property.
51 */
52 PackageJsonUpdate.prototype.addChange = function (propertyPath, value, positioning) {
53 if (positioning === void 0) { positioning = 'unimportant'; }
54 this.ensureNotApplied();
55 this.changes.push([propertyPath, value, positioning]);
56 return this;
57 };
58 /**
59 * Write the recorded changes to the associated `package.json` file (and optionally a
60 * pre-existing, in-memory representation of it).
61 *
62 * @param packageJsonPath The path to the `package.json` file that needs to be updated.
63 * @param parsedJson A pre-existing, in-memory representation of the `package.json` file that
64 * needs to be updated as well.
65 */
66 PackageJsonUpdate.prototype.writeChanges = function (packageJsonPath, parsedJson) {
67 this.ensureNotApplied();
68 this.writeChangesImpl(this.changes, packageJsonPath, parsedJson);
69 this.applied = true;
70 };
71 PackageJsonUpdate.prototype.ensureNotApplied = function () {
72 if (this.applied) {
73 throw new Error('Trying to apply a `PackageJsonUpdate` that has already been applied.');
74 }
75 };
76 return PackageJsonUpdate;
77 }());
78 exports.PackageJsonUpdate = PackageJsonUpdate;
79 /** A `PackageJsonUpdater` that writes directly to the file-system. */
80 var DirectPackageJsonUpdater = /** @class */ (function () {
81 function DirectPackageJsonUpdater(fs) {
82 this.fs = fs;
83 }
84 DirectPackageJsonUpdater.prototype.createUpdate = function () {
85 var _this = this;
86 return new PackageJsonUpdate(function () {
87 var args = [];
88 for (var _i = 0; _i < arguments.length; _i++) {
89 args[_i] = arguments[_i];
90 }
91 return _this.writeChanges.apply(_this, tslib_1.__spreadArray([], tslib_1.__read(args)));
92 });
93 };
94 DirectPackageJsonUpdater.prototype.writeChanges = function (changes, packageJsonPath, preExistingParsedJson) {
95 var e_1, _a;
96 if (changes.length === 0) {
97 throw new Error("No changes to write to '" + packageJsonPath + "'.");
98 }
99 // Read and parse the `package.json` content.
100 // NOTE: We are not using `preExistingParsedJson` (even if specified) to avoid corrupting the
101 // content on disk in case `preExistingParsedJson` is outdated.
102 var parsedJson = this.fs.exists(packageJsonPath) ?
103 JSON.parse(this.fs.readFile(packageJsonPath)) :
104 {};
105 try {
106 // Apply all changes to both the canonical representation (read from disk) and any pre-existing,
107 // in-memory representation.
108 for (var changes_1 = tslib_1.__values(changes), changes_1_1 = changes_1.next(); !changes_1_1.done; changes_1_1 = changes_1.next()) {
109 var _b = tslib_1.__read(changes_1_1.value, 3), propPath = _b[0], value = _b[1], positioning = _b[2];
110 if (propPath.length === 0) {
111 throw new Error("Missing property path for writing value to '" + packageJsonPath + "'.");
112 }
113 applyChange(parsedJson, propPath, value, positioning);
114 if (preExistingParsedJson) {
115 // No need to take property positioning into account for in-memory representations.
116 applyChange(preExistingParsedJson, propPath, value, 'unimportant');
117 }
118 }
119 }
120 catch (e_1_1) { e_1 = { error: e_1_1 }; }
121 finally {
122 try {
123 if (changes_1_1 && !changes_1_1.done && (_a = changes_1.return)) _a.call(changes_1);
124 }
125 finally { if (e_1) throw e_1.error; }
126 }
127 // Ensure the containing directory exists (in case this is a synthesized `package.json` due to a
128 // custom configuration) and write the updated content to disk.
129 this.fs.ensureDir(file_system_1.dirname(packageJsonPath));
130 this.fs.writeFile(packageJsonPath, JSON.stringify(parsedJson, null, 2) + "\n");
131 };
132 return DirectPackageJsonUpdater;
133 }());
134 exports.DirectPackageJsonUpdater = DirectPackageJsonUpdater;
135 // Helpers
136 function applyChange(ctx, propPath, value, positioning) {
137 var lastPropIdx = propPath.length - 1;
138 var lastProp = propPath[lastPropIdx];
139 for (var i = 0; i < lastPropIdx; i++) {
140 var key = propPath[i];
141 var newCtx = ctx.hasOwnProperty(key) ? ctx[key] : (ctx[key] = {});
142 if ((typeof newCtx !== 'object') || (newCtx === null) || Array.isArray(newCtx)) {
143 throw new Error("Property path '" + propPath.join('.') + "' does not point to an object.");
144 }
145 ctx = newCtx;
146 }
147 ctx[lastProp] = value;
148 positionProperty(ctx, lastProp, positioning);
149 }
150 exports.applyChange = applyChange;
151 function movePropBefore(ctx, prop, isNextProp) {
152 var allProps = Object.keys(ctx);
153 var otherProps = allProps.filter(function (p) { return p !== prop; });
154 var nextPropIdx = otherProps.findIndex(isNextProp);
155 var propsToShift = (nextPropIdx === -1) ? [] : otherProps.slice(nextPropIdx);
156 movePropToEnd(ctx, prop);
157 propsToShift.forEach(function (p) { return movePropToEnd(ctx, p); });
158 }
159 function movePropToEnd(ctx, prop) {
160 var value = ctx[prop];
161 delete ctx[prop];
162 ctx[prop] = value;
163 }
164 function positionProperty(ctx, prop, positioning) {
165 switch (positioning) {
166 case 'alphabetic':
167 movePropBefore(ctx, prop, function (p) { return p > prop; });
168 break;
169 case 'unimportant':
170 // Leave the property order unchanged; i.e. newly added properties will be last and existing
171 // ones will remain in their old position.
172 break;
173 default:
174 if ((typeof positioning !== 'object') || (positioning.before === undefined)) {
175 throw new Error("Unknown positioning (" + JSON.stringify(positioning) + ") for property '" + prop + "'.");
176 }
177 movePropBefore(ctx, prop, function (p) { return p === positioning.before; });
178 break;
179 }
180 }
181});
182//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"package_json_updater.js","sourceRoot":"","sources":["../../../../../../../../packages/compiler-cli/ngcc/src/writing/package_json_updater.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 {AbsoluteFsPath, dirname, FileSystem} from '../../../src/ngtsc/file_system';\nimport {JsonObject, JsonValue} from '../packages/entry_point';\n\n\nexport type PackageJsonChange = [string[], JsonValue, PackageJsonPropertyPositioning];\nexport type PackageJsonPropertyPositioning = 'unimportant'|'alphabetic'|{before: string};\nexport type WritePackageJsonChangesFn =\n    (changes: PackageJsonChange[], packageJsonPath: AbsoluteFsPath, parsedJson?: JsonObject) =>\n        void;\n\n/**\n * A utility object that can be used to safely update values in a `package.json` file.\n *\n * Example usage:\n * ```ts\n * const updatePackageJson = packageJsonUpdater\n *     .createUpdate()\n *     .addChange(['name'], 'package-foo')\n *     .addChange(['scripts', 'foo'], 'echo FOOOO...', 'unimportant')\n *     .addChange(['dependencies', 'baz'], '1.0.0', 'alphabetic')\n *     .addChange(['dependencies', 'bar'], '2.0.0', {before: 'baz'})\n *     .writeChanges('/foo/package.json');\n *     // or\n *     // .writeChanges('/foo/package.json', inMemoryParsedJson);\n * ```\n */\nexport interface PackageJsonUpdater {\n  /**\n   * Create a `PackageJsonUpdate` object, which provides a fluent API for batching updates to a\n   * `package.json` file. (Batching the updates is useful, because it avoids unnecessary I/O\n   * operations.)\n   */\n  createUpdate(): PackageJsonUpdate;\n\n  /**\n   * Write a set of changes to the specified `package.json` file (and optionally a pre-existing,\n   * in-memory representation of it).\n   *\n   * @param changes The set of changes to apply.\n   * @param packageJsonPath The path to the `package.json` file that needs to be updated.\n   * @param parsedJson A pre-existing, in-memory representation of the `package.json` file that\n   *                   needs to be updated as well.\n   */\n  writeChanges(\n      changes: PackageJsonChange[], packageJsonPath: AbsoluteFsPath, parsedJson?: JsonObject): void;\n}\n\n/**\n * A utility class providing a fluent API for recording multiple changes to a `package.json` file\n * (and optionally its in-memory parsed representation).\n *\n * NOTE: This class should generally not be instantiated directly; instances are implicitly created\n *       via `PackageJsonUpdater#createUpdate()`.\n */\nexport class PackageJsonUpdate {\n  private changes: PackageJsonChange[] = [];\n  private applied = false;\n\n  constructor(private writeChangesImpl: WritePackageJsonChangesFn) {}\n\n  /**\n   * Record a change to a `package.json` property.\n   *\n   * If the ancestor objects do not yet exist in the `package.json` file, they will be created. The\n   * positioning of the property can also be specified. (If the property already exists, it will be\n   * moved accordingly.)\n   *\n   * NOTE: Property positioning is only guaranteed to be respected in the serialized `package.json`\n   *       file. Positioning will not be taken into account when updating in-memory representations.\n   *\n   * NOTE 2: Property positioning only affects the last property in `propertyPath`. Ancestor\n   *         objects' positioning will not be affected.\n   *\n   * @param propertyPath The path of a (possibly nested) property to add/update.\n   * @param value The new value to set the property to.\n   * @param position The desired position for the added/updated property.\n   */\n  addChange(\n      propertyPath: string[], value: JsonValue,\n      positioning: PackageJsonPropertyPositioning = 'unimportant'): this {\n    this.ensureNotApplied();\n    this.changes.push([propertyPath, value, positioning]);\n    return this;\n  }\n\n  /**\n   * Write the recorded changes to the associated `package.json` file (and optionally a\n   * pre-existing, in-memory representation of it).\n   *\n   * @param packageJsonPath The path to the `package.json` file that needs to be updated.\n   * @param parsedJson A pre-existing, in-memory representation of the `package.json` file that\n   *                   needs to be updated as well.\n   */\n  writeChanges(packageJsonPath: AbsoluteFsPath, parsedJson?: JsonObject): void {\n    this.ensureNotApplied();\n    this.writeChangesImpl(this.changes, packageJsonPath, parsedJson);\n    this.applied = true;\n  }\n\n  private ensureNotApplied() {\n    if (this.applied) {\n      throw new Error('Trying to apply a `PackageJsonUpdate` that has already been applied.');\n    }\n  }\n}\n\n/** A `PackageJsonUpdater` that writes directly to the file-system. */\nexport class DirectPackageJsonUpdater implements PackageJsonUpdater {\n  constructor(private fs: FileSystem) {}\n\n  createUpdate(): PackageJsonUpdate {\n    return new PackageJsonUpdate((...args) => this.writeChanges(...args));\n  }\n\n  writeChanges(\n      changes: PackageJsonChange[], packageJsonPath: AbsoluteFsPath,\n      preExistingParsedJson?: JsonObject): void {\n    if (changes.length === 0) {\n      throw new Error(`No changes to write to '${packageJsonPath}'.`);\n    }\n\n    // Read and parse the `package.json` content.\n    // NOTE: We are not using `preExistingParsedJson` (even if specified) to avoid corrupting the\n    //       content on disk in case `preExistingParsedJson` is outdated.\n    const parsedJson = this.fs.exists(packageJsonPath) ?\n        JSON.parse(this.fs.readFile(packageJsonPath)) as JsonObject :\n        {};\n\n    // Apply all changes to both the canonical representation (read from disk) and any pre-existing,\n    // in-memory representation.\n    for (const [propPath, value, positioning] of changes) {\n      if (propPath.length === 0) {\n        throw new Error(`Missing property path for writing value to '${packageJsonPath}'.`);\n      }\n\n      applyChange(parsedJson, propPath, value, positioning);\n\n      if (preExistingParsedJson) {\n        // No need to take property positioning into account for in-memory representations.\n        applyChange(preExistingParsedJson, propPath, value, 'unimportant');\n      }\n    }\n\n    // Ensure the containing directory exists (in case this is a synthesized `package.json` due to a\n    // custom configuration) and write the updated content to disk.\n    this.fs.ensureDir(dirname(packageJsonPath));\n    this.fs.writeFile(packageJsonPath, `${JSON.stringify(parsedJson, null, 2)}\\n`);\n  }\n}\n\n// Helpers\nexport function applyChange(\n    ctx: JsonObject, propPath: string[], value: JsonValue,\n    positioning: PackageJsonPropertyPositioning): void {\n  const lastPropIdx = propPath.length - 1;\n  const lastProp = propPath[lastPropIdx];\n\n  for (let i = 0; i < lastPropIdx; i++) {\n    const key = propPath[i];\n    const newCtx = ctx.hasOwnProperty(key) ? ctx[key] : (ctx[key] = {});\n\n    if ((typeof newCtx !== 'object') || (newCtx === null) || Array.isArray(newCtx)) {\n      throw new Error(`Property path '${propPath.join('.')}' does not point to an object.`);\n    }\n\n    ctx = newCtx;\n  }\n\n  ctx[lastProp] = value;\n  positionProperty(ctx, lastProp, positioning);\n}\n\nfunction movePropBefore(ctx: JsonObject, prop: string, isNextProp: (p: string) => boolean): void {\n  const allProps = Object.keys(ctx);\n  const otherProps = allProps.filter(p => p !== prop);\n  const nextPropIdx = otherProps.findIndex(isNextProp);\n  const propsToShift = (nextPropIdx === -1) ? [] : otherProps.slice(nextPropIdx);\n\n  movePropToEnd(ctx, prop);\n  propsToShift.forEach(p => movePropToEnd(ctx, p));\n}\n\nfunction movePropToEnd(ctx: JsonObject, prop: string): void {\n  const value = ctx[prop];\n  delete ctx[prop];\n  ctx[prop] = value;\n}\n\nfunction positionProperty(\n    ctx: JsonObject, prop: string, positioning: PackageJsonPropertyPositioning): void {\n  switch (positioning) {\n    case 'alphabetic':\n      movePropBefore(ctx, prop, p => p > prop);\n      break;\n    case 'unimportant':\n      // Leave the property order unchanged; i.e. newly added properties will be last and existing\n      // ones will remain in their old position.\n      break;\n    default:\n      if ((typeof positioning !== 'object') || (positioning.before === undefined)) {\n        throw new Error(\n            `Unknown positioning (${JSON.stringify(positioning)}) for property '${prop}'.`);\n      }\n\n      movePropBefore(ctx, prop, p => p === positioning.before);\n      break;\n  }\n}\n"]}
Note: See TracBrowser for help on using the repository browser.