@decaf-ts/logging 0.3.12 → 0.3.14

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (43) hide show
  1. package/dist/logging.cjs +542 -99
  2. package/dist/logging.esm.cjs +539 -100
  3. package/lib/LoggedClass.cjs +9 -12
  4. package/lib/LoggedClass.d.ts +6 -10
  5. package/lib/constants.cjs +40 -8
  6. package/lib/constants.d.ts +35 -7
  7. package/lib/decorators.cjs +114 -50
  8. package/lib/decorators.d.ts +58 -43
  9. package/lib/environment.cjs +73 -22
  10. package/lib/environment.d.ts +46 -29
  11. package/lib/esm/LoggedClass.d.ts +6 -10
  12. package/lib/esm/LoggedClass.js +9 -12
  13. package/lib/esm/constants.d.ts +35 -7
  14. package/lib/esm/constants.js +40 -8
  15. package/lib/esm/decorators.d.ts +58 -43
  16. package/lib/esm/decorators.js +113 -50
  17. package/lib/esm/environment.d.ts +46 -29
  18. package/lib/esm/environment.js +73 -22
  19. package/lib/esm/filters/LogFilter.d.ts +37 -0
  20. package/lib/esm/filters/LogFilter.js +30 -1
  21. package/lib/esm/filters/PatternFilter.d.ts +46 -0
  22. package/lib/esm/filters/PatternFilter.js +41 -1
  23. package/lib/esm/index.d.ts +7 -10
  24. package/lib/esm/index.js +8 -11
  25. package/lib/esm/logging.d.ts +14 -0
  26. package/lib/esm/logging.js +22 -1
  27. package/lib/esm/time.d.ts +149 -0
  28. package/lib/esm/time.js +212 -0
  29. package/lib/esm/types.d.ts +89 -51
  30. package/lib/esm/types.js +1 -1
  31. package/lib/filters/LogFilter.cjs +30 -1
  32. package/lib/filters/LogFilter.d.ts +37 -0
  33. package/lib/filters/PatternFilter.cjs +41 -1
  34. package/lib/filters/PatternFilter.d.ts +46 -0
  35. package/lib/index.cjs +8 -11
  36. package/lib/index.d.ts +7 -10
  37. package/lib/logging.cjs +22 -1
  38. package/lib/logging.d.ts +14 -0
  39. package/lib/time.cjs +217 -0
  40. package/lib/time.d.ts +149 -0
  41. package/lib/types.cjs +1 -1
  42. package/lib/types.d.ts +89 -51
  43. package/package.json +2 -2
@@ -1,11 +1,14 @@
1
1
  import { LogLevel } from "./constants";
2
+ export type ArgFormatFunction = (...args: any[]) => string;
3
+ export type ReturnFormatFunction = (e?: Error, result?: any) => string;
2
4
  /**
3
- * @description Method decorator for logging function calls
4
- * @summary Creates a decorator that logs method calls with specified level, benchmarking, and verbosity
5
- * @param {LogLevel} level - The log level to use (default: LogLevel.info)
6
- * @param {boolean} [benchmark=false] - Whether to log execution time (default: false)
7
- * @param {number} [verbosity=0] - The verbosity level for the log messages (default: 0)
8
- * @return {Function} A method decorator that wraps the original method with logging
5
+ * @description Method decorator for logging function calls.
6
+ * @summary Wraps class methods to automatically log entry, exit, timing, and optional custom messages at a configurable {@link LogLevel}.
7
+ * @param {LogLevel} level - Log level applied to the generated log statements (defaults to `LogLevel.info`).
8
+ * @param {number} [verbosity=0] - Verbosity threshold required for the entry log to appear.
9
+ * @param {ArgFormatFunction} [entryMessage] - Formatter invoked with the original method arguments to describe the invocation.
10
+ * @param {ReturnFormatFunction} [exitMessage] - Optional formatter that describes the outcome or failure of the call.
11
+ * @return {function(any, any, PropertyDescriptor): void} Method decorator proxy that injects logging behavior.
9
12
  * @function log
10
13
  * @mermaid
11
14
  * sequenceDiagram
@@ -30,64 +33,76 @@ import { LogLevel } from "./constants";
30
33
  * end
31
34
  * @category Method Decorators
32
35
  */
33
- export declare function log(level?: LogLevel, benchmark?: boolean, verbosity?: number): (target: any, propertyKey?: any, descriptor?: PropertyDescriptor) => void;
36
+ export declare function log(level?: LogLevel, verbosity?: number, entryMessage?: ArgFormatFunction, exitMessage?: ReturnFormatFunction): (target: any, propertyKey?: any, descriptor?: any) => any;
34
37
  /**
35
- * @description Method decorator for logging function calls with debug level
36
- * @summary Convenience wrapper around the log decorator that uses LogLevel.debug
37
- * @param {boolean} [benchmark=false] - Whether to log execution time (default: false)
38
- * @return {Function} A method decorator that wraps the original method with debug logging
38
+ * @description Method decorator that records execution time at the benchmark level.
39
+ * @summary Wraps the target method to emit {@link Logger.benchmark} entries capturing completion time or failure latency.
40
+ * @return {function(any, any, PropertyDescriptor): void} Method decorator proxy that benchmarks the original implementation.
41
+ * @function benchmark
42
+ * @mermaid
43
+ * sequenceDiagram
44
+ * participant Caller
45
+ * participant Decorator as benchmark
46
+ * participant Method as Original Method
47
+ * Caller->>Decorator: invoke()
48
+ * Decorator->>Method: Reflect.apply(...)
49
+ * alt Promise result
50
+ * Method-->>Decorator: Promise
51
+ * Decorator->>Decorator: attach then()
52
+ * Decorator->>Decorator: log completion duration
53
+ * else Synchronous result
54
+ * Method-->>Decorator: value
55
+ * Decorator->>Decorator: log completion duration
56
+ * end
57
+ * Decorator-->>Caller: return result
58
+ * @category Method Decorators
59
+ */
60
+ export declare function benchmark(): (target: any, propertyKey?: any, descriptor?: any) => any;
61
+ /**
62
+ * @description Method decorator for logging function calls with debug level.
63
+ * @summary Convenience wrapper around {@link log} that logs using `LogLevel.debug`.
64
+ * @return {function(any, any, PropertyDescriptor): void} Debug-level logging decorator.
39
65
  * @function debug
40
66
  * @category Method Decorators
41
67
  */
42
- export declare function debug(benchmark?: boolean): (target: any, propertyKey?: any, descriptor?: PropertyDescriptor) => void;
68
+ export declare function debug(): (target: any, propertyKey?: any, descriptor?: any) => any;
43
69
  /**
44
- * @description Method decorator for logging function calls with info level
45
- * @summary Convenience wrapper around the log decorator that uses LogLevel.info
46
- * @param {boolean} [benchmark=false] - Whether to log execution time (default: false)
47
- * @return {Function} A method decorator that wraps the original method with info logging
70
+ * @description Method decorator for logging function calls with info level.
71
+ * @summary Convenience wrapper around {@link log} that logs using `LogLevel.info`.
72
+ * @return {function(any, any, PropertyDescriptor): void} Info-level logging decorator.
48
73
  * @function info
49
74
  * @category Method Decorators
50
75
  */
51
- export declare function info(benchmark?: boolean): (target: any, propertyKey?: any, descriptor?: PropertyDescriptor) => void;
76
+ export declare function info(): (target: any, propertyKey?: any, descriptor?: any) => any;
52
77
  /**
53
- * @description Method decorator for logging function calls with silly level
54
- * @summary Convenience wrapper around the log decorator that uses LogLevel.silly
55
- * @param {boolean} [benchmark=false] - Whether to log execution time (default: false)
56
- * @return {Function} A method decorator that wraps the original method with silly logging
78
+ * @description Method decorator for logging function calls with silly level.
79
+ * @summary Convenience wrapper around {@link log} that logs using `LogLevel.silly`.
80
+ * @return {function(any, any, PropertyDescriptor): void} Silly-level logging decorator.
57
81
  * @function silly
58
82
  * @category Method Decorators
59
83
  */
60
- export declare function silly(benchmark?: boolean): (target: any, propertyKey?: any, descriptor?: PropertyDescriptor) => void;
84
+ export declare function silly(): (target: any, propertyKey?: any, descriptor?: any) => any;
61
85
  /**
62
- * @description Method decorator for logging function calls with verbose level
63
- * @summary Convenience wrapper around the log decorator that uses LogLevel.verbose with configurable verbosity
64
- * @return {Function} A method decorator that wraps the original method with verbose logging
86
+ * @description Method decorator for logging function calls with verbose level.
87
+ * @summary Convenience wrapper around {@link log} that logs using `LogLevel.verbose` with configurable verbosity.
88
+ * @return {function(any, any, PropertyDescriptor): void} Verbose logging decorator.
65
89
  * @function verbose
90
+ * @category Method Decorators
66
91
  */
67
92
  export declare function verbose(): (target: any, propertyKey?: any, descriptor?: any) => void;
68
93
  /**
69
- * @description Method decorator for logging function calls with verbose level
70
- * @summary Convenience wrapper around the log decorator that uses LogLevel.verbose with configurable verbosity
71
- * @param {boolean} benchmark - Whether to log execution time
72
- * @return {Function} A method decorator that wraps the original method with verbose logging
73
- * @function verbose
74
- */
75
- export declare function verbose(benchmark: boolean): (target: any, propertyKey?: any, descriptor?: any) => void;
76
- /**
77
- * @description Method decorator for logging function calls with verbose level
78
- * @summary Convenience wrapper around the log decorator that uses LogLevel.verbose with configurable verbosity
79
- * @param {number} verbosity - The verbosity level for the log messages (default: 0)
80
- * @return {Function} A method decorator that wraps the original method with verbose logging
94
+ * @description Method decorator for logging function calls with verbose level.
95
+ * @summary Convenience wrapper around {@link log} that logs using `LogLevel.verbose` while toggling benchmarking.
96
+ * @return {function(any, PropertyDescriptor): void} Verbose logging decorator.
81
97
  * @function verbose
82
98
  * @category Method Decorators
83
99
  */
84
- export declare function verbose(verbosity: number | boolean): (target: any, propertyKey?: any, descriptor?: any) => void;
100
+ export declare function verbose(): (target: any, propertyKey?: any, descriptor?: any) => void;
85
101
  /**
86
- * @description Creates a decorator that makes a method non-configurable
87
- * @summary This decorator prevents a method from being overridden by making it non-configurable.
88
- * It throws an error if used on anything other than a method.
89
- * @return {Function} A decorator function that can be applied to methods
102
+ * @description Creates a decorator that makes a method non-configurable.
103
+ * @summary Prevents overriding by marking the method descriptor as non-configurable, throwing if applied to non-method targets.
104
+ * @return {function(object, any, PropertyDescriptor): PropertyDescriptor|undefined} Decorator that hardens the method descriptor.
90
105
  * @function final
91
106
  * @category Method Decorators
92
107
  */
93
- export declare function final(): (target: object, propertyKey?: any, descriptor?: PropertyDescriptor) => PropertyDescriptor;
108
+ export declare function final(): (target: object, propertyKey?: any, descriptor?: any) => any;
@@ -6,13 +6,33 @@ const text_1 = require("./text.cjs");
6
6
  const web_1 = require("./web.cjs");
7
7
  const constants_1 = require("./constants.cjs");
8
8
  /**
9
- * @class Environment
10
- * @extends {ObjectAccumulator<T>}
9
+ * @description Environment accumulator that lazily reads from runtime sources.
10
+ * @summary Extends {@link ObjectAccumulator} to merge configuration objects while resolving values from Node or browser environment variables on demand.
11
11
  * @template T
12
- * @description A class representing an environment with accumulation capabilities.
13
- * @summary Manages environment-related data and provides methods for accumulation and key retrieval.
14
- * @param {T} [initialData] - The initial data to populate the environment with.
12
+ * @class Environment
13
+ * @example
14
+ * const Config = Environment.accumulate({ logging: { level: "info" } });
15
+ * console.log(Config.logging.level);
16
+ * console.log(String(Config.logging.level)); // => LOGGING__LEVEL key when serialized
17
+ * @mermaid
18
+ * sequenceDiagram
19
+ * participant Client
20
+ * participant Env as Environment
21
+ * participant Process as process.env
22
+ * participant Browser as globalThis.ENV
23
+ * Client->>Env: accumulate(partialConfig)
24
+ * Env->>Env: expand(values)
25
+ * Client->>Env: Config.logging.level
26
+ * alt Browser runtime
27
+ * Env->>Browser: lookup ENV key
28
+ * Browser-->>Env: resolved value
29
+ * else Node runtime
30
+ * Env->>Process: lookup ENV key
31
+ * Process-->>Env: resolved value
32
+ * end
33
+ * Env-->>Client: merged value
15
34
  */
35
+ const EmptyValue = Symbol("EnvironmentEmpty");
16
36
  class Environment extends typed_object_accumulator_1.ObjectAccumulator {
17
37
  /**
18
38
  * @static
@@ -26,10 +46,10 @@ class Environment extends typed_object_accumulator_1.ObjectAccumulator {
26
46
  super();
27
47
  }
28
48
  /**
29
- * @description Retrieves a value from the environment
30
- * @summary Gets a value from the environment variables, handling browser and Node.js environments differently
31
- * @param {string} k - The key to retrieve from the environment
32
- * @return {unknown} The value from the environment, or undefined if not found
49
+ * @description Retrieves a value from the runtime environment.
50
+ * @summary Handles browser and Node.js environments by normalizing keys and parsing values.
51
+ * @param {string} k - Key to resolve from the environment.
52
+ * @return {unknown} Value resolved from the environment or `undefined` when absent.
33
53
  */
34
54
  fromEnv(k) {
35
55
  let env;
@@ -43,6 +63,12 @@ class Environment extends typed_object_accumulator_1.ObjectAccumulator {
43
63
  }
44
64
  return this.parseEnvValue(env[k]);
45
65
  }
66
+ /**
67
+ * @description Converts stringified environment values into native types.
68
+ * @summary Interprets booleans and numbers while leaving other types unchanged.
69
+ * @param {unknown} val - Raw value retrieved from the environment.
70
+ * @return {unknown} Parsed value converted to boolean, number, or left as-is.
71
+ */
46
72
  parseEnvValue(val) {
47
73
  if (typeof val !== "string")
48
74
  return val;
@@ -56,10 +82,10 @@ class Environment extends typed_object_accumulator_1.ObjectAccumulator {
56
82
  return val;
57
83
  }
58
84
  /**
59
- * @description Expands an object into the environment
60
- * @summary Defines properties on the environment object that can be accessed as getters and setters
61
- * @template V - Type of the object being expanded
62
- * @param {V} value - The object to expand into the environment
85
+ * @description Expands an object into the environment.
86
+ * @summary Defines lazy properties that first consult runtime variables before falling back to seeded values.
87
+ * @template V - Type of the object being expanded.
88
+ * @param {V} value - Object to expose through environment getters and setters.
63
89
  * @return {void}
64
90
  */
65
91
  expand(value) {
@@ -72,9 +98,9 @@ class Environment extends typed_object_accumulator_1.ObjectAccumulator {
72
98
  if (v && typeof v === "object") {
73
99
  return Environment.buildEnvProxy(v, [k]);
74
100
  }
75
- // If the model provides an empty string, expose a proxy that composes ENV keys
101
+ // If the model provides an empty string, mark with EmptyValue so instance proxy can return undefined without enabling key composition
76
102
  if (v === "") {
77
- return Environment.buildEnvProxy(undefined, [k]);
103
+ return EmptyValue;
78
104
  }
79
105
  return v;
80
106
  },
@@ -90,10 +116,10 @@ class Environment extends typed_object_accumulator_1.ObjectAccumulator {
90
116
  * @protected
91
117
  * @static
92
118
  * @description Retrieves or creates the singleton instance of the Environment class.
93
- * @summary Ensures only one instance of the Environment class exists.
119
+ * @summary Ensures only one {@link Environment} instance is created, wrapping it in a proxy to compose ENV keys on demand.
94
120
  * @template E
95
- * @param {...unknown[]} args - Arguments to pass to the factory function if a new instance is created.
96
- * @return {E} The singleton instance of the Environment class.
121
+ * @param {...unknown[]} args - Arguments forwarded to the factory when instantiating the singleton.
122
+ * @return {E} Singleton environment instance.
97
123
  */
98
124
  static instance(...args) {
99
125
  if (!Environment._instance) {
@@ -101,6 +127,8 @@ class Environment extends typed_object_accumulator_1.ObjectAccumulator {
101
127
  const proxied = new Proxy(base, {
102
128
  get(target, prop, receiver) {
103
129
  const value = Reflect.get(target, prop, receiver);
130
+ if (value === EmptyValue)
131
+ return undefined;
104
132
  if (typeof value !== "undefined")
105
133
  return value;
106
134
  if (typeof prop === "string") {
@@ -119,10 +147,11 @@ class Environment extends typed_object_accumulator_1.ObjectAccumulator {
119
147
  /**
120
148
  * @static
121
149
  * @description Accumulates the given value into the environment.
122
- * @summary Adds new properties to the environment from the provided object.
150
+ * @summary Adds new properties, hiding raw descriptors to avoid leaking enumeration semantics.
151
+ * @template T
123
152
  * @template V
124
- * @param {V} value - The object to accumulate into the environment.
125
- * @return {V} The updated environment instance.
153
+ * @param {V} value - Object to merge into the environment.
154
+ * @return {Environment} Updated environment reference.
126
155
  */
127
156
  static accumulate(value) {
128
157
  const instance = Environment.instance();
@@ -137,9 +166,22 @@ class Environment extends typed_object_accumulator_1.ObjectAccumulator {
137
166
  });
138
167
  return instance.accumulate(value);
139
168
  }
169
+ /**
170
+ * @description Retrieves a value using a dot-path key from the accumulated environment.
171
+ * @summary Delegates to the singleton instance to access stored configuration.
172
+ * @param {string} key - Key to resolve from the environment store.
173
+ * @return {unknown} Stored value corresponding to the provided key.
174
+ */
140
175
  static get(key) {
141
176
  return Environment._instance.get(key);
142
177
  }
178
+ /**
179
+ * @description Builds a proxy that composes environment keys for nested properties.
180
+ * @summary Allows chained property access to emit uppercase ENV identifiers while honoring existing runtime overrides.
181
+ * @param {any} current - Seed model segment used when projecting nested structures.
182
+ * @param {string[]} path - Accumulated path segments leading to the proxy.
183
+ * @return {any} Proxy that resolves environment values or composes additional proxies for deeper paths.
184
+ */
143
185
  static buildEnvProxy(current, path) {
144
186
  const buildKey = (p) => p.map((seg) => (0, text_1.toENVFormat)(seg)).join(constants_1.ENV_PATH_DELIMITER);
145
187
  // Helper to read from the active environment given a composed key
@@ -176,6 +218,9 @@ class Environment extends typed_object_accumulator_1.ObjectAccumulator {
176
218
  const isNextObject = nextModel && typeof nextModel === "object";
177
219
  if (isNextObject)
178
220
  return Environment.buildEnvProxy(nextModel, nextPath);
221
+ // If the model marks this leaf as an empty string, treat as undefined (no proxy)
222
+ if (nextModel === "")
223
+ return undefined;
179
224
  // Always return a proxy for further path composition when no ENV value;
180
225
  // do not surface primitive model defaults here (this API is for key composition).
181
226
  return Environment.buildEnvProxy(undefined, nextPath);
@@ -209,9 +254,15 @@ class Environment extends typed_object_accumulator_1.ObjectAccumulator {
209
254
  }
210
255
  }
211
256
  exports.Environment = Environment;
257
+ /**
258
+ * @description Singleton environment instance seeded with default logging configuration.
259
+ * @summary Combines {@link DefaultLoggingConfig} with runtime environment variables to provide consistent logging defaults across platforms.
260
+ * @const LoggedEnvironment
261
+ * @memberOf module:Logging
262
+ */
212
263
  exports.LoggedEnvironment = Environment.accumulate(Object.assign({}, constants_1.DefaultLoggingConfig, {
213
264
  env: ((0, web_1.isBrowser)() && globalThis[constants_1.BrowserEnvKey]
214
265
  ? globalThis[constants_1.BrowserEnvKey]["NODE_ENV"]
215
266
  : globalThis.process.env["NODE_ENV"]) || "development",
216
267
  }));
217
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"environment.js","sourceRoot":"","sources":["../src/environment.ts"],"names":[],"mappings":"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","sourcesContent":["import { ObjectAccumulator } from \"typed-object-accumulator\";\nimport { toENVFormat } from \"./text\";\nimport { isBrowser } from \"./web\";\nimport {\n  BrowserEnvKey,\n  DefaultLoggingConfig,\n  ENV_PATH_DELIMITER,\n} from \"./constants\";\n\n/**\n * @description Factory type for creating Environment instances.\n * @summary Defines a function type that creates and returns Environment instances.\n *\n * @template T - The type of object the Environment will accumulate.\n * @template E - The specific Environment type to be created, extending Environment<T>.\n * @typedef {function(...unknown[]): E} EnvironmentFactory\n * @memberOf module:Logging\n */\nexport type EnvironmentFactory<T extends object, E extends Environment<T>> = (\n  ...args: unknown[]\n) => E;\n\n/**\n * @class Environment\n * @extends {ObjectAccumulator<T>}\n * @template T\n * @description A class representing an environment with accumulation capabilities.\n * @summary Manages environment-related data and provides methods for accumulation and key retrieval.\n * @param {T} [initialData] - The initial data to populate the environment with.\n */\nexport class Environment<T extends object> extends ObjectAccumulator<T> {\n  /**\n   * @static\n   * @protected\n   * @description A factory function for creating Environment instances.\n   * @summary Defines how new instances of the Environment class should be created.\n   * @return {Environment<any>} A new instance of the Environment class.\n   */\n  protected static factory: EnvironmentFactory<any, any> =\n    (): Environment<any> => new Environment();\n\n  /**\n   * @static\n   * @private\n   * @description The singleton instance of the Environment class.\n   * @type {Environment<any>}\n   */\n  private static _instance: Environment<any>;\n\n  protected constructor() {\n    super();\n  }\n\n  /**\n   * @description Retrieves a value from the environment\n   * @summary Gets a value from the environment variables, handling browser and Node.js environments differently\n   * @param {string} k - The key to retrieve from the environment\n   * @return {unknown} The value from the environment, or undefined if not found\n   */\n  protected fromEnv(k: string) {\n    let env: Record<string, unknown>;\n    if (isBrowser()) {\n      env =\n        (\n          globalThis as typeof globalThis & {\n            [BrowserEnvKey]: Record<string, any>;\n          }\n        )[BrowserEnvKey] || {};\n    } else {\n      env = globalThis.process.env;\n      k = toENVFormat(k);\n    }\n    return this.parseEnvValue(env[k]);\n  }\n\n  protected parseEnvValue(val: unknown) {\n    if (typeof val !== \"string\") return val;\n    if (val === \"true\") return true;\n    if (val === \"false\") return false;\n    const result = parseFloat(val);\n    if (!isNaN(result)) return result;\n    return val;\n  }\n\n  /**\n   * @description Expands an object into the environment\n   * @summary Defines properties on the environment object that can be accessed as getters and setters\n   * @template V - Type of the object being expanded\n   * @param {V} value - The object to expand into the environment\n   * @return {void}\n   */\n  protected override expand<V extends object>(value: V): void {\n    Object.entries(value).forEach(([k, v]) => {\n      Object.defineProperty(this, k, {\n        get: () => {\n          const fromEnv = this.fromEnv(k);\n          if (typeof fromEnv !== \"undefined\") return fromEnv;\n          if (v && typeof v === \"object\") {\n            return Environment.buildEnvProxy(v as any, [k]);\n          }\n          // If the model provides an empty string, expose a proxy that composes ENV keys\n          if (v === \"\") {\n            return Environment.buildEnvProxy(undefined, [k]);\n          }\n          return v;\n        },\n        set: (val: V[keyof V]) => {\n          v = val;\n        },\n        configurable: true,\n        enumerable: true,\n      });\n    });\n  }\n\n  /**\n   * @protected\n   * @static\n   * @description Retrieves or creates the singleton instance of the Environment class.\n   * @summary Ensures only one instance of the Environment class exists.\n   * @template E\n   * @param {...unknown[]} args - Arguments to pass to the factory function if a new instance is created.\n   * @return {E} The singleton instance of the Environment class.\n   */\n  protected static instance<E extends Environment<any>>(...args: unknown[]): E {\n    if (!Environment._instance) {\n      const base = Environment.factory(...args) as E;\n      const proxied = new Proxy(base as any, {\n        get(target, prop, receiver) {\n          const value = Reflect.get(target, prop, receiver);\n          if (typeof value !== \"undefined\") return value;\n          if (typeof prop === \"string\") {\n            // Avoid interfering with logging config lookups for optional fields like 'app'\n            if (prop === \"app\") return undefined;\n            return Environment.buildEnvProxy(undefined, [prop]);\n          }\n          return value;\n        },\n      });\n      Environment._instance = proxied as any;\n    }\n    return Environment._instance as E;\n  }\n\n  /**\n   * @static\n   * @description Accumulates the given value into the environment.\n   * @summary Adds new properties to the environment from the provided object.\n   * @template V\n   * @param {V} value - The object to accumulate into the environment.\n   * @return {V} The updated environment instance.\n   */\n  static accumulate<V extends object>(\n    value: V\n  ): typeof Environment._instance &\n    V &\n    ObjectAccumulator<typeof Environment._instance & V> {\n    const instance = Environment.instance();\n    Object.keys(instance as any).forEach((key) => {\n      const desc = Object.getOwnPropertyDescriptor(instance as any, key);\n      if (desc && desc.configurable && desc.enumerable) {\n        Object.defineProperty(instance as any, key, {\n          ...desc,\n          enumerable: false,\n        });\n      }\n    });\n    return instance.accumulate(value);\n  }\n\n  static get(key: string) {\n    return Environment._instance.get(key);\n  }\n\n  private static buildEnvProxy(current: any, path: string[]): any {\n    const buildKey = (p: string[]) =>\n      p.map((seg) => toENVFormat(seg)).join(ENV_PATH_DELIMITER);\n\n    // Helper to read from the active environment given a composed key\n    const readEnv = (key: string): unknown => {\n      if (isBrowser()) {\n        const env = (\n          globalThis as typeof globalThis & {\n            [BrowserEnvKey]?: Record<string, unknown>;\n          }\n        )[BrowserEnvKey];\n        return env ? env[key] : undefined;\n      }\n      return (globalThis as any)?.process?.env?.[key];\n    };\n\n    const handler: ProxyHandler<any> = {\n      get(_target, prop: string | symbol) {\n        if (prop === Symbol.toPrimitive) {\n          return () => buildKey(path);\n        }\n        if (prop === \"toString\") {\n          return () => buildKey(path);\n        }\n        if (prop === \"valueOf\") {\n          return () => buildKey(path);\n        }\n        if (typeof prop === \"symbol\") return undefined;\n\n        const nextModel =\n          current && Object.prototype.hasOwnProperty.call(current, prop)\n            ? (current as any)[prop]\n            : undefined;\n        const nextPath = [...path, prop];\n        const composedKey = buildKey(nextPath);\n\n        // If an ENV value exists for this path, return it directly\n        const envValue = readEnv(composedKey);\n        if (typeof envValue !== \"undefined\") return envValue;\n\n        // Otherwise, if the model has an object at this path, keep drilling with a proxy\n        const isNextObject = nextModel && typeof nextModel === \"object\";\n        if (isNextObject) return Environment.buildEnvProxy(nextModel, nextPath);\n\n        // Always return a proxy for further path composition when no ENV value;\n        // do not surface primitive model defaults here (this API is for key composition).\n        return Environment.buildEnvProxy(undefined, nextPath);\n      },\n      ownKeys() {\n        return current ? Reflect.ownKeys(current) : [];\n      },\n      getOwnPropertyDescriptor(_t, p) {\n        if (!current) return undefined as any;\n        if (Object.prototype.hasOwnProperty.call(current, p)) {\n          return { enumerable: true, configurable: true } as PropertyDescriptor;\n        }\n        return undefined as any;\n      },\n    };\n\n    const target = {} as any;\n    return new Proxy(target, handler);\n  }\n\n  /**\n   * @static\n   * @description Retrieves the keys of the environment, optionally converting them to ENV format.\n   * @summary Gets all keys in the environment, with an option to format them for environment variables.\n   * @param {boolean} [toEnv=true] - Whether to convert the keys to ENV format.\n   * @return {string[]} An array of keys from the environment.\n   */\n  static keys(toEnv: boolean = true): string[] {\n    return Environment.instance()\n      .keys()\n      .map((k) => (toEnv ? toENVFormat(k) : k));\n  }\n}\n\nexport const LoggedEnvironment = Environment.accumulate(\n  Object.assign({}, DefaultLoggingConfig, {\n    env:\n      (isBrowser() && (globalThis as any)[BrowserEnvKey]\n        ? (globalThis as any)[BrowserEnvKey][\"NODE_ENV\"]\n        : (globalThis as any).process.env[\"NODE_ENV\"]) || \"development\",\n  })\n);\n"]}
268
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"environment.js","sourceRoot":"","sources":["../src/environment.ts"],"names":[],"mappings":";;;AAAA,uEAA6D;AAC7D,qCAAqC;AACrC,mCAAkC;AAClC,+CAIqB;AAcrB;;;;;;;;;;;;;;;;;;;;;;;;;;GA0BG;AACH,MAAM,UAAU,GAAG,MAAM,CAAC,kBAAkB,CAAC,CAAC;AAE9C,MAAa,WAA8B,SAAQ,4CAAoB;IACrE;;;;;;OAMG;aACc,YAAO,GACtB,GAAqB,EAAE,CAAC,IAAI,WAAW,EAAE,CAAC;IAU5C;QACE,KAAK,EAAE,CAAC;IACV,CAAC;IAED;;;;;OAKG;IACO,OAAO,CAAC,CAAS;QACzB,IAAI,GAA4B,CAAC;QACjC,IAAI,IAAA,eAAS,GAAE,EAAE,CAAC;YAChB,GAAG;gBAEC,UAGD,CAAC,yBAAa,CAAC,IAAI,EAAE,CAAC;QAC3B,CAAC;aAAM,CAAC;YACN,GAAG,GAAG,UAAU,CAAC,OAAO,CAAC,GAAG,CAAC;YAC7B,CAAC,GAAG,IAAA,kBAAW,EAAC,CAAC,CAAC,CAAC;QACrB,CAAC;QACD,OAAO,IAAI,CAAC,aAAa,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,CAAC;IACpC,CAAC;IAED;;;;;OAKG;IACO,aAAa,CAAC,GAAY;QAClC,IAAI,OAAO,GAAG,KAAK,QAAQ;YAAE,OAAO,GAAG,CAAC;QACxC,IAAI,GAAG,KAAK,MAAM;YAAE,OAAO,IAAI,CAAC;QAChC,IAAI,GAAG,KAAK,OAAO;YAAE,OAAO,KAAK,CAAC;QAClC,MAAM,MAAM,GAAG,UAAU,CAAC,GAAG,CAAC,CAAC;QAC/B,IAAI,CAAC,KAAK,CAAC,MAAM,CAAC;YAAE,OAAO,MAAM,CAAC;QAClC,OAAO,GAAG,CAAC;IACb,CAAC;IAED;;;;;;OAMG;IACgB,MAAM,CAAmB,KAAQ;QAClD,MAAM,CAAC,OAAO,CAAC,KAAK,CAAC,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC,CAAC,EAAE,EAAE;YACvC,MAAM,CAAC,cAAc,CAAC,IAAI,EAAE,CAAC,EAAE;gBAC7B,GAAG,EAAE,GAAG,EAAE;oBACR,MAAM,OAAO,GAAG,IAAI,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC;oBAChC,IAAI,OAAO,OAAO,KAAK,WAAW;wBAAE,OAAO,OAAO,CAAC;oBACnD,IAAI,CAAC,IAAI,OAAO,CAAC,KAAK,QAAQ,EAAE,CAAC;wBAC/B,OAAO,WAAW,CAAC,aAAa,CAAC,CAAQ,EAAE,CAAC,CAAC,CAAC,CAAC,CAAC;oBAClD,CAAC;oBACD,sIAAsI;oBACtI,IAAI,CAAC,KAAK,EAAE,EAAE,CAAC;wBACb,OAAO,UAAmC,CAAC;oBAC7C,CAAC;oBACD,OAAO,CAAC,CAAC;gBACX,CAAC;gBACD,GAAG,EAAE,CAAC,GAAe,EAAE,EAAE;oBACvB,CAAC,GAAG,GAAG,CAAC;gBACV,CAAC;gBACD,YAAY,EAAE,IAAI;gBAClB,UAAU,EAAE,IAAI;aACjB,CAAC,CAAC;QACL,CAAC,CAAC,CAAC;IACL,CAAC;IAED;;;;;;;;OAQG;IACO,MAAM,CAAC,QAAQ,CAA6B,GAAG,IAAe;QACtE,IAAI,CAAC,WAAW,CAAC,SAAS,EAAE,CAAC;YAC3B,MAAM,IAAI,GAAG,WAAW,CAAC,OAAO,CAAC,GAAG,IAAI,CAAM,CAAC;YAC/C,MAAM,OAAO,GAAG,IAAI,KAAK,CAAC,IAAW,EAAE;gBACrC,GAAG,CAAC,MAAM,EAAE,IAAI,EAAE,QAAQ;oBACxB,MAAM,KAAK,GAAG,OAAO,CAAC,GAAG,CAAC,MAAM,EAAE,IAAI,EAAE,QAAQ,CAAC,CAAC;oBAClD,IAAI,KAAK,KAAK,UAAU;wBAAE,OAAO,SAAS,CAAC;oBAC3C,IAAI,OAAO,KAAK,KAAK,WAAW;wBAAE,OAAO,KAAK,CAAC;oBAC/C,IAAI,OAAO,IAAI,KAAK,QAAQ,EAAE,CAAC;wBAC7B,+EAA+E;wBAC/E,IAAI,IAAI,KAAK,KAAK;4BAAE,OAAO,SAAS,CAAC;wBACrC,OAAO,WAAW,CAAC,aAAa,CAAC,SAAS,EAAE,CAAC,IAAI,CAAC,CAAC,CAAC;oBACtD,CAAC;oBACD,OAAO,KAAK,CAAC;gBACf,CAAC;aACF,CAAC,CAAC;YACH,WAAW,CAAC,SAAS,GAAG,OAAc,CAAC;QACzC,CAAC;QACD,OAAO,WAAW,CAAC,SAAc,CAAC;IACpC,CAAC;IAED;;;;;;;;OAQG;IACH,MAAM,CAAC,UAAU,CACf,KAAQ;QAER,MAAM,QAAQ,GAAG,WAAW,CAAC,QAAQ,EAA0B,CAAC;QAChE,MAAM,CAAC,IAAI,CAAC,QAAe,CAAC,CAAC,OAAO,CAAC,CAAC,GAAG,EAAE,EAAE;YAC3C,MAAM,IAAI,GAAG,MAAM,CAAC,wBAAwB,CAAC,QAAe,EAAE,GAAG,CAAC,CAAC;YACnE,IAAI,IAAI,IAAI,IAAI,CAAC,YAAY,IAAI,IAAI,CAAC,UAAU,EAAE,CAAC;gBACjD,MAAM,CAAC,cAAc,CAAC,QAAe,EAAE,GAAG,EAAE;oBAC1C,GAAG,IAAI;oBACP,UAAU,EAAE,KAAK;iBAClB,CAAC,CAAC;YACL,CAAC;QACH,CAAC,CAAC,CAAC;QACH,OAAO,QAAQ,CAAC,UAAU,CAAC,KAAK,CAE7B,CAAC;IACN,CAAC;IAED;;;;;OAKG;IACH,MAAM,CAAC,GAAG,CAAC,GAAW;QACpB,OAAO,WAAW,CAAC,SAAS,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC;IACxC,CAAC;IAED;;;;;;OAMG;IACK,MAAM,CAAC,aAAa,CAAC,OAAY,EAAE,IAAc;QACvD,MAAM,QAAQ,GAAG,CAAC,CAAW,EAAE,EAAE,CAC/B,CAAC,CAAC,GAAG,CAAC,CAAC,GAAG,EAAE,EAAE,CAAC,IAAA,kBAAW,EAAC,GAAG,CAAC,CAAC,CAAC,IAAI,CAAC,8BAAkB,CAAC,CAAC;QAE5D,kEAAkE;QAClE,MAAM,OAAO,GAAG,CAAC,GAAW,EAAW,EAAE;YACvC,IAAI,IAAA,eAAS,GAAE,EAAE,CAAC;gBAChB,MAAM,GAAG,GACP,UAGD,CAAC,yBAAa,CAAC,CAAC;gBACjB,OAAO,GAAG,CAAC,CAAC,CAAC,GAAG,CAAC,GAAG,CAAC,CAAC,CAAC,CAAC,SAAS,CAAC;YACpC,CAAC;YACD,OAAQ,UAAkB,EAAE,OAAO,EAAE,GAAG,EAAE,CAAC,GAAG,CAAC,CAAC;QAClD,CAAC,CAAC;QAEF,MAAM,OAAO,GAAsB;YACjC,GAAG,CAAC,OAAO,EAAE,IAAqB;gBAChC,IAAI,IAAI,KAAK,MAAM,CAAC,WAAW,EAAE,CAAC;oBAChC,OAAO,GAAG,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;gBAC9B,CAAC;gBACD,IAAI,IAAI,KAAK,UAAU,EAAE,CAAC;oBACxB,OAAO,GAAG,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;gBAC9B,CAAC;gBACD,IAAI,IAAI,KAAK,SAAS,EAAE,CAAC;oBACvB,OAAO,GAAG,EAAE,CAAC,QAAQ,CAAC,IAAI,CAAC,CAAC;gBAC9B,CAAC;gBACD,IAAI,OAAO,IAAI,KAAK,QAAQ;oBAAE,OAAO,SAAS,CAAC;gBAE/C,MAAM,SAAS,GACb,OAAO,IAAI,MAAM,CAAC,SAAS,CAAC,cAAc,CAAC,IAAI,CAAC,OAAO,EAAE,IAAI,CAAC;oBAC5D,CAAC,CAAE,OAAe,CAAC,IAAI,CAAC;oBACxB,CAAC,CAAC,SAAS,CAAC;gBAChB,MAAM,QAAQ,GAAG,CAAC,GAAG,IAAI,EAAE,IAAI,CAAC,CAAC;gBACjC,MAAM,WAAW,GAAG,QAAQ,CAAC,QAAQ,CAAC,CAAC;gBAEvC,2DAA2D;gBAC3D,MAAM,QAAQ,GAAG,OAAO,CAAC,WAAW,CAAC,CAAC;gBACtC,IAAI,OAAO,QAAQ,KAAK,WAAW;oBAAE,OAAO,QAAQ,CAAC;gBAErD,iFAAiF;gBACjF,MAAM,YAAY,GAAG,SAAS,IAAI,OAAO,SAAS,KAAK,QAAQ,CAAC;gBAChE,IAAI,YAAY;oBAAE,OAAO,WAAW,CAAC,aAAa,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;gBAExE,iFAAiF;gBACjF,IAAI,SAAS,KAAK,EAAE;oBAAE,OAAO,SAAS,CAAC;gBAEvC,wEAAwE;gBACxE,kFAAkF;gBAClF,OAAO,WAAW,CAAC,aAAa,CAAC,SAAS,EAAE,QAAQ,CAAC,CAAC;YACxD,CAAC;YACD,OAAO;gBACL,OAAO,OAAO,CAAC,CAAC,CAAC,OAAO,CAAC,OAAO,CAAC,OAAO,CAAC,CAAC,CAAC,CAAC,EAAE,CAAC;YACjD,CAAC;YACD,wBAAwB,CAAC,EAAE,EAAE,CAAC;gBAC5B,IAAI,CAAC,OAAO;oBAAE,OAAO,SAAgB,CAAC;gBACtC,IAAI,MAAM,CAAC,SAAS,CAAC,cAAc,CAAC,IAAI,CAAC,OAAO,EAAE,CAAC,CAAC,EAAE,CAAC;oBACrD,OAAO,EAAE,UAAU,EAAE,IAAI,EAAE,YAAY,EAAE,IAAI,EAAwB,CAAC;gBACxE,CAAC;gBACD,OAAO,SAAgB,CAAC;YAC1B,CAAC;SACF,CAAC;QAEF,MAAM,MAAM,GAAG,EAAS,CAAC;QACzB,OAAO,IAAI,KAAK,CAAC,MAAM,EAAE,OAAO,CAAC,CAAC;IACpC,CAAC;IAED;;;;;;OAMG;IACH,MAAM,CAAC,IAAI,CAAC,QAAiB,IAAI;QAC/B,OAAO,WAAW,CAAC,QAAQ,EAAE;aAC1B,IAAI,EAAE;aACN,GAAG,CAAC,CAAC,CAAC,EAAE,EAAE,CAAC,CAAC,KAAK,CAAC,CAAC,CAAC,IAAA,kBAAW,EAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC,CAAC;IAC9C,CAAC;;AApPH,kCAqPC;AAED;;;;;GAKG;AACU,QAAA,iBAAiB,GAAG,WAAW,CAAC,UAAU,CACrD,MAAM,CAAC,MAAM,CAAC,EAAE,EAAE,gCAAoB,EAAE;IACtC,GAAG,EACD,CAAC,IAAA,eAAS,GAAE,IAAK,UAAkB,CAAC,yBAAa,CAAC;QAChD,CAAC,CAAE,UAAkB,CAAC,yBAAa,CAAC,CAAC,UAAU,CAAC;QAChD,CAAC,CAAE,UAAkB,CAAC,OAAO,CAAC,GAAG,CAAC,UAAU,CAAC,CAAC,IAAI,aAAa;CACpE,CAAC,CACH,CAAC","sourcesContent":["import { ObjectAccumulator } from \"typed-object-accumulator\";\nimport { toENVFormat } from \"./text\";\nimport { isBrowser } from \"./web\";\nimport {\n  BrowserEnvKey,\n  DefaultLoggingConfig,\n  ENV_PATH_DELIMITER,\n} from \"./constants\";\n\n/**\n * @description Factory type for creating Environment instances.\n * @summary Describes factories that construct {@link Environment} derivatives with custom initialization.\n * @template T - The type of object the Environment will accumulate.\n * @template E - The specific Environment type to be created, extending Environment<T>.\n * @typedef {function(unknown[]): E} EnvironmentFactory\n * @memberOf module:Logging\n */\nexport type EnvironmentFactory<T extends object, E extends Environment<T>> = (\n  ...args: unknown[]\n) => E;\n\n/**\n * @description Environment accumulator that lazily reads from runtime sources.\n * @summary Extends {@link ObjectAccumulator} to merge configuration objects while resolving values from Node or browser environment variables on demand.\n * @template T\n * @class Environment\n * @example\n * const Config = Environment.accumulate({ logging: { level: \"info\" } });\n * console.log(Config.logging.level);\n * console.log(String(Config.logging.level)); // => LOGGING__LEVEL key when serialized\n * @mermaid\n * sequenceDiagram\n *   participant Client\n *   participant Env as Environment\n *   participant Process as process.env\n *   participant Browser as globalThis.ENV\n *   Client->>Env: accumulate(partialConfig)\n *   Env->>Env: expand(values)\n *   Client->>Env: Config.logging.level\n *   alt Browser runtime\n *     Env->>Browser: lookup ENV key\n *     Browser-->>Env: resolved value\n *   else Node runtime\n *     Env->>Process: lookup ENV key\n *     Process-->>Env: resolved value\n *   end\n *   Env-->>Client: merged value\n */\nconst EmptyValue = Symbol(\"EnvironmentEmpty\");\n\nexport class Environment<T extends object> extends ObjectAccumulator<T> {\n  /**\n   * @static\n   * @protected\n   * @description A factory function for creating Environment instances.\n   * @summary Defines how new instances of the Environment class should be created.\n   * @return {Environment<any>} A new instance of the Environment class.\n   */\n  protected static factory: EnvironmentFactory<any, any> =\n    (): Environment<any> => new Environment();\n\n  /**\n   * @static\n   * @private\n   * @description The singleton instance of the Environment class.\n   * @type {Environment<any>}\n   */\n  private static _instance: Environment<any>;\n\n  protected constructor() {\n    super();\n  }\n\n  /**\n   * @description Retrieves a value from the runtime environment.\n   * @summary Handles browser and Node.js environments by normalizing keys and parsing values.\n   * @param {string} k - Key to resolve from the environment.\n   * @return {unknown} Value resolved from the environment or `undefined` when absent.\n   */\n  protected fromEnv(k: string) {\n    let env: Record<string, unknown>;\n    if (isBrowser()) {\n      env =\n        (\n          globalThis as typeof globalThis & {\n            [BrowserEnvKey]: Record<string, any>;\n          }\n        )[BrowserEnvKey] || {};\n    } else {\n      env = globalThis.process.env;\n      k = toENVFormat(k);\n    }\n    return this.parseEnvValue(env[k]);\n  }\n\n  /**\n   * @description Converts stringified environment values into native types.\n   * @summary Interprets booleans and numbers while leaving other types unchanged.\n   * @param {unknown} val - Raw value retrieved from the environment.\n   * @return {unknown} Parsed value converted to boolean, number, or left as-is.\n   */\n  protected parseEnvValue(val: unknown) {\n    if (typeof val !== \"string\") return val;\n    if (val === \"true\") return true;\n    if (val === \"false\") return false;\n    const result = parseFloat(val);\n    if (!isNaN(result)) return result;\n    return val;\n  }\n\n  /**\n   * @description Expands an object into the environment.\n   * @summary Defines lazy properties that first consult runtime variables before falling back to seeded values.\n   * @template V - Type of the object being expanded.\n   * @param {V} value - Object to expose through environment getters and setters.\n   * @return {void}\n   */\n  protected override expand<V extends object>(value: V): void {\n    Object.entries(value).forEach(([k, v]) => {\n      Object.defineProperty(this, k, {\n        get: () => {\n          const fromEnv = this.fromEnv(k);\n          if (typeof fromEnv !== \"undefined\") return fromEnv;\n          if (v && typeof v === \"object\") {\n            return Environment.buildEnvProxy(v as any, [k]);\n          }\n          // If the model provides an empty string, mark with EmptyValue so instance proxy can return undefined without enabling key composition\n          if (v === \"\") {\n            return EmptyValue as unknown as V[keyof V];\n          }\n          return v;\n        },\n        set: (val: V[keyof V]) => {\n          v = val;\n        },\n        configurable: true,\n        enumerable: true,\n      });\n    });\n  }\n\n  /**\n   * @protected\n   * @static\n   * @description Retrieves or creates the singleton instance of the Environment class.\n   * @summary Ensures only one {@link Environment} instance is created, wrapping it in a proxy to compose ENV keys on demand.\n   * @template E\n   * @param {...unknown[]} args - Arguments forwarded to the factory when instantiating the singleton.\n   * @return {E} Singleton environment instance.\n   */\n  protected static instance<E extends Environment<any>>(...args: unknown[]): E {\n    if (!Environment._instance) {\n      const base = Environment.factory(...args) as E;\n      const proxied = new Proxy(base as any, {\n        get(target, prop, receiver) {\n          const value = Reflect.get(target, prop, receiver);\n          if (value === EmptyValue) return undefined;\n          if (typeof value !== \"undefined\") return value;\n          if (typeof prop === \"string\") {\n            // Avoid interfering with logging config lookups for optional fields like 'app'\n            if (prop === \"app\") return undefined;\n            return Environment.buildEnvProxy(undefined, [prop]);\n          }\n          return value;\n        },\n      });\n      Environment._instance = proxied as any;\n    }\n    return Environment._instance as E;\n  }\n\n  /**\n   * @static\n   * @description Accumulates the given value into the environment.\n   * @summary Adds new properties, hiding raw descriptors to avoid leaking enumeration semantics.\n   * @template T\n   * @template V\n   * @param {V} value - Object to merge into the environment.\n   * @return {Environment} Updated environment reference.\n   */\n  static accumulate<V extends object, TBase extends object = object>(\n    value: V\n  ): Environment<TBase & V> & TBase & V {\n    const instance = Environment.instance<Environment<TBase & V>>();\n    Object.keys(instance as any).forEach((key) => {\n      const desc = Object.getOwnPropertyDescriptor(instance as any, key);\n      if (desc && desc.configurable && desc.enumerable) {\n        Object.defineProperty(instance as any, key, {\n          ...desc,\n          enumerable: false,\n        });\n      }\n    });\n    return instance.accumulate(value) as unknown as Environment<TBase & V> &\n      TBase &\n      V;\n  }\n\n  /**\n   * @description Retrieves a value using a dot-path key from the accumulated environment.\n   * @summary Delegates to the singleton instance to access stored configuration.\n   * @param {string} key - Key to resolve from the environment store.\n   * @return {unknown} Stored value corresponding to the provided key.\n   */\n  static get(key: string) {\n    return Environment._instance.get(key);\n  }\n\n  /**\n   * @description Builds a proxy that composes environment keys for nested properties.\n   * @summary Allows chained property access to emit uppercase ENV identifiers while honoring existing runtime overrides.\n   * @param {any} current - Seed model segment used when projecting nested structures.\n   * @param {string[]} path - Accumulated path segments leading to the proxy.\n   * @return {any} Proxy that resolves environment values or composes additional proxies for deeper paths.\n   */\n  private static buildEnvProxy(current: any, path: string[]): any {\n    const buildKey = (p: string[]) =>\n      p.map((seg) => toENVFormat(seg)).join(ENV_PATH_DELIMITER);\n\n    // Helper to read from the active environment given a composed key\n    const readEnv = (key: string): unknown => {\n      if (isBrowser()) {\n        const env = (\n          globalThis as typeof globalThis & {\n            [BrowserEnvKey]?: Record<string, unknown>;\n          }\n        )[BrowserEnvKey];\n        return env ? env[key] : undefined;\n      }\n      return (globalThis as any)?.process?.env?.[key];\n    };\n\n    const handler: ProxyHandler<any> = {\n      get(_target, prop: string | symbol) {\n        if (prop === Symbol.toPrimitive) {\n          return () => buildKey(path);\n        }\n        if (prop === \"toString\") {\n          return () => buildKey(path);\n        }\n        if (prop === \"valueOf\") {\n          return () => buildKey(path);\n        }\n        if (typeof prop === \"symbol\") return undefined;\n\n        const nextModel =\n          current && Object.prototype.hasOwnProperty.call(current, prop)\n            ? (current as any)[prop]\n            : undefined;\n        const nextPath = [...path, prop];\n        const composedKey = buildKey(nextPath);\n\n        // If an ENV value exists for this path, return it directly\n        const envValue = readEnv(composedKey);\n        if (typeof envValue !== \"undefined\") return envValue;\n\n        // Otherwise, if the model has an object at this path, keep drilling with a proxy\n        const isNextObject = nextModel && typeof nextModel === \"object\";\n        if (isNextObject) return Environment.buildEnvProxy(nextModel, nextPath);\n\n        // If the model marks this leaf as an empty string, treat as undefined (no proxy)\n        if (nextModel === \"\") return undefined;\n\n        // Always return a proxy for further path composition when no ENV value;\n        // do not surface primitive model defaults here (this API is for key composition).\n        return Environment.buildEnvProxy(undefined, nextPath);\n      },\n      ownKeys() {\n        return current ? Reflect.ownKeys(current) : [];\n      },\n      getOwnPropertyDescriptor(_t, p) {\n        if (!current) return undefined as any;\n        if (Object.prototype.hasOwnProperty.call(current, p)) {\n          return { enumerable: true, configurable: true } as PropertyDescriptor;\n        }\n        return undefined as any;\n      },\n    };\n\n    const target = {} as any;\n    return new Proxy(target, handler);\n  }\n\n  /**\n   * @static\n   * @description Retrieves the keys of the environment, optionally converting them to ENV format.\n   * @summary Gets all keys in the environment, with an option to format them for environment variables.\n   * @param {boolean} [toEnv=true] - Whether to convert the keys to ENV format.\n   * @return {string[]} An array of keys from the environment.\n   */\n  static keys(toEnv: boolean = true): string[] {\n    return Environment.instance()\n      .keys()\n      .map((k) => (toEnv ? toENVFormat(k) : k));\n  }\n}\n\n/**\n * @description Singleton environment instance seeded with default logging configuration.\n * @summary Combines {@link DefaultLoggingConfig} with runtime environment variables to provide consistent logging defaults across platforms.\n * @const LoggedEnvironment\n * @memberOf module:Logging\n */\nexport const LoggedEnvironment = Environment.accumulate(\n  Object.assign({}, DefaultLoggingConfig, {\n    env:\n      (isBrowser() && (globalThis as any)[BrowserEnvKey]\n        ? (globalThis as any)[BrowserEnvKey][\"NODE_ENV\"]\n        : (globalThis as any).process.env[\"NODE_ENV\"]) || \"development\",\n  })\n);\n"]}
@@ -1,22 +1,13 @@
1
1
  import { ObjectAccumulator } from "typed-object-accumulator";
2
2
  /**
3
3
  * @description Factory type for creating Environment instances.
4
- * @summary Defines a function type that creates and returns Environment instances.
5
- *
4
+ * @summary Describes factories that construct {@link Environment} derivatives with custom initialization.
6
5
  * @template T - The type of object the Environment will accumulate.
7
6
  * @template E - The specific Environment type to be created, extending Environment<T>.
8
- * @typedef {function(...unknown[]): E} EnvironmentFactory
7
+ * @typedef {function(unknown[]): E} EnvironmentFactory
9
8
  * @memberOf module:Logging
10
9
  */
11
10
  export type EnvironmentFactory<T extends object, E extends Environment<T>> = (...args: unknown[]) => E;
12
- /**
13
- * @class Environment
14
- * @extends {ObjectAccumulator<T>}
15
- * @template T
16
- * @description A class representing an environment with accumulation capabilities.
17
- * @summary Manages environment-related data and provides methods for accumulation and key retrieval.
18
- * @param {T} [initialData] - The initial data to populate the environment with.
19
- */
20
11
  export declare class Environment<T extends object> extends ObjectAccumulator<T> {
21
12
  /**
22
13
  * @static
@@ -35,18 +26,24 @@ export declare class Environment<T extends object> extends ObjectAccumulator<T>
35
26
  private static _instance;
36
27
  protected constructor();
37
28
  /**
38
- * @description Retrieves a value from the environment
39
- * @summary Gets a value from the environment variables, handling browser and Node.js environments differently
40
- * @param {string} k - The key to retrieve from the environment
41
- * @return {unknown} The value from the environment, or undefined if not found
29
+ * @description Retrieves a value from the runtime environment.
30
+ * @summary Handles browser and Node.js environments by normalizing keys and parsing values.
31
+ * @param {string} k - Key to resolve from the environment.
32
+ * @return {unknown} Value resolved from the environment or `undefined` when absent.
42
33
  */
43
34
  protected fromEnv(k: string): unknown;
35
+ /**
36
+ * @description Converts stringified environment values into native types.
37
+ * @summary Interprets booleans and numbers while leaving other types unchanged.
38
+ * @param {unknown} val - Raw value retrieved from the environment.
39
+ * @return {unknown} Parsed value converted to boolean, number, or left as-is.
40
+ */
44
41
  protected parseEnvValue(val: unknown): unknown;
45
42
  /**
46
- * @description Expands an object into the environment
47
- * @summary Defines properties on the environment object that can be accessed as getters and setters
48
- * @template V - Type of the object being expanded
49
- * @param {V} value - The object to expand into the environment
43
+ * @description Expands an object into the environment.
44
+ * @summary Defines lazy properties that first consult runtime variables before falling back to seeded values.
45
+ * @template V - Type of the object being expanded.
46
+ * @param {V} value - Object to expose through environment getters and setters.
50
47
  * @return {void}
51
48
  */
52
49
  protected expand<V extends object>(value: V): void;
@@ -54,22 +51,36 @@ export declare class Environment<T extends object> extends ObjectAccumulator<T>
54
51
  * @protected
55
52
  * @static
56
53
  * @description Retrieves or creates the singleton instance of the Environment class.
57
- * @summary Ensures only one instance of the Environment class exists.
54
+ * @summary Ensures only one {@link Environment} instance is created, wrapping it in a proxy to compose ENV keys on demand.
58
55
  * @template E
59
- * @param {...unknown[]} args - Arguments to pass to the factory function if a new instance is created.
60
- * @return {E} The singleton instance of the Environment class.
56
+ * @param {...unknown[]} args - Arguments forwarded to the factory when instantiating the singleton.
57
+ * @return {E} Singleton environment instance.
61
58
  */
62
59
  protected static instance<E extends Environment<any>>(...args: unknown[]): E;
63
60
  /**
64
61
  * @static
65
62
  * @description Accumulates the given value into the environment.
66
- * @summary Adds new properties to the environment from the provided object.
63
+ * @summary Adds new properties, hiding raw descriptors to avoid leaking enumeration semantics.
64
+ * @template T
67
65
  * @template V
68
- * @param {V} value - The object to accumulate into the environment.
69
- * @return {V} The updated environment instance.
66
+ * @param {V} value - Object to merge into the environment.
67
+ * @return {Environment} Updated environment reference.
68
+ */
69
+ static accumulate<V extends object, TBase extends object = object>(value: V): Environment<TBase & V> & TBase & V;
70
+ /**
71
+ * @description Retrieves a value using a dot-path key from the accumulated environment.
72
+ * @summary Delegates to the singleton instance to access stored configuration.
73
+ * @param {string} key - Key to resolve from the environment store.
74
+ * @return {unknown} Stored value corresponding to the provided key.
70
75
  */
71
- static accumulate<V extends object>(value: V): typeof Environment._instance & V & ObjectAccumulator<typeof Environment._instance & V>;
72
76
  static get(key: string): any;
77
+ /**
78
+ * @description Builds a proxy that composes environment keys for nested properties.
79
+ * @summary Allows chained property access to emit uppercase ENV identifiers while honoring existing runtime overrides.
80
+ * @param {any} current - Seed model segment used when projecting nested structures.
81
+ * @param {string[]} path - Accumulated path segments leading to the proxy.
82
+ * @return {any} Proxy that resolves environment values or composes additional proxies for deeper paths.
83
+ */
73
84
  private static buildEnvProxy;
74
85
  /**
75
86
  * @static
@@ -80,8 +91,14 @@ export declare class Environment<T extends object> extends ObjectAccumulator<T>
80
91
  */
81
92
  static keys(toEnv?: boolean): string[];
82
93
  }
83
- export declare const LoggedEnvironment: Environment<any> & import("./types").LoggingConfig & {
94
+ /**
95
+ * @description Singleton environment instance seeded with default logging configuration.
96
+ * @summary Combines {@link DefaultLoggingConfig} with runtime environment variables to provide consistent logging defaults across platforms.
97
+ * @const LoggedEnvironment
98
+ * @memberOf module:Logging
99
+ */
100
+ export declare const LoggedEnvironment: Environment<object & import("./types").LoggingConfig & {
84
101
  env: any;
85
- } & ObjectAccumulator<Environment<any> & import("./types").LoggingConfig & {
102
+ }> & object & import("./types").LoggingConfig & {
86
103
  env: any;
87
- }>;
104
+ };
@@ -1,11 +1,7 @@
1
1
  import { Logger } from "./types";
2
2
  /**
3
- * @description Base class that provides a ready-to-use logger instance
4
- * @summary LoggedClass is a convenience abstract class that injects a type-safe logger
5
- * into derived classes through a protected getter. Subclasses can directly access
6
- * this.log to emit messages without manually creating a logger. This promotes
7
- * consistent, context-aware logging across the codebase.
8
- * @param {void} [constructor] - No constructor arguments; subclasses may define their own
3
+ * @description Base class that provides a ready-to-use logger instance.
4
+ * @summary Supplies inheriting classes with a lazily created, context-aware {@link Logger} via the protected `log` getter, promoting consistent structured logging without manual wiring.
9
5
  * @class LoggedClass
10
6
  * @example
11
7
  * class UserService extends LoggedClass {
@@ -32,11 +28,11 @@ import { Logger } from "./types";
32
28
  * Instance->>Logger: info/debug/error(...)
33
29
  */
34
30
  export declare abstract class LoggedClass {
31
+ private _log?;
35
32
  /**
36
- * @description Lazily provides a context-aware logger for the current instance
37
- * @summary Uses Logging.for(this) to create a logger whose context is the
38
- * subclass name, allowing uniform and structured logs from any inheriting class.
39
- * @return {Logger} A logger bound to the subclass context
33
+ * @description Lazily provides a context-aware logger for the current instance.
34
+ * @summary Calls {@link Logging.for} with the subclass instance to obtain a logger whose context matches the subclass name.
35
+ * @return {Logger} Logger bound to the subclass context.
40
36
  */
41
37
  protected get log(): Logger;
42
38
  protected constructor();
@@ -1,11 +1,7 @@
1
1
  import { Logging } from "./logging.js";
2
2
  /**
3
- * @description Base class that provides a ready-to-use logger instance
4
- * @summary LoggedClass is a convenience abstract class that injects a type-safe logger
5
- * into derived classes through a protected getter. Subclasses can directly access
6
- * this.log to emit messages without manually creating a logger. This promotes
7
- * consistent, context-aware logging across the codebase.
8
- * @param {void} [constructor] - No constructor arguments; subclasses may define their own
3
+ * @description Base class that provides a ready-to-use logger instance.
4
+ * @summary Supplies inheriting classes with a lazily created, context-aware {@link Logger} via the protected `log` getter, promoting consistent structured logging without manual wiring.
9
5
  * @class LoggedClass
10
6
  * @example
11
7
  * class UserService extends LoggedClass {
@@ -33,14 +29,15 @@ import { Logging } from "./logging.js";
33
29
  */
34
30
  export class LoggedClass {
35
31
  /**
36
- * @description Lazily provides a context-aware logger for the current instance
37
- * @summary Uses Logging.for(this) to create a logger whose context is the
38
- * subclass name, allowing uniform and structured logs from any inheriting class.
39
- * @return {Logger} A logger bound to the subclass context
32
+ * @description Lazily provides a context-aware logger for the current instance.
33
+ * @summary Calls {@link Logging.for} with the subclass instance to obtain a logger whose context matches the subclass name.
34
+ * @return {Logger} Logger bound to the subclass context.
40
35
  */
41
36
  get log() {
42
- return Logging.for(this);
37
+ if (!this._log)
38
+ this._log = Logging.for(this);
39
+ return this._log;
43
40
  }
44
41
  constructor() { }
45
42
  }
46
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiTG9nZ2VkQ2xhc3MuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvTG9nZ2VkQ2xhc3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLE9BQU8sRUFBRSxxQkFBa0I7QUFHcEM7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7R0ErQkc7QUFDSCxNQUFNLE9BQWdCLFdBQVc7SUFDL0I7Ozs7O09BS0c7SUFDSCxJQUFjLEdBQUc7UUFDZixPQUFPLE9BQU8sQ0FBQyxHQUFHLENBQUMsSUFBVyxDQUFDLENBQUM7SUFDbEMsQ0FBQztJQUVELGdCQUF5QixDQUFDO0NBQzNCIiwic291cmNlc0NvbnRlbnQiOlsiaW1wb3J0IHsgTG9nZ2luZyB9IGZyb20gXCIuL2xvZ2dpbmdcIjtcbmltcG9ydCB7IExvZ2dlciB9IGZyb20gXCIuL3R5cGVzXCI7XG5cbi8qKlxuICogQGRlc2NyaXB0aW9uIEJhc2UgY2xhc3MgdGhhdCBwcm92aWRlcyBhIHJlYWR5LXRvLXVzZSBsb2dnZXIgaW5zdGFuY2VcbiAqIEBzdW1tYXJ5IExvZ2dlZENsYXNzIGlzIGEgY29udmVuaWVuY2UgYWJzdHJhY3QgY2xhc3MgdGhhdCBpbmplY3RzIGEgdHlwZS1zYWZlIGxvZ2dlclxuICogaW50byBkZXJpdmVkIGNsYXNzZXMgdGhyb3VnaCBhIHByb3RlY3RlZCBnZXR0ZXIuIFN1YmNsYXNzZXMgY2FuIGRpcmVjdGx5IGFjY2Vzc1xuICogdGhpcy5sb2cgdG8gZW1pdCBtZXNzYWdlcyB3aXRob3V0IG1hbnVhbGx5IGNyZWF0aW5nIGEgbG9nZ2VyLiBUaGlzIHByb21vdGVzXG4gKiBjb25zaXN0ZW50LCBjb250ZXh0LWF3YXJlIGxvZ2dpbmcgYWNyb3NzIHRoZSBjb2RlYmFzZS5cbiAqIEBwYXJhbSB7dm9pZH0gW2NvbnN0cnVjdG9yXSAtIE5vIGNvbnN0cnVjdG9yIGFyZ3VtZW50czsgc3ViY2xhc3NlcyBtYXkgZGVmaW5lIHRoZWlyIG93blxuICogQGNsYXNzIExvZ2dlZENsYXNzXG4gKiBAZXhhbXBsZVxuICogY2xhc3MgVXNlclNlcnZpY2UgZXh0ZW5kcyBMb2dnZWRDbGFzcyB7XG4gKiAgIGNyZWF0ZSh1c2VyOiBVc2VyKSB7XG4gKiAgICAgdGhpcy5sb2cuaW5mbyhgQ3JlYXRpbmcgdXNlciAke3VzZXIuaWR9YCk7XG4gKiAgIH1cbiAqIH1cbiAqXG4gKiBjb25zdCBzdmMgPSBuZXcgVXNlclNlcnZpY2UoKTtcbiAqIHN2Yy5jcmVhdGUoeyBpZDogXCI0MlwiIH0pO1xuICogQG1lcm1haWRcbiAqIHNlcXVlbmNlRGlhZ3JhbVxuICogICBwYXJ0aWNpcGFudCBDbGllbnRcbiAqICAgcGFydGljaXBhbnQgSW5zdGFuY2UgYXMgU3ViY2xhc3MgSW5zdGFuY2VcbiAqICAgcGFydGljaXBhbnQgR2V0dGVyIGFzIExvZ2dlZENsYXNzLmxvZ1xuICogICBwYXJ0aWNpcGFudCBMb2dnaW5nIGFzIExvZ2dpbmdcbiAqICAgcGFydGljaXBhbnQgTG9nZ2VyIGFzIExvZ2dlclxuICpcbiAqICAgQ2xpZW50LT4+SW5zdGFuY2U6IGNhbGwgc29tZU1ldGhvZCgpXG4gKiAgIEluc3RhbmNlLT4+R2V0dGVyOiBhY2Nlc3MgdGhpcy5sb2dcbiAqICAgR2V0dGVyLT4+TG9nZ2luZzogTG9nZ2luZy5mb3IodGhpcylcbiAqICAgTG9nZ2luZy0tPj5HZXR0ZXI6IHJldHVybiBMb2dnZXJcbiAqICAgR2V0dGVyLS0+Pkluc3RhbmNlOiByZXR1cm4gTG9nZ2VyXG4gKiAgIEluc3RhbmNlLT4+TG9nZ2VyOiBpbmZvL2RlYnVnL2Vycm9yKC4uLilcbiAqL1xuZXhwb3J0IGFic3RyYWN0IGNsYXNzIExvZ2dlZENsYXNzIHtcbiAgLyoqXG4gICAqIEBkZXNjcmlwdGlvbiBMYXppbHkgcHJvdmlkZXMgYSBjb250ZXh0LWF3YXJlIGxvZ2dlciBmb3IgdGhlIGN1cnJlbnQgaW5zdGFuY2VcbiAgICogQHN1bW1hcnkgVXNlcyBMb2dnaW5nLmZvcih0aGlzKSB0byBjcmVhdGUgYSBsb2dnZXIgd2hvc2UgY29udGV4dCBpcyB0aGVcbiAgICogc3ViY2xhc3MgbmFtZSwgYWxsb3dpbmcgdW5pZm9ybSBhbmQgc3RydWN0dXJlZCBsb2dzIGZyb20gYW55IGluaGVyaXRpbmcgY2xhc3MuXG4gICAqIEByZXR1cm4ge0xvZ2dlcn0gQSBsb2dnZXIgYm91bmQgdG8gdGhlIHN1YmNsYXNzIGNvbnRleHRcbiAgICovXG4gIHByb3RlY3RlZCBnZXQgbG9nKCk6IExvZ2dlciB7XG4gICAgcmV0dXJuIExvZ2dpbmcuZm9yKHRoaXMgYXMgYW55KTtcbiAgfVxuXG4gIHByb3RlY3RlZCBjb25zdHJ1Y3RvcigpIHt9XG59XG4iXX0=
43
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoiTG9nZ2VkQ2xhc3MuanMiLCJzb3VyY2VSb290IjoiIiwic291cmNlcyI6WyIuLi8uLi9zcmMvTG9nZ2VkQ2xhc3MudHMiXSwibmFtZXMiOltdLCJtYXBwaW5ncyI6IkFBQUEsT0FBTyxFQUFFLE9BQU8sRUFBRSxxQkFBa0I7QUFHcEM7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7Ozs7OztHQTJCRztBQUNILE1BQU0sT0FBZ0IsV0FBVztJQUcvQjs7OztPQUlHO0lBQ0gsSUFBYyxHQUFHO1FBQ2YsSUFBSSxDQUFDLElBQUksQ0FBQyxJQUFJO1lBQUUsSUFBSSxDQUFDLElBQUksR0FBRyxPQUFPLENBQUMsR0FBRyxDQUFDLElBQVcsQ0FBQyxDQUFDO1FBQ3JELE9BQU8sSUFBSSxDQUFDLElBQUksQ0FBQztJQUNuQixDQUFDO0lBRUQsZ0JBQXlCLENBQUM7Q0FDM0IiLCJzb3VyY2VzQ29udGVudCI6WyJpbXBvcnQgeyBMb2dnaW5nIH0gZnJvbSBcIi4vbG9nZ2luZ1wiO1xuaW1wb3J0IHsgTG9nZ2VyIH0gZnJvbSBcIi4vdHlwZXNcIjtcblxuLyoqXG4gKiBAZGVzY3JpcHRpb24gQmFzZSBjbGFzcyB0aGF0IHByb3ZpZGVzIGEgcmVhZHktdG8tdXNlIGxvZ2dlciBpbnN0YW5jZS5cbiAqIEBzdW1tYXJ5IFN1cHBsaWVzIGluaGVyaXRpbmcgY2xhc3NlcyB3aXRoIGEgbGF6aWx5IGNyZWF0ZWQsIGNvbnRleHQtYXdhcmUge0BsaW5rIExvZ2dlcn0gdmlhIHRoZSBwcm90ZWN0ZWQgYGxvZ2AgZ2V0dGVyLCBwcm9tb3RpbmcgY29uc2lzdGVudCBzdHJ1Y3R1cmVkIGxvZ2dpbmcgd2l0aG91dCBtYW51YWwgd2lyaW5nLlxuICogQGNsYXNzIExvZ2dlZENsYXNzXG4gKiBAZXhhbXBsZVxuICogY2xhc3MgVXNlclNlcnZpY2UgZXh0ZW5kcyBMb2dnZWRDbGFzcyB7XG4gKiAgIGNyZWF0ZSh1c2VyOiBVc2VyKSB7XG4gKiAgICAgdGhpcy5sb2cuaW5mbyhgQ3JlYXRpbmcgdXNlciAke3VzZXIuaWR9YCk7XG4gKiAgIH1cbiAqIH1cbiAqXG4gKiBjb25zdCBzdmMgPSBuZXcgVXNlclNlcnZpY2UoKTtcbiAqIHN2Yy5jcmVhdGUoeyBpZDogXCI0MlwiIH0pO1xuICogQG1lcm1haWRcbiAqIHNlcXVlbmNlRGlhZ3JhbVxuICogICBwYXJ0aWNpcGFudCBDbGllbnRcbiAqICAgcGFydGljaXBhbnQgSW5zdGFuY2UgYXMgU3ViY2xhc3MgSW5zdGFuY2VcbiAqICAgcGFydGljaXBhbnQgR2V0dGVyIGFzIExvZ2dlZENsYXNzLmxvZ1xuICogICBwYXJ0aWNpcGFudCBMb2dnaW5nIGFzIExvZ2dpbmdcbiAqICAgcGFydGljaXBhbnQgTG9nZ2VyIGFzIExvZ2dlclxuICpcbiAqICAgQ2xpZW50LT4+SW5zdGFuY2U6IGNhbGwgc29tZU1ldGhvZCgpXG4gKiAgIEluc3RhbmNlLT4+R2V0dGVyOiBhY2Nlc3MgdGhpcy5sb2dcbiAqICAgR2V0dGVyLT4+TG9nZ2luZzogTG9nZ2luZy5mb3IodGhpcylcbiAqICAgTG9nZ2luZy0tPj5HZXR0ZXI6IHJldHVybiBMb2dnZXJcbiAqICAgR2V0dGVyLS0+Pkluc3RhbmNlOiByZXR1cm4gTG9nZ2VyXG4gKiAgIEluc3RhbmNlLT4+TG9nZ2VyOiBpbmZvL2RlYnVnL2Vycm9yKC4uLilcbiAqL1xuZXhwb3J0IGFic3RyYWN0IGNsYXNzIExvZ2dlZENsYXNzIHtcbiAgcHJpdmF0ZSBfbG9nPzogTG9nZ2VyO1xuXG4gIC8qKlxuICAgKiBAZGVzY3JpcHRpb24gTGF6aWx5IHByb3ZpZGVzIGEgY29udGV4dC1hd2FyZSBsb2dnZXIgZm9yIHRoZSBjdXJyZW50IGluc3RhbmNlLlxuICAgKiBAc3VtbWFyeSBDYWxscyB7QGxpbmsgTG9nZ2luZy5mb3J9IHdpdGggdGhlIHN1YmNsYXNzIGluc3RhbmNlIHRvIG9idGFpbiBhIGxvZ2dlciB3aG9zZSBjb250ZXh0IG1hdGNoZXMgdGhlIHN1YmNsYXNzIG5hbWUuXG4gICAqIEByZXR1cm4ge0xvZ2dlcn0gTG9nZ2VyIGJvdW5kIHRvIHRoZSBzdWJjbGFzcyBjb250ZXh0LlxuICAgKi9cbiAgcHJvdGVjdGVkIGdldCBsb2coKTogTG9nZ2VyIHtcbiAgICBpZiAoIXRoaXMuX2xvZykgdGhpcy5fbG9nID0gTG9nZ2luZy5mb3IodGhpcyBhcyBhbnkpO1xuICAgIHJldHVybiB0aGlzLl9sb2c7XG4gIH1cblxuICBwcm90ZWN0ZWQgY29uc3RydWN0b3IoKSB7fVxufVxuIl19