projen
Version:
CDK for software projects
156 lines • 14.8 kB
JavaScript
;
var _a;
Object.defineProperty(exports, "__esModule", { value: true });
exports.JsonPatch = exports.TestFailureBehavior = void 0;
const JSII_RTTI_SYMBOL_1 = Symbol.for("jsii.rtti");
// inspired by https://github.com/cdk8s-team/cdk8s-core/blob/2.x/src/json-patch.ts
// under Apache 2.0 license
const fast_json_patch_1 = require("fast-json-patch");
var TestFailureBehavior;
(function (TestFailureBehavior) {
/**
* Skip the current patch operation and continue with the next operation.
*/
TestFailureBehavior["SKIP"] = "skip";
/**
* Fail the whole file synthesis.
*/
TestFailureBehavior["FAIL_SYNTHESIS"] = "fail";
})(TestFailureBehavior || (exports.TestFailureBehavior = TestFailureBehavior = {}));
const TEST_FAILURE_BEHAVIOR_SYMBOL = Symbol.for("testFailureBehavior");
/**
* Utility for applying RFC-6902 JSON-Patch to a document.
*
* Use the the `JsonPatch.apply(doc, ...ops)` function to apply a set of
* operations to a JSON document and return the result.
*
* Operations can be created using the factory methods `JsonPatch.add()`,
* `JsonPatch.remove()`, etc.
*
* @example
*
*const output = JsonPatch.apply(input,
* JsonPatch.replace('/world/hi/there', 'goodbye'),
* JsonPatch.add('/world/foo/', 'boom'),
* JsonPatch.remove('/hello'));
*
*/
class JsonPatch {
/**
* Applies a set of JSON-Patch (RFC-6902) operations to `document` and returns the result.
* @param document The document to patch
* @param ops The operations to apply
* @returns The result document
*/
static apply(document, ...ops) {
try {
const result = (0, fast_json_patch_1.applyPatch)(document, (0, fast_json_patch_1.deepClone)(ops.map((o) => o._toJson())));
return result.newDocument;
}
catch (e) {
if (e instanceof fast_json_patch_1.JsonPatchError && e.name === "TEST_OPERATION_FAILED") {
const op = ops[e.index];
if (TEST_FAILURE_BEHAVIOR_SYMBOL in op) {
const failureBehavior = op[TEST_FAILURE_BEHAVIOR_SYMBOL];
if (failureBehavior === TestFailureBehavior.SKIP) {
return document;
}
}
}
throw e;
}
}
/**
* Adds a value to an object or inserts it into an array. In the case of an
* array, the value is inserted before the given index. The - character can be
* used instead of an index to insert at the end of an array.
*
* @example JsonPatch.add('/biscuits/1', { "name": "Ginger Nut" })
*/
static add(path, value) {
return new JsonPatch({
op: "add",
path,
value,
});
}
/**
* Removes a value from an object or array.
*
* @example JsonPatch.remove('/biscuits')
* @example JsonPatch.remove('/biscuits/0')
*/
static remove(path) {
return new JsonPatch({ op: "remove", path });
}
/**
* Replaces a value. Equivalent to a “remove” followed by an “add”.
*
* @example JsonPatch.replace('/biscuits/0/name', 'Chocolate Digestive')
*/
static replace(path, value) {
return new JsonPatch({
op: "replace",
path,
value,
});
}
/**
* Copies a value from one location to another within the JSON document. Both
* from and path are JSON Pointers.
*
* @example JsonPatch.copy('/biscuits/0', '/best_biscuit')
*/
static copy(from, path) {
return new JsonPatch({ op: "copy", from, path });
}
/**
* Moves a value from one location to the other. Both from and path are JSON Pointers.
*
* @example JsonPatch.move('/biscuits', '/cookies')
*/
static move(from, path) {
return new JsonPatch({ op: "move", from, path });
}
/**
* Tests that the specified value is set in the document. If the test fails,
* then the patch as a whole should not apply.
*
* @example JsonPatch.test('/best_biscuit/name', 'Choco Leibniz')
*/
static test(path, value, failureBehavior = TestFailureBehavior.SKIP) {
const patch = new JsonPatch({
op: "test",
path,
value,
});
Object.defineProperty(patch, TEST_FAILURE_BEHAVIOR_SYMBOL, {
writable: false,
value: failureBehavior,
});
return patch;
}
/**
* Escapes a json pointer path
* @param path The raw pointer
* @return the Escaped path
*/
static escapePath(path) {
return (0, fast_json_patch_1.escapePathComponent)(path);
}
constructor(operation) {
this.operation = operation;
}
/**
* Returns the JSON representation of this JSON patch operation.
*
* @internal
*/
_toJson() {
return this.operation;
}
}
exports.JsonPatch = JsonPatch;
_a = JSII_RTTI_SYMBOL_1;
JsonPatch[_a] = { fqn: "projen.JsonPatch", version: "0.99.17" };
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"json-patch.js","sourceRoot":"","sources":["../src/json-patch.ts"],"names":[],"mappings":"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","sourcesContent":["// inspired by https://github.com/cdk8s-team/cdk8s-core/blob/2.x/src/json-patch.ts\n// under Apache 2.0 license\nimport {\n  applyPatch,\n  deepClone,\n  Operation,\n  AddOperation,\n  RemoveOperation,\n  ReplaceOperation,\n  MoveOperation,\n  CopyOperation,\n  TestOperation,\n  escapePathComponent,\n  JsonPatchError,\n} from \"fast-json-patch\";\n\nexport enum TestFailureBehavior {\n  /**\n   * Skip the current patch operation and continue with the next operation.\n   */\n  SKIP = \"skip\",\n  /**\n   * Fail the whole file synthesis.\n   */\n  FAIL_SYNTHESIS = \"fail\",\n}\n\nconst TEST_FAILURE_BEHAVIOR_SYMBOL = Symbol.for(\"testFailureBehavior\");\n\n/**\n * Utility for applying RFC-6902 JSON-Patch to a document.\n *\n * Use the the `JsonPatch.apply(doc, ...ops)` function to apply a set of\n * operations to a JSON document and return the result.\n *\n * Operations can be created using the factory methods `JsonPatch.add()`,\n * `JsonPatch.remove()`, etc.\n *\n * @example\n *\n *const output = JsonPatch.apply(input,\n *  JsonPatch.replace('/world/hi/there', 'goodbye'),\n *  JsonPatch.add('/world/foo/', 'boom'),\n *  JsonPatch.remove('/hello'));\n *\n */\nexport class JsonPatch {\n  /**\n   * Applies a set of JSON-Patch (RFC-6902) operations to `document` and returns the result.\n   * @param document The document to patch\n   * @param ops The operations to apply\n   * @returns The result document\n   */\n  public static apply(document: any, ...ops: JsonPatch[]): any {\n    try {\n      const result = applyPatch(\n        document,\n        deepClone(ops.map((o) => o._toJson())),\n      );\n      return result.newDocument;\n    } catch (e) {\n      if (e instanceof JsonPatchError && e.name === \"TEST_OPERATION_FAILED\") {\n        const op = ops[e.index!];\n        if (TEST_FAILURE_BEHAVIOR_SYMBOL in op) {\n          const failureBehavior = op[TEST_FAILURE_BEHAVIOR_SYMBOL];\n          if (failureBehavior === TestFailureBehavior.SKIP) {\n            return document;\n          }\n        }\n      }\n      throw e;\n    }\n  }\n\n  /**\n   * Adds a value to an object or inserts it into an array. In the case of an\n   * array, the value is inserted before the given index. The - character can be\n   * used instead of an index to insert at the end of an array.\n   *\n   * @example JsonPatch.add('/biscuits/1', { \"name\": \"Ginger Nut\" })\n   */\n  public static add(path: string, value: any) {\n    return new JsonPatch({\n      op: \"add\",\n      path,\n      value,\n    } satisfies AddOperation<any>);\n  }\n\n  /**\n   * Removes a value from an object or array.\n   *\n   * @example JsonPatch.remove('/biscuits')\n   * @example JsonPatch.remove('/biscuits/0')\n   */\n  public static remove(path: string) {\n    return new JsonPatch({ op: \"remove\", path } satisfies RemoveOperation);\n  }\n\n  /**\n   * Replaces a value. Equivalent to a “remove” followed by an “add”.\n   *\n   * @example JsonPatch.replace('/biscuits/0/name', 'Chocolate Digestive')\n   */\n  public static replace(path: string, value: any) {\n    return new JsonPatch({\n      op: \"replace\",\n      path,\n      value,\n    } satisfies ReplaceOperation<any>);\n  }\n\n  /**\n   * Copies a value from one location to another within the JSON document. Both\n   * from and path are JSON Pointers.\n   *\n   * @example JsonPatch.copy('/biscuits/0', '/best_biscuit')\n   */\n  public static copy(from: string, path: string) {\n    return new JsonPatch({ op: \"copy\", from, path } satisfies CopyOperation);\n  }\n\n  /**\n   * Moves a value from one location to the other. Both from and path are JSON Pointers.\n   *\n   * @example JsonPatch.move('/biscuits', '/cookies')\n   */\n  public static move(from: string, path: string) {\n    return new JsonPatch({ op: \"move\", from, path } satisfies MoveOperation);\n  }\n\n  /**\n   * Tests that the specified value is set in the document. If the test fails,\n   * then the patch as a whole should not apply.\n   *\n   * @example JsonPatch.test('/best_biscuit/name', 'Choco Leibniz')\n   */\n  public static test(\n    path: string,\n    value: any,\n    failureBehavior: TestFailureBehavior = TestFailureBehavior.SKIP,\n  ) {\n    const patch = new JsonPatch({\n      op: \"test\",\n      path,\n      value,\n    } satisfies TestOperation<any>);\n    Object.defineProperty(patch, TEST_FAILURE_BEHAVIOR_SYMBOL, {\n      writable: false,\n      value: failureBehavior,\n    });\n    return patch;\n  }\n\n  /**\n   * Escapes a json pointer path\n   * @param path The raw pointer\n   * @return the Escaped path\n   */\n  public static escapePath(path: string): string {\n    return escapePathComponent(path);\n  }\n\n  private constructor(private readonly operation: Operation) {}\n\n  /**\n   * Returns the JSON representation of this JSON patch operation.\n   *\n   * @internal\n   */\n  public _toJson(): any {\n    return this.operation;\n  }\n}\n"]}