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"]} |
---|