UNPKG

@decaf-ts/decorator-validation

Version:
93 lines 13.1 kB
"use strict"; Object.defineProperty(exports, "__esModule", { value: true }); exports.PathProxyEngine = void 0; const constants_1 = require("./../constants/index.cjs"); const strings_1 = require("./strings.cjs"); const fallbackGetParent = (target) => { return target[constants_1.VALIDATION_PARENT_KEY]; }; const fallbackGetValue = (target, prop) => { if (!Object.prototype.hasOwnProperty.call(target, prop)) throw new Error((0, strings_1.sf)(constants_1.COMPARISON_ERROR_MESSAGES.PROPERTY_NOT_EXIST, prop)); return target[prop]; }; /** * Standard path resolution utility for accessing nested object properties. * Provides consistent dot-notation access to both parent and child properties * across complex object structures. * * - Dot-notation path resolution ('object.child.property') * - Parent traversal using '../' notation * - Configurable property access behavior * - Null/undefined safety checks */ class PathProxyEngine { /** * Creates a path-aware proxy for the target object * @template T - The type of the target object * @param {T} rootTarget - The target object to proxy * @param opts - Configuration options * @param opts.getValue - Custom function to get property value * @param opts.getParent - Custom function to get parent object * @param opts.ignoreUndefined - Whether to ignore undefined values in paths * @param opts.ignoreNull - Whether to ignore null values in paths * @returns A proxy object with path access capabilities */ static create(rootTarget, opts) { const { getValue, getParent, ignoreUndefined, ignoreNull } = { getParent: fallbackGetParent, getValue: fallbackGetValue, ignoreNull: false, ignoreUndefined: false, ...opts, }; const proxy = new Proxy({}, { get(target, prop) { if (prop === "getValueFromPath") { return function (path) { const parts = PathProxyEngine.parsePath(path); let current = rootTarget; for (let i = 0; i < parts.length; i++) { const part = parts[i]; if (part === "..") { const parent = getParent(current); if (!parent || typeof parent !== "object") { throw new Error((0, strings_1.sf)(constants_1.COMPARISON_ERROR_MESSAGES.CONTEXT_NOT_OBJECT_COMPARISON, i + 1, path)); } current = parent; //PathProxyEngine.create(parentTarget, opts); continue; } current = getValue(current, part); if (!ignoreUndefined && typeof current === "undefined") throw new Error((0, strings_1.sf)(constants_1.COMPARISON_ERROR_MESSAGES.PROPERTY_INVALID, path, part)); if (!ignoreNull && current === null) throw new Error((0, strings_1.sf)(constants_1.COMPARISON_ERROR_MESSAGES.PROPERTY_INVALID, path, part)); } return current; }; } return target[prop]; }, }); // Object.defineProperty(proxy, PROXY_PROP, { // value: true, // overwrite by proxy behavior // enumerable: false, // configurable: false, // writable: false, // }); return proxy; } /** * Parses a path string into individual components * @param path - The path string to parse (e.g., "user.address.city") * @returns An array of path components * @throws Error if the path is invalid */ static parsePath(path) { if (typeof path !== "string" || !path.trim()) throw new Error((0, strings_1.sf)(constants_1.COMPARISON_ERROR_MESSAGES.INVALID_PATH, path)); return path.match(/(\.\.|[^/.]+)/g) || []; } } exports.PathProxyEngine = PathProxyEngine; //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PathProxy.js","sourceRoot":"","sources":["../../src/utils/PathProxy.ts"],"names":[],"mappings":"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","sourcesContent":["import { COMPARISON_ERROR_MESSAGES, VALIDATION_PARENT_KEY } from \"../constants\";\nimport { sf } from \"./strings\";\n\nconst fallbackGetParent = (target: any) => {\n  return target[VALIDATION_PARENT_KEY];\n};\n\nconst fallbackGetValue = (target: any, prop: string) => {\n  if (!Object.prototype.hasOwnProperty.call(target, prop))\n    throw new Error(sf(COMPARISON_ERROR_MESSAGES.PROPERTY_NOT_EXIST, prop));\n  return target[prop];\n};\n\n/**\n * Proxy object that provides path-based access to nested properties\n * @template T - The type of the target object being proxied\n */\nexport type PathProxy<T> = T & {\n  // [PROXY_PROP]: boolean;\n  getValueFromPath: (path: string, fallback?: any) => any;\n};\n\n/**\n * Standard path resolution utility for accessing nested object properties.\n * Provides consistent dot-notation access to both parent and child properties\n * across complex object structures.\n *\n * - Dot-notation path resolution ('object.child.property')\n * - Parent traversal using '../' notation\n * - Configurable property access behavior\n * - Null/undefined safety checks\n */\nexport class PathProxyEngine {\n  /**\n   * Creates a path-aware proxy for the target object\n   * @template T - The type of the target object\n   * @param {T} rootTarget - The target object to proxy\n   * @param opts - Configuration options\n   * @param opts.getValue - Custom function to get property value\n   * @param opts.getParent - Custom function to get parent object\n   * @param opts.ignoreUndefined - Whether to ignore undefined values in paths\n   * @param opts.ignoreNull - Whether to ignore null values in paths\n   * @returns A proxy object with path access capabilities\n   */\n  static create<T extends object>(\n    rootTarget: T,\n    opts?: {\n      getValue?: (target: T, prop: string) => any;\n      getParent?: (target: T) => any;\n      ignoreUndefined: boolean;\n      ignoreNull: boolean;\n    }\n  ): PathProxy<T> {\n    const { getValue, getParent, ignoreUndefined, ignoreNull } = {\n      getParent: fallbackGetParent,\n      getValue: fallbackGetValue,\n      ignoreNull: false,\n      ignoreUndefined: false,\n      ...opts,\n    };\n\n    const proxy = new Proxy({} as any, {\n      get(target, prop) {\n        if (prop === \"getValueFromPath\") {\n          return function (path: string): any {\n            const parts = PathProxyEngine.parsePath(path);\n            let current: any = rootTarget;\n\n            for (let i = 0; i < parts.length; i++) {\n              const part = parts[i];\n              if (part === \"..\") {\n                const parent = getParent(current);\n                if (!parent || typeof parent !== \"object\") {\n                  throw new Error(\n                    sf(\n                      COMPARISON_ERROR_MESSAGES.CONTEXT_NOT_OBJECT_COMPARISON,\n                      i + 1,\n                      path\n                    )\n                  );\n                }\n                current = parent; //PathProxyEngine.create(parentTarget, opts);\n                continue;\n              }\n\n              current = getValue(current, part);\n              if (!ignoreUndefined && typeof current === \"undefined\")\n                throw new Error(\n                  sf(COMPARISON_ERROR_MESSAGES.PROPERTY_INVALID, path, part)\n                );\n\n              if (!ignoreNull && current === null)\n                throw new Error(\n                  sf(COMPARISON_ERROR_MESSAGES.PROPERTY_INVALID, path, part)\n                );\n            }\n\n            return current;\n          };\n        }\n\n        return target[prop];\n      },\n    });\n\n    // Object.defineProperty(proxy, PROXY_PROP, {\n    //   value: true, // overwrite by proxy behavior\n    //   enumerable: false,\n    //   configurable: false,\n    //   writable: false,\n    // });\n\n    return proxy as PathProxy<T>;\n  }\n\n  /**\n   * Parses a path string into individual components\n   * @param path - The path string to parse (e.g., \"user.address.city\")\n   * @returns An array of path components\n   * @throws Error if the path is invalid\n   */\n  private static parsePath(path: string): string[] {\n    if (typeof path !== \"string\" || !path.trim())\n      throw new Error(sf(COMPARISON_ERROR_MESSAGES.INVALID_PATH, path));\n    return path.match(/(\\.\\.|[^/.]+)/g) || [];\n  }\n}\n"]}