UNPKG

projen

Version:

CDK for software projects

156 lines 14.8 kB
"use strict"; 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"]}