@decaf-ts/decorator-validation
Version:
simple decorator based validation engine
89 lines • 12.9 kB
JavaScript
import { COMPARISON_ERROR_MESSAGES, VALIDATION_PARENT_KEY } from "./../constants/index.js";
import { sf } from "./strings.js";
const fallbackGetParent = (target) => {
return target[VALIDATION_PARENT_KEY];
};
const fallbackGetValue = (target, prop) => {
if (!Object.prototype.hasOwnProperty.call(target, prop))
throw new Error(sf(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
*/
export 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(sf(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(sf(COMPARISON_ERROR_MESSAGES.PROPERTY_INVALID, path, part));
if (!ignoreNull && current === null)
throw new Error(sf(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(sf(COMPARISON_ERROR_MESSAGES.INVALID_PATH, path));
return path.match(/(\.\.|[^/.]+)/g) || [];
}
}
//# 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"]}