@abdokouta/react-config 1.0.0

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 (42) hide show
  1. package/.examples/01-basic-usage.ts +289 -0
  2. package/.examples/02-env-helper.ts +282 -0
  3. package/.examples/README.md +285 -0
  4. package/.prettierrc.js +1 -0
  5. package/README.md +261 -0
  6. package/__tests__/config.module.test.ts +244 -0
  7. package/__tests__/drivers/env.driver.test.ts +259 -0
  8. package/__tests__/services/config.service.test.ts +328 -0
  9. package/__tests__/setup.d.ts +11 -0
  10. package/__tests__/vitest.setup.ts +30 -0
  11. package/config/config.config.ts +62 -0
  12. package/dist/index.d.mts +474 -0
  13. package/dist/index.d.ts +474 -0
  14. package/dist/index.js +516 -0
  15. package/dist/index.js.map +1 -0
  16. package/dist/index.mjs +501 -0
  17. package/dist/index.mjs.map +1 -0
  18. package/eslint.config.js +13 -0
  19. package/package.json +77 -0
  20. package/src/config.module.ts +154 -0
  21. package/src/constants/index.ts +5 -0
  22. package/src/constants/tokens.constant.ts +38 -0
  23. package/src/drivers/env.driver.ts +194 -0
  24. package/src/drivers/file.driver.ts +81 -0
  25. package/src/drivers/index.ts +6 -0
  26. package/src/index.ts +92 -0
  27. package/src/interfaces/config-driver.interface.ts +30 -0
  28. package/src/interfaces/config-module-options.interface.ts +84 -0
  29. package/src/interfaces/config-service.interface.ts +71 -0
  30. package/src/interfaces/index.ts +8 -0
  31. package/src/interfaces/vite-config-plugin-options.interface.ts +56 -0
  32. package/src/plugins/index.ts +5 -0
  33. package/src/plugins/vite.plugin.ts +115 -0
  34. package/src/services/config.service.ts +172 -0
  35. package/src/services/index.ts +5 -0
  36. package/src/utils/get-nested-value.util.ts +56 -0
  37. package/src/utils/index.ts +6 -0
  38. package/src/utils/load-config-file.util.ts +37 -0
  39. package/src/utils/scan-config-files.util.ts +40 -0
  40. package/tsconfig.json +28 -0
  41. package/tsup.config.ts +105 -0
  42. package/vitest.config.ts +66 -0
package/dist/index.mjs ADDED
@@ -0,0 +1,501 @@
1
+ var __defProp = Object.defineProperty;
2
+ var __getOwnPropDesc = Object.getOwnPropertyDescriptor;
3
+ var __defNormalProp = (obj, key, value) => key in obj ? __defProp(obj, key, { enumerable: true, configurable: true, writable: true, value }) : obj[key] = value;
4
+ var __require = /* @__PURE__ */ ((x) => typeof require !== "undefined" ? require : typeof Proxy !== "undefined" ? new Proxy(x, {
5
+ get: (a, b) => (typeof require !== "undefined" ? require : a)[b]
6
+ }) : x)(function(x) {
7
+ if (typeof require !== "undefined") return require.apply(this, arguments);
8
+ throw Error('Dynamic require of "' + x + '" is not supported');
9
+ });
10
+ var __decorateClass = (decorators, target, key, kind) => {
11
+ var result = kind > 1 ? void 0 : kind ? __getOwnPropDesc(target, key) : target;
12
+ for (var i = decorators.length - 1, decorator; i >= 0; i--)
13
+ if (decorator = decorators[i])
14
+ result = (kind ? decorator(target, key, result) : decorator(result)) || result;
15
+ if (kind && result) __defProp(target, key, result);
16
+ return result;
17
+ };
18
+ var __decorateParam = (index, decorator) => (target, key) => decorator(target, key, index);
19
+ var __publicField = (obj, key, value) => __defNormalProp(obj, typeof key !== "symbol" ? key + "" : key, value);
20
+
21
+ // src/config.module.ts
22
+ import { Module } from "@abdokouta/react-di";
23
+
24
+ // src/utils/get-nested-value.util.ts
25
+ function getNestedValue(obj, path, defaultValue) {
26
+ const keys = path.split(".");
27
+ let current = obj;
28
+ for (const key of keys) {
29
+ if (current === null || current === void 0) {
30
+ return defaultValue;
31
+ }
32
+ current = current[key];
33
+ }
34
+ return current !== void 0 ? current : defaultValue;
35
+ }
36
+ function hasNestedValue(obj, path) {
37
+ const keys = path.split(".");
38
+ let current = obj;
39
+ for (const key of keys) {
40
+ if (current === null || current === void 0 || !(key in current)) {
41
+ return false;
42
+ }
43
+ current = current[key];
44
+ }
45
+ return true;
46
+ }
47
+
48
+ // src/drivers/env.driver.ts
49
+ var EnvDriver = class {
50
+ constructor(options = {}) {
51
+ this.options = options;
52
+ __publicField(this, "config", {});
53
+ __publicField(this, "loaded", false);
54
+ }
55
+ /**
56
+ * Load environment variables
57
+ */
58
+ load() {
59
+ if (this.loaded) {
60
+ console.log("[EnvDriver] Already loaded, returning cached config");
61
+ return this.config;
62
+ }
63
+ console.log("[EnvDriver] Loading environment variables...");
64
+ console.log("[EnvDriver] Options:", { ...this.options });
65
+ if (!this.options.ignoreEnvFile) {
66
+ this.loadDotEnv();
67
+ }
68
+ const globalName = this.options.globalName || "__APP_CONFIG__";
69
+ if (typeof window !== "undefined" && window[globalName]) {
70
+ this.config = { ...window[globalName] };
71
+ console.log(`[EnvDriver] Loaded config from window.${globalName}:`, Object.keys(this.config).length, "keys");
72
+ } else if (typeof process !== "undefined" && process.env) {
73
+ this.config = { ...process.env };
74
+ console.log("[EnvDriver] Loaded config from process.env:", Object.keys(this.config).length, "keys");
75
+ } else {
76
+ console.warn("[EnvDriver] No config source available (neither window." + globalName + " nor process.env)");
77
+ this.config = {};
78
+ }
79
+ console.log("[EnvDriver] Initial config keys:", [...Object.keys(this.config).filter((k) => k.includes("APP") || k.includes("VITE"))]);
80
+ if (this.options.envPrefix !== false) {
81
+ console.log("[EnvDriver] Stripping prefix...");
82
+ this.stripPrefix();
83
+ console.log("[EnvDriver] After stripPrefix, config keys:", [...Object.keys(this.config).filter((k) => k.includes("APP") || k.includes("VITE"))]);
84
+ }
85
+ if (this.options.expandVariables) {
86
+ this.expandEnvVariables();
87
+ }
88
+ this.loaded = true;
89
+ console.log("[EnvDriver] Load complete. Sample values:", {
90
+ APP_NAME: this.config.APP_NAME,
91
+ VITE_APP_NAME: this.config.VITE_APP_NAME
92
+ });
93
+ return this.config;
94
+ }
95
+ /**
96
+ * Get configuration value
97
+ */
98
+ get(key, defaultValue) {
99
+ if (!this.loaded) {
100
+ this.load();
101
+ }
102
+ const value = getNestedValue(this.config, key, defaultValue);
103
+ console.log(`[EnvDriver] get("${key}", "${defaultValue}") = "${value}"`);
104
+ return value;
105
+ }
106
+ /**
107
+ * Check if key exists
108
+ */
109
+ has(key) {
110
+ if (!this.loaded) {
111
+ this.load();
112
+ }
113
+ return hasNestedValue(this.config, key);
114
+ }
115
+ /**
116
+ * Get all configuration
117
+ */
118
+ all() {
119
+ if (!this.loaded) {
120
+ this.load();
121
+ }
122
+ return { ...this.config };
123
+ }
124
+ /**
125
+ * Load .env file using dotenv
126
+ */
127
+ loadDotEnv() {
128
+ try {
129
+ const dotenv = __require("dotenv");
130
+ const paths = Array.isArray(this.options.envFilePath) ? this.options.envFilePath : [this.options.envFilePath || ".env"];
131
+ for (const path of paths) {
132
+ dotenv.config({ path });
133
+ }
134
+ } catch (error) {
135
+ }
136
+ }
137
+ /**
138
+ * Expand environment variables (e.g., ${VAR_NAME})
139
+ */
140
+ expandEnvVariables() {
141
+ const regex = /\$\{([^}]+)\}/g;
142
+ const expand = (value) => {
143
+ return value.replace(regex, (_, key) => {
144
+ return this.config[key] || "";
145
+ });
146
+ };
147
+ for (const [key, value] of Object.entries(this.config)) {
148
+ if (typeof value === "string" && value.includes("${")) {
149
+ this.config[key] = expand(value);
150
+ }
151
+ }
152
+ }
153
+ /**
154
+ * Strip environment variable prefix
155
+ * Auto-detects framework (Vite, Next.js) or uses custom prefix
156
+ */
157
+ stripPrefix() {
158
+ let prefix = this.options.envPrefix;
159
+ if (prefix === "auto" || prefix === void 0) {
160
+ const hasViteVars = Object.keys(this.config).some((key) => key.startsWith("VITE_"));
161
+ if (hasViteVars || typeof import.meta !== "undefined") {
162
+ prefix = "VITE_";
163
+ } else if (Object.keys(this.config).some((key) => key.startsWith("NEXT_PUBLIC_"))) {
164
+ prefix = "NEXT_PUBLIC_";
165
+ } else {
166
+ return;
167
+ }
168
+ }
169
+ if (typeof prefix === "string" && prefix.length > 0) {
170
+ const newConfig = {};
171
+ for (const [key, value] of Object.entries(this.config)) {
172
+ if (key.startsWith(prefix)) {
173
+ const unprefixedKey = key.substring(prefix.length);
174
+ newConfig[unprefixedKey] = value;
175
+ newConfig[key] = value;
176
+ } else {
177
+ newConfig[key] = value;
178
+ }
179
+ }
180
+ this.config = newConfig;
181
+ }
182
+ }
183
+ };
184
+
185
+ // src/drivers/file.driver.ts
186
+ var FileDriver = class {
187
+ constructor(options = {}) {
188
+ __publicField(this, "config", {});
189
+ __publicField(this, "loaded", false);
190
+ if (options.config) {
191
+ this.config = options.config;
192
+ this.loaded = true;
193
+ }
194
+ }
195
+ /**
196
+ * Load configuration
197
+ * Config should be pre-loaded via Vite plugin or passed in constructor
198
+ */
199
+ async load() {
200
+ if (this.loaded) {
201
+ return this.config;
202
+ }
203
+ const isServer = typeof globalThis !== "undefined" && typeof globalThis.window === "undefined";
204
+ if (isServer) {
205
+ throw new Error(
206
+ "FileDriver requires pre-loaded configuration. Use Vite plugin for client-side or pass config in constructor for server-side."
207
+ );
208
+ }
209
+ this.loaded = true;
210
+ return this.config;
211
+ }
212
+ /**
213
+ * Get configuration value
214
+ */
215
+ get(key, defaultValue) {
216
+ if (!this.loaded) {
217
+ throw new Error("Configuration not loaded. Call load() first or use async initialization.");
218
+ }
219
+ return getNestedValue(this.config, key, defaultValue);
220
+ }
221
+ /**
222
+ * Check if key exists
223
+ */
224
+ has(key) {
225
+ if (!this.loaded) {
226
+ throw new Error("Configuration not loaded. Call load() first or use async initialization.");
227
+ }
228
+ return hasNestedValue(this.config, key);
229
+ }
230
+ /**
231
+ * Get all configuration
232
+ */
233
+ all() {
234
+ if (!this.loaded) {
235
+ throw new Error("Configuration not loaded. Call load() first or use async initialization.");
236
+ }
237
+ return { ...this.config };
238
+ }
239
+ };
240
+
241
+ // src/services/config.service.ts
242
+ import { Inject, Injectable } from "@abdokouta/react-di";
243
+
244
+ // src/constants/tokens.constant.ts
245
+ var CONFIG_DRIVER = /* @__PURE__ */ Symbol("CONFIG_DRIVER");
246
+ var CONFIG_OPTIONS = /* @__PURE__ */ Symbol("CONFIG_OPTIONS");
247
+ var CONFIG_SERVICE = /* @__PURE__ */ Symbol("CONFIG_SERVICE");
248
+
249
+ // src/services/config.service.ts
250
+ var ConfigService = class {
251
+ constructor(driver) {
252
+ this.driver = driver;
253
+ }
254
+ /**
255
+ * Get configuration value
256
+ */
257
+ get(key, defaultValue) {
258
+ return this.driver.get(key, defaultValue);
259
+ }
260
+ /**
261
+ * Get configuration value or throw if not found
262
+ */
263
+ getOrThrow(key) {
264
+ const value = this.get(key);
265
+ if (value === void 0) {
266
+ throw new Error(`Configuration key "${key}" is required but not set`);
267
+ }
268
+ return value;
269
+ }
270
+ /**
271
+ * Get string value
272
+ */
273
+ getString(key, defaultValue) {
274
+ const value = this.get(key, defaultValue);
275
+ return value !== void 0 ? String(value) : void 0;
276
+ }
277
+ /**
278
+ * Get string value or throw
279
+ */
280
+ getStringOrThrow(key) {
281
+ return String(this.getOrThrow(key));
282
+ }
283
+ /**
284
+ * Get number value
285
+ */
286
+ getNumber(key, defaultValue) {
287
+ const value = this.get(key, defaultValue);
288
+ if (value === void 0) {
289
+ return void 0;
290
+ }
291
+ const parsed = Number(value);
292
+ return isNaN(parsed) ? defaultValue : parsed;
293
+ }
294
+ /**
295
+ * Get number value or throw
296
+ */
297
+ getNumberOrThrow(key) {
298
+ const value = this.getNumber(key);
299
+ if (value === void 0) {
300
+ throw new Error(`Configuration key "${key}" is required but not set`);
301
+ }
302
+ return value;
303
+ }
304
+ /**
305
+ * Get boolean value
306
+ * Treats 'true', '1', 'yes', 'on' as true
307
+ */
308
+ getBool(key, defaultValue) {
309
+ const value = this.get(key, defaultValue);
310
+ if (value === void 0) {
311
+ return void 0;
312
+ }
313
+ if (typeof value === "boolean") {
314
+ return value;
315
+ }
316
+ return ["true", "1", "yes", "on"].includes(String(value).toLowerCase());
317
+ }
318
+ /**
319
+ * Get boolean value or throw
320
+ */
321
+ getBoolOrThrow(key) {
322
+ const value = this.getBool(key);
323
+ if (value === void 0) {
324
+ throw new Error(`Configuration key "${key}" is required but not set`);
325
+ }
326
+ return value;
327
+ }
328
+ /**
329
+ * Get array value (comma-separated string or actual array)
330
+ */
331
+ getArray(key, defaultValue) {
332
+ const value = this.get(key, defaultValue);
333
+ if (value === void 0) {
334
+ return void 0;
335
+ }
336
+ if (Array.isArray(value)) {
337
+ return value.map(String);
338
+ }
339
+ return String(value).split(",").map((v) => v.trim()).filter(Boolean);
340
+ }
341
+ /**
342
+ * Get JSON value
343
+ */
344
+ getJson(key, defaultValue) {
345
+ const value = this.get(key, defaultValue);
346
+ if (value === void 0) {
347
+ return void 0;
348
+ }
349
+ if (typeof value === "object") {
350
+ return value;
351
+ }
352
+ try {
353
+ return JSON.parse(String(value));
354
+ } catch {
355
+ return defaultValue;
356
+ }
357
+ }
358
+ /**
359
+ * Check if configuration key exists
360
+ */
361
+ has(key) {
362
+ return this.driver.has(key);
363
+ }
364
+ /**
365
+ * Get all configuration values
366
+ */
367
+ all() {
368
+ return this.driver.all();
369
+ }
370
+ /**
371
+ * Clear cache (no-op since we don't cache in ConfigService)
372
+ */
373
+ clearCache() {
374
+ }
375
+ };
376
+ ConfigService = __decorateClass([
377
+ Injectable(),
378
+ __decorateParam(0, Inject(CONFIG_DRIVER))
379
+ ], ConfigService);
380
+
381
+ // src/config.module.ts
382
+ var ConfigModule = class {
383
+ /**
384
+ * Register configuration module with options
385
+ *
386
+ * @param options - Configuration options
387
+ * @returns Dynamic module
388
+ */
389
+ static forRoot(options = {}) {
390
+ const driver = this.createDriver(options);
391
+ const isGlobal = options.isGlobal ?? false;
392
+ const providers = [
393
+ {
394
+ provide: CONFIG_OPTIONS,
395
+ useValue: options,
396
+ isGlobal
397
+ },
398
+ {
399
+ provide: CONFIG_DRIVER,
400
+ useValue: driver,
401
+ isGlobal
402
+ },
403
+ ConfigService,
404
+ {
405
+ provide: CONFIG_SERVICE,
406
+ useExisting: ConfigService,
407
+ isGlobal
408
+ }
409
+ ];
410
+ return {
411
+ module: ConfigModule,
412
+ providers,
413
+ exports: [ConfigService]
414
+ };
415
+ }
416
+ /**
417
+ * Register configuration module asynchronously
418
+ *
419
+ * @param options - Async configuration options
420
+ * @returns Dynamic module
421
+ */
422
+ static async forRootAsync(options) {
423
+ const resolvedOptions = options.useFactory ? await options.useFactory() : options;
424
+ return this.forRoot(resolvedOptions);
425
+ }
426
+ /**
427
+ * Create configuration driver based on options
428
+ */
429
+ static createDriver(options) {
430
+ const driverType = options.driver || "env";
431
+ switch (driverType) {
432
+ case "env":
433
+ const envDriver = new EnvDriver({
434
+ envFilePath: options.envFilePath,
435
+ ignoreEnvFile: options.ignoreEnvFile,
436
+ expandVariables: options.expandVariables,
437
+ envPrefix: options.envPrefix,
438
+ globalName: options.globalName
439
+ });
440
+ envDriver.load();
441
+ if (options.load) {
442
+ this.mergeCustomConfig(envDriver, options.load);
443
+ }
444
+ return envDriver;
445
+ case "file":
446
+ const fileDriver = new FileDriver({
447
+ config: typeof options.load === "object" ? options.load : void 0
448
+ });
449
+ return fileDriver;
450
+ default:
451
+ throw new Error(`Unknown configuration driver: ${driverType}`);
452
+ }
453
+ }
454
+ /**
455
+ * Merge custom configuration into driver
456
+ */
457
+ static mergeCustomConfig(driver, load) {
458
+ const customConfig = typeof load === "function" ? load() : load;
459
+ if (customConfig instanceof Promise) {
460
+ customConfig.then((config) => {
461
+ Object.assign(driver.all(), config);
462
+ });
463
+ } else {
464
+ Object.assign(driver.all(), customConfig);
465
+ }
466
+ }
467
+ };
468
+ ConfigModule = __decorateClass([
469
+ Module({})
470
+ ], ConfigModule);
471
+
472
+ // src/utils/load-config-file.util.ts
473
+ async function loadConfigFile(filePath) {
474
+ try {
475
+ const module = await import(
476
+ /* @vite-ignore */
477
+ filePath
478
+ );
479
+ const config = module.default || module;
480
+ if (typeof config === "function") {
481
+ return await config();
482
+ }
483
+ return config;
484
+ } catch (error) {
485
+ console.warn(
486
+ `[vite-plugin-config] Failed to load config file: ${filePath}`,
487
+ error
488
+ );
489
+ return {};
490
+ }
491
+ }
492
+ export {
493
+ ConfigModule,
494
+ ConfigService,
495
+ EnvDriver,
496
+ FileDriver,
497
+ getNestedValue,
498
+ hasNestedValue,
499
+ loadConfigFile
500
+ };
501
+ //# sourceMappingURL=index.mjs.map
@@ -0,0 +1 @@
1
+ {"version":3,"sources":["../src/config.module.ts","../src/utils/get-nested-value.util.ts","../src/drivers/env.driver.ts","../src/drivers/file.driver.ts","../src/services/config.service.ts","../src/constants/tokens.constant.ts","../src/utils/load-config-file.util.ts"],"sourcesContent":["import { Module, DynamicModule } from '@abdokouta/react-di';\n\nimport { EnvDriver } from './drivers/env.driver';\nimport { FileDriver } from './drivers/file.driver';\nimport { ConfigService } from './services/config.service';\nimport type { ConfigDriver } from './interfaces/config-driver.interface';\nimport type { ConfigModuleOptions } from './interfaces/config-module-options.interface';\nimport { CONFIG_DRIVER, CONFIG_OPTIONS, CONFIG_SERVICE } from './constants/tokens.constant';\n\n/**\n * Configuration Module\n * \n * Provides configuration management with multiple drivers.\n * Similar to NestJS ConfigModule.\n * \n * @example\n * ```typescript\n * // Using environment variables (default)\n * @Module({\n * imports: [\n * ConfigModule.forRoot({\n * envFilePath: '.env',\n * isGlobal: true,\n * }),\n * ],\n * })\n * export class AppModule {}\n * ```\n * \n * @example\n * ```typescript\n * // Using file-based configuration\n * @Module({\n * imports: [\n * ConfigModule.forRoot({\n * driver: 'file',\n * filePattern: 'config/**\\/*.config.ts',\n * isGlobal: true,\n * }),\n * ],\n * })\n * export class AppModule {}\n * ```\n */\n@Module({})\nexport class ConfigModule {\n /**\n * Register configuration module with options\n * \n * @param options - Configuration options\n * @returns Dynamic module\n */\n static forRoot(options: ConfigModuleOptions = {}): DynamicModule {\n const driver = this.createDriver(options);\n \n const isGlobal = options.isGlobal ?? false;\n \n const providers = [\n {\n provide: CONFIG_OPTIONS,\n useValue: options,\n isGlobal,\n },\n {\n provide: CONFIG_DRIVER,\n useValue: driver,\n isGlobal,\n },\n ConfigService,\n {\n provide: CONFIG_SERVICE,\n useExisting: ConfigService,\n isGlobal,\n },\n ];\n\n return {\n module: ConfigModule,\n providers: providers as any,\n exports: [ConfigService],\n };\n }\n\n /**\n * Register configuration module asynchronously\n * \n * @param options - Async configuration options\n * @returns Dynamic module\n */\n static async forRootAsync(\n options: ConfigModuleOptions & {\n useFactory?: () => Promise<ConfigModuleOptions> | ConfigModuleOptions;\n }\n ): Promise<DynamicModule> {\n const resolvedOptions = options.useFactory\n ? await options.useFactory()\n : options;\n\n return this.forRoot(resolvedOptions);\n }\n\n /**\n * Create configuration driver based on options\n */\n private static createDriver(options: ConfigModuleOptions): ConfigDriver {\n const driverType = options.driver || 'env';\n\n switch (driverType) {\n case 'env':\n const envDriver = new EnvDriver({\n envFilePath: options.envFilePath,\n ignoreEnvFile: options.ignoreEnvFile,\n expandVariables: options.expandVariables,\n envPrefix: options.envPrefix,\n globalName: options.globalName,\n });\n envDriver.load();\n \n // Merge custom load function if provided\n if (options.load) {\n this.mergeCustomConfig(envDriver, options.load);\n }\n \n return envDriver;\n\n case 'file':\n const fileDriver = new FileDriver({\n config: typeof options.load === 'object' ? options.load : undefined,\n });\n return fileDriver;\n\n default:\n throw new Error(`Unknown configuration driver: ${driverType}`);\n }\n }\n\n /**\n * Merge custom configuration into driver\n */\n private static mergeCustomConfig(\n driver: ConfigDriver,\n load: Record<string, any> | (() => Record<string, any> | Promise<Record<string, any>>)\n ): void {\n const customConfig = typeof load === 'function' ? load() : load;\n \n if (customConfig instanceof Promise) {\n customConfig.then(config => {\n Object.assign(driver.all(), config);\n });\n } else {\n Object.assign(driver.all(), customConfig);\n }\n }\n}\n","/**\n * Get nested value from object using dot notation\n * \n * @param obj - Source object\n * @param path - Dot-notated path (e.g., 'database.host')\n * @param defaultValue - Default value if path not found\n * @returns Value at path or default value\n * \n * @example\n * ```typescript\n * const config = { database: { host: 'localhost' } };\n * getNestedValue(config, 'database.host'); // 'localhost'\n * getNestedValue(config, 'database.port', 5432); // 5432\n * ```\n */\nexport function getNestedValue<T = any>(\n obj: Record<string, any>,\n path: string,\n defaultValue?: T\n): T | undefined {\n const keys = path.split('.');\n let current: any = obj;\n\n for (const key of keys) {\n if (current === null || current === undefined) {\n return defaultValue;\n }\n current = current[key];\n }\n\n return current !== undefined ? current : defaultValue;\n}\n\n/**\n * Check if nested path exists in object\n * \n * @param obj - Source object\n * @param path - Dot-notated path\n * @returns True if path exists\n */\nexport function hasNestedValue(\n obj: Record<string, any>,\n path: string\n): boolean {\n const keys = path.split('.');\n let current: any = obj;\n\n for (const key of keys) {\n if (current === null || current === undefined || !(key in current)) {\n return false;\n }\n current = current[key];\n }\n\n return true;\n}\n","import type { ConfigDriver } from '../interfaces/config-driver.interface';\nimport { getNestedValue, hasNestedValue } from '../utils/get-nested-value.util';\n\n/**\n * Environment Variable Configuration Driver\n * \n * Loads configuration from environment variables (process.env).\n * Supports .env files via dotenv.\n */\nexport class EnvDriver implements ConfigDriver {\n private config: Record<string, any> = {};\n private loaded = false;\n\n constructor(\n private options: {\n envFilePath?: string | string[];\n ignoreEnvFile?: boolean;\n expandVariables?: boolean;\n envPrefix?: string | false;\n globalName?: string; // Custom global variable name\n } = {}\n ) {}\n\n /**\n * Load environment variables\n */\n load(): Record<string, any> {\n if (this.loaded) {\n console.log('[EnvDriver] Already loaded, returning cached config');\n return this.config;\n }\n\n console.log('[EnvDriver] Loading environment variables...');\n console.log('[EnvDriver] Options:', { ...this.options });\n\n // Load .env file if not ignored\n if (!this.options.ignoreEnvFile) {\n this.loadDotEnv();\n }\n\n // Get global config name (default: __APP_CONFIG__)\n const globalName = this.options.globalName || '__APP_CONFIG__';\n\n // Try to load from custom global variable first (browser environment)\n if (typeof window !== 'undefined' && (window as any)[globalName]) {\n this.config = { ...(window as any)[globalName] };\n console.log(`[EnvDriver] Loaded config from window.${globalName}:`, Object.keys(this.config).length, 'keys');\n }\n // Fallback to process.env (Node.js environment or backward compatibility)\n else if (typeof process !== 'undefined' && process.env) {\n this.config = { ...process.env };\n console.log('[EnvDriver] Loaded config from process.env:', Object.keys(this.config).length, 'keys');\n }\n // No config source available\n else {\n console.warn('[EnvDriver] No config source available (neither window.' + globalName + ' nor process.env)');\n this.config = {};\n }\n\n console.log('[EnvDriver] Initial config keys:', [...Object.keys(this.config).filter(k => k.includes('APP') || k.includes('VITE'))]);\n\n // Strip prefix if configured\n if (this.options.envPrefix !== false) {\n console.log('[EnvDriver] Stripping prefix...');\n this.stripPrefix();\n console.log('[EnvDriver] After stripPrefix, config keys:', [...Object.keys(this.config).filter(k => k.includes('APP') || k.includes('VITE'))]);\n }\n\n // Expand variables if enabled\n if (this.options.expandVariables) {\n this.expandEnvVariables();\n }\n\n this.loaded = true;\n console.log('[EnvDriver] Load complete. Sample values:', {\n APP_NAME: this.config.APP_NAME,\n VITE_APP_NAME: this.config.VITE_APP_NAME,\n });\n return this.config;\n }\n\n /**\n * Get configuration value\n */\n get<T = any>(key: string, defaultValue?: T): T | undefined {\n if (!this.loaded) {\n this.load();\n }\n const value = getNestedValue(this.config, key, defaultValue);\n console.log(`[EnvDriver] get(\"${key}\", \"${defaultValue}\") = \"${value}\"`);\n return value;\n }\n\n /**\n * Check if key exists\n */\n has(key: string): boolean {\n if (!this.loaded) {\n this.load();\n }\n return hasNestedValue(this.config, key);\n }\n\n /**\n * Get all configuration\n */\n all(): Record<string, any> {\n if (!this.loaded) {\n this.load();\n }\n return { ...this.config };\n }\n\n /**\n * Load .env file using dotenv\n */\n private loadDotEnv(): void {\n try {\n // Try to load dotenv\n const dotenv = require('dotenv');\n const paths = Array.isArray(this.options.envFilePath)\n ? this.options.envFilePath\n : [this.options.envFilePath || '.env'];\n\n for (const path of paths) {\n dotenv.config({ path });\n }\n } catch (error) {\n // dotenv not installed, skip\n }\n }\n\n /**\n * Expand environment variables (e.g., ${VAR_NAME})\n */\n private expandEnvVariables(): void {\n const regex = /\\$\\{([^}]+)\\}/g;\n\n const expand = (value: string): string => {\n return value.replace(regex, (_, key) => {\n return this.config[key] || '';\n });\n };\n\n for (const [key, value] of Object.entries(this.config)) {\n if (typeof value === 'string' && value.includes('${')) {\n this.config[key] = expand(value);\n }\n }\n }\n\n /**\n * Strip environment variable prefix\n * Auto-detects framework (Vite, Next.js) or uses custom prefix\n */\n private stripPrefix(): void {\n let prefix = this.options.envPrefix;\n\n // Auto-detect framework prefix\n if (prefix === 'auto' || prefix === undefined) {\n // Check for Vite (import.meta.env exists or VITE_ variables present)\n const hasViteVars = Object.keys(this.config).some(key => key.startsWith('VITE_'));\n if (hasViteVars || typeof import.meta !== 'undefined') {\n prefix = 'VITE_';\n }\n // Check for Next.js (NEXT_PUBLIC_ variables present)\n else if (Object.keys(this.config).some(key => key.startsWith('NEXT_PUBLIC_'))) {\n prefix = 'NEXT_PUBLIC_';\n }\n // No framework detected, don't strip\n else {\n return;\n }\n }\n\n // Strip the prefix from all matching keys\n if (typeof prefix === 'string' && prefix.length > 0) {\n const newConfig: Record<string, any> = {};\n \n for (const [key, value] of Object.entries(this.config)) {\n if (key.startsWith(prefix)) {\n // Add both prefixed and unprefixed versions\n const unprefixedKey = key.substring(prefix.length);\n newConfig[unprefixedKey] = value;\n newConfig[key] = value; // Keep original too\n } else {\n newConfig[key] = value;\n }\n }\n \n this.config = newConfig;\n }\n }\n}\n","import type { ConfigDriver } from '@/interfaces/config-driver.interface';\nimport { getNestedValue, hasNestedValue } from '@/utils/get-nested-value.util';\n\n/**\n * File-based Configuration Driver\n * \n * Loads configuration from TypeScript/JavaScript files.\n * This is a server-side only driver. For client-side, use the Vite plugin.\n * \n * @see packages/pixielity/config/src/plugins/vite-config.plugin.ts\n */\nexport class FileDriver implements ConfigDriver {\n private config: Record<string, any> = {};\n private loaded = false;\n\n constructor(\n options: {\n config?: Record<string, any>;\n } = {}\n ) {\n // Pre-loaded config from Vite plugin or server\n if (options.config) {\n this.config = options.config;\n this.loaded = true;\n }\n }\n\n /**\n * Load configuration\n * Config should be pre-loaded via Vite plugin or passed in constructor\n */\n async load(): Promise<Record<string, any>> {\n if (this.loaded) {\n return this.config;\n }\n\n // If running on server (Node.js), throw error\n const isServer = typeof globalThis !== 'undefined' && \n typeof (globalThis as typeof globalThis & { window?: any }).window === 'undefined';\n \n if (isServer) {\n throw new Error(\n 'FileDriver requires pre-loaded configuration. ' +\n 'Use Vite plugin for client-side or pass config in constructor for server-side.'\n );\n }\n\n this.loaded = true;\n return this.config;\n }\n\n /**\n * Get configuration value\n */\n get<T = any>(key: string, defaultValue?: T): T | undefined {\n if (!this.loaded) {\n throw new Error('Configuration not loaded. Call load() first or use async initialization.');\n }\n return getNestedValue(this.config, key, defaultValue);\n }\n\n /**\n * Check if key exists\n */\n has(key: string): boolean {\n if (!this.loaded) {\n throw new Error('Configuration not loaded. Call load() first or use async initialization.');\n }\n return hasNestedValue(this.config, key);\n }\n\n /**\n * Get all configuration\n */\n all(): Record<string, any> {\n if (!this.loaded) {\n throw new Error('Configuration not loaded. Call load() first or use async initialization.');\n }\n return { ...this.config };\n }\n}\n","import { Inject, Injectable } from \"@abdokouta/react-di\";\n\nimport { CONFIG_DRIVER } from \"@/constants/tokens.constant\";\nimport type { ConfigDriver } from \"@/interfaces/config-driver.interface\";\nimport type { ConfigServiceInterface } from \"@/interfaces/config-service.interface\";\n\n/**\n * Configuration Service\n *\n * Provides type-safe access to configuration values with various getter methods.\n * Similar to NestJS ConfigService.\n *\n * @example\n * ```typescript\n * class MyService {\n * constructor(private config: ConfigService) {}\n *\n * getDbConfig() {\n * return {\n * host: this.config.getString('DB_HOST', 'localhost'),\n * port: this.config.getNumber('DB_PORT', 5432),\n * ssl: this.config.getBool('DB_SSL', false),\n * };\n * }\n * }\n * ```\n */\n@Injectable()\nexport class ConfigService implements ConfigServiceInterface {\n constructor(\n @Inject(CONFIG_DRIVER)\n private driver: ConfigDriver,\n ) {}\n\n /**\n * Get configuration value\n */\n get<T = any>(key: string, defaultValue?: T): T | undefined {\n return this.driver.get<T>(key, defaultValue);\n }\n\n /**\n * Get configuration value or throw if not found\n */\n getOrThrow<T = any>(key: string): T {\n const value = this.get<T>(key);\n if (value === undefined) {\n throw new Error(`Configuration key \"${key}\" is required but not set`);\n }\n return value;\n }\n\n /**\n * Get string value\n */\n getString(key: string, defaultValue?: string): string | undefined {\n const value = this.get(key, defaultValue);\n return value !== undefined ? String(value) : undefined;\n }\n\n /**\n * Get string value or throw\n */\n getStringOrThrow(key: string): string {\n return String(this.getOrThrow(key));\n }\n\n /**\n * Get number value\n */\n getNumber(key: string, defaultValue?: number): number | undefined {\n const value = this.get(key, defaultValue);\n if (value === undefined) {\n return undefined;\n }\n const parsed = Number(value);\n return isNaN(parsed) ? defaultValue : parsed;\n }\n\n /**\n * Get number value or throw\n */\n getNumberOrThrow(key: string): number {\n const value = this.getNumber(key);\n if (value === undefined) {\n throw new Error(`Configuration key \"${key}\" is required but not set`);\n }\n return value;\n }\n\n /**\n * Get boolean value\n * Treats 'true', '1', 'yes', 'on' as true\n */\n getBool(key: string, defaultValue?: boolean): boolean | undefined {\n const value = this.get(key, defaultValue);\n if (value === undefined) {\n return undefined;\n }\n if (typeof value === \"boolean\") {\n return value;\n }\n return [\"true\", \"1\", \"yes\", \"on\"].includes(String(value).toLowerCase());\n }\n\n /**\n * Get boolean value or throw\n */\n getBoolOrThrow(key: string): boolean {\n const value = this.getBool(key);\n if (value === undefined) {\n throw new Error(`Configuration key \"${key}\" is required but not set`);\n }\n return value;\n }\n\n /**\n * Get array value (comma-separated string or actual array)\n */\n getArray(key: string, defaultValue?: string[]): string[] | undefined {\n const value = this.get(key, defaultValue);\n if (value === undefined) {\n return undefined;\n }\n if (Array.isArray(value)) {\n return value.map(String);\n }\n return String(value)\n .split(\",\")\n .map((v) => v.trim())\n .filter(Boolean);\n }\n\n /**\n * Get JSON value\n */\n getJson<T = any>(key: string, defaultValue?: T): T | undefined {\n const value = this.get(key, defaultValue);\n if (value === undefined) {\n return undefined;\n }\n if (typeof value === \"object\") {\n return value as T;\n }\n try {\n return JSON.parse(String(value)) as T;\n } catch {\n return defaultValue;\n }\n }\n\n /**\n * Check if configuration key exists\n */\n has(key: string): boolean {\n return this.driver.has(key);\n }\n\n /**\n * Get all configuration values\n */\n all(): Record<string, any> {\n return this.driver.all();\n }\n\n /**\n * Clear cache (no-op since we don't cache in ConfigService)\n */\n clearCache(): void {\n // No-op - caching should be done at a higher level if needed\n }\n}\n","/**\n * Dependency Injection Tokens\n * \n * Used for injecting configuration dependencies.\n */\n\n/**\n * Configuration driver token\n * \n * @example\n * ```typescript\n * @Inject(CONFIG_DRIVER)\n * private driver: ConfigDriver\n * ```\n */\nexport const CONFIG_DRIVER = Symbol('CONFIG_DRIVER');\n\n/**\n * Configuration options token\n * \n * @example\n * ```typescript\n * @Inject(CONFIG_OPTIONS)\n * private options: ConfigModuleOptions\n * ```\n */\nexport const CONFIG_OPTIONS = Symbol('CONFIG_OPTIONS');\n\n/**\n * Configuration service token\n * \n * @example\n * ```typescript\n * @Inject(CONFIG_SERVICE)\n * private config: ConfigService\n * ```\n */\nexport const CONFIG_SERVICE = Symbol('CONFIG_SERVICE');\n","/**\n * Load Config File Utility\n *\n * Dynamically loads and parses a config file.\n */\n\n/**\n * Load and parse config file\n *\n * @param filePath - Absolute path to the config file\n * @returns Parsed config object\n */\nexport async function loadConfigFile(\n filePath: string\n): Promise<Record<string, any>> {\n try {\n // Dynamic import of the config file\n // @ts-ignore - Dynamic import path\n const module = await import(/* @vite-ignore */ filePath);\n\n // Extract config object (could be default export or named export)\n const config = module.default || module;\n\n // If it's a function, call it\n if (typeof config === 'function') {\n return await config();\n }\n\n return config;\n } catch (error) {\n console.warn(\n `[vite-plugin-config] Failed to load config file: ${filePath}`,\n error\n );\n return {};\n }\n}\n"],"mappings":";;;;;;;;;;;;;;;;;;;;;AAAA,SAAS,cAA6B;;;ACe/B,SAAS,eACd,KACA,MACA,cACe;AACf,QAAM,OAAO,KAAK,MAAM,GAAG;AAC3B,MAAI,UAAe;AAEnB,aAAW,OAAO,MAAM;AACtB,QAAI,YAAY,QAAQ,YAAY,QAAW;AAC7C,aAAO;AAAA,IACT;AACA,cAAU,QAAQ,GAAG;AAAA,EACvB;AAEA,SAAO,YAAY,SAAY,UAAU;AAC3C;AASO,SAAS,eACd,KACA,MACS;AACT,QAAM,OAAO,KAAK,MAAM,GAAG;AAC3B,MAAI,UAAe;AAEnB,aAAW,OAAO,MAAM;AACtB,QAAI,YAAY,QAAQ,YAAY,UAAa,EAAE,OAAO,UAAU;AAClE,aAAO;AAAA,IACT;AACA,cAAU,QAAQ,GAAG;AAAA,EACvB;AAEA,SAAO;AACT;;;AC9CO,IAAM,YAAN,MAAwC;AAAA,EAI7C,YACU,UAMJ,CAAC,GACL;AAPQ;AAJV,wBAAQ,UAA8B,CAAC;AACvC,wBAAQ,UAAS;AAAA,EAUd;AAAA;AAAA;AAAA;AAAA,EAKH,OAA4B;AAC1B,QAAI,KAAK,QAAQ;AACf,cAAQ,IAAI,qDAAqD;AACjE,aAAO,KAAK;AAAA,IACd;AAEA,YAAQ,IAAI,8CAA8C;AAC1D,YAAQ,IAAI,wBAAwB,EAAE,GAAG,KAAK,QAAQ,CAAC;AAGvD,QAAI,CAAC,KAAK,QAAQ,eAAe;AAC/B,WAAK,WAAW;AAAA,IAClB;AAGA,UAAM,aAAa,KAAK,QAAQ,cAAc;AAG9C,QAAI,OAAO,WAAW,eAAgB,OAAe,UAAU,GAAG;AAChE,WAAK,SAAS,EAAE,GAAI,OAAe,UAAU,EAAE;AAC/C,cAAQ,IAAI,yCAAyC,UAAU,KAAK,OAAO,KAAK,KAAK,MAAM,EAAE,QAAQ,MAAM;AAAA,IAC7G,WAES,OAAO,YAAY,eAAe,QAAQ,KAAK;AACtD,WAAK,SAAS,EAAE,GAAG,QAAQ,IAAI;AAC/B,cAAQ,IAAI,+CAA+C,OAAO,KAAK,KAAK,MAAM,EAAE,QAAQ,MAAM;AAAA,IACpG,OAEK;AACH,cAAQ,KAAK,4DAA4D,aAAa,mBAAmB;AACzG,WAAK,SAAS,CAAC;AAAA,IACjB;AAEA,YAAQ,IAAI,oCAAoC,CAAC,GAAG,OAAO,KAAK,KAAK,MAAM,EAAE,OAAO,OAAK,EAAE,SAAS,KAAK,KAAK,EAAE,SAAS,MAAM,CAAC,CAAC,CAAC;AAGlI,QAAI,KAAK,QAAQ,cAAc,OAAO;AACpC,cAAQ,IAAI,iCAAiC;AAC7C,WAAK,YAAY;AACjB,cAAQ,IAAI,+CAA+C,CAAC,GAAG,OAAO,KAAK,KAAK,MAAM,EAAE,OAAO,OAAK,EAAE,SAAS,KAAK,KAAK,EAAE,SAAS,MAAM,CAAC,CAAC,CAAC;AAAA,IAC/I;AAGA,QAAI,KAAK,QAAQ,iBAAiB;AAChC,WAAK,mBAAmB;AAAA,IAC1B;AAEA,SAAK,SAAS;AACd,YAAQ,IAAI,6CAA6C;AAAA,MACvD,UAAU,KAAK,OAAO;AAAA,MACtB,eAAe,KAAK,OAAO;AAAA,IAC7B,CAAC;AACD,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA,EAKA,IAAa,KAAa,cAAiC;AACzD,QAAI,CAAC,KAAK,QAAQ;AAChB,WAAK,KAAK;AAAA,IACZ;AACA,UAAM,QAAQ,eAAe,KAAK,QAAQ,KAAK,YAAY;AAC3D,YAAQ,IAAI,oBAAoB,GAAG,OAAO,YAAY,SAAS,KAAK,GAAG;AACvE,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,IAAI,KAAsB;AACxB,QAAI,CAAC,KAAK,QAAQ;AAChB,WAAK,KAAK;AAAA,IACZ;AACA,WAAO,eAAe,KAAK,QAAQ,GAAG;AAAA,EACxC;AAAA;AAAA;AAAA;AAAA,EAKA,MAA2B;AACzB,QAAI,CAAC,KAAK,QAAQ;AAChB,WAAK,KAAK;AAAA,IACZ;AACA,WAAO,EAAE,GAAG,KAAK,OAAO;AAAA,EAC1B;AAAA;AAAA;AAAA;AAAA,EAKQ,aAAmB;AACzB,QAAI;AAEF,YAAM,SAAS,UAAQ,QAAQ;AAC/B,YAAM,QAAQ,MAAM,QAAQ,KAAK,QAAQ,WAAW,IAChD,KAAK,QAAQ,cACb,CAAC,KAAK,QAAQ,eAAe,MAAM;AAEvC,iBAAW,QAAQ,OAAO;AACxB,eAAO,OAAO,EAAE,KAAK,CAAC;AAAA,MACxB;AAAA,IACF,SAAS,OAAO;AAAA,IAEhB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKQ,qBAA2B;AACjC,UAAM,QAAQ;AAEd,UAAM,SAAS,CAAC,UAA0B;AACxC,aAAO,MAAM,QAAQ,OAAO,CAAC,GAAG,QAAQ;AACtC,eAAO,KAAK,OAAO,GAAG,KAAK;AAAA,MAC7B,CAAC;AAAA,IACH;AAEA,eAAW,CAAC,KAAK,KAAK,KAAK,OAAO,QAAQ,KAAK,MAAM,GAAG;AACtD,UAAI,OAAO,UAAU,YAAY,MAAM,SAAS,IAAI,GAAG;AACrD,aAAK,OAAO,GAAG,IAAI,OAAO,KAAK;AAAA,MACjC;AAAA,IACF;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA,EAMQ,cAAoB;AAC1B,QAAI,SAAS,KAAK,QAAQ;AAG1B,QAAI,WAAW,UAAU,WAAW,QAAW;AAE7C,YAAM,cAAc,OAAO,KAAK,KAAK,MAAM,EAAE,KAAK,SAAO,IAAI,WAAW,OAAO,CAAC;AAChF,UAAI,eAAe,OAAO,gBAAgB,aAAa;AACrD,iBAAS;AAAA,MACX,WAES,OAAO,KAAK,KAAK,MAAM,EAAE,KAAK,SAAO,IAAI,WAAW,cAAc,CAAC,GAAG;AAC7E,iBAAS;AAAA,MACX,OAEK;AACH;AAAA,MACF;AAAA,IACF;AAGA,QAAI,OAAO,WAAW,YAAY,OAAO,SAAS,GAAG;AACnD,YAAM,YAAiC,CAAC;AAExC,iBAAW,CAAC,KAAK,KAAK,KAAK,OAAO,QAAQ,KAAK,MAAM,GAAG;AACtD,YAAI,IAAI,WAAW,MAAM,GAAG;AAE1B,gBAAM,gBAAgB,IAAI,UAAU,OAAO,MAAM;AACjD,oBAAU,aAAa,IAAI;AAC3B,oBAAU,GAAG,IAAI;AAAA,QACnB,OAAO;AACL,oBAAU,GAAG,IAAI;AAAA,QACnB;AAAA,MACF;AAEA,WAAK,SAAS;AAAA,IAChB;AAAA,EACF;AACF;;;ACtLO,IAAM,aAAN,MAAyC;AAAA,EAI9C,YACE,UAEI,CAAC,GACL;AAPF,wBAAQ,UAA8B,CAAC;AACvC,wBAAQ,UAAS;AAQf,QAAI,QAAQ,QAAQ;AAClB,WAAK,SAAS,QAAQ;AACtB,WAAK,SAAS;AAAA,IAChB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,MAAM,OAAqC;AACzC,QAAI,KAAK,QAAQ;AACf,aAAO,KAAK;AAAA,IACd;AAGA,UAAM,WAAW,OAAO,eAAe,eACtB,OAAQ,WAAoD,WAAW;AAExF,QAAI,UAAU;AACZ,YAAM,IAAI;AAAA,QACR;AAAA,MAEF;AAAA,IACF;AAEA,SAAK,SAAS;AACd,WAAO,KAAK;AAAA,EACd;AAAA;AAAA;AAAA;AAAA,EAKA,IAAa,KAAa,cAAiC;AACzD,QAAI,CAAC,KAAK,QAAQ;AAChB,YAAM,IAAI,MAAM,0EAA0E;AAAA,IAC5F;AACA,WAAO,eAAe,KAAK,QAAQ,KAAK,YAAY;AAAA,EACtD;AAAA;AAAA;AAAA;AAAA,EAKA,IAAI,KAAsB;AACxB,QAAI,CAAC,KAAK,QAAQ;AAChB,YAAM,IAAI,MAAM,0EAA0E;AAAA,IAC5F;AACA,WAAO,eAAe,KAAK,QAAQ,GAAG;AAAA,EACxC;AAAA;AAAA;AAAA;AAAA,EAKA,MAA2B;AACzB,QAAI,CAAC,KAAK,QAAQ;AAChB,YAAM,IAAI,MAAM,0EAA0E;AAAA,IAC5F;AACA,WAAO,EAAE,GAAG,KAAK,OAAO;AAAA,EAC1B;AACF;;;AChFA,SAAS,QAAQ,kBAAkB;;;ACe5B,IAAM,gBAAgB,uBAAO,eAAe;AAW5C,IAAM,iBAAiB,uBAAO,gBAAgB;AAW9C,IAAM,iBAAiB,uBAAO,gBAAgB;;;ADT9C,IAAM,gBAAN,MAAsD;AAAA,EAC3D,YAEU,QACR;AADQ;AAAA,EACP;AAAA;AAAA;AAAA;AAAA,EAKH,IAAa,KAAa,cAAiC;AACzD,WAAO,KAAK,OAAO,IAAO,KAAK,YAAY;AAAA,EAC7C;AAAA;AAAA;AAAA;AAAA,EAKA,WAAoB,KAAgB;AAClC,UAAM,QAAQ,KAAK,IAAO,GAAG;AAC7B,QAAI,UAAU,QAAW;AACvB,YAAM,IAAI,MAAM,sBAAsB,GAAG,2BAA2B;AAAA,IACtE;AACA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,UAAU,KAAa,cAA2C;AAChE,UAAM,QAAQ,KAAK,IAAI,KAAK,YAAY;AACxC,WAAO,UAAU,SAAY,OAAO,KAAK,IAAI;AAAA,EAC/C;AAAA;AAAA;AAAA;AAAA,EAKA,iBAAiB,KAAqB;AACpC,WAAO,OAAO,KAAK,WAAW,GAAG,CAAC;AAAA,EACpC;AAAA;AAAA;AAAA;AAAA,EAKA,UAAU,KAAa,cAA2C;AAChE,UAAM,QAAQ,KAAK,IAAI,KAAK,YAAY;AACxC,QAAI,UAAU,QAAW;AACvB,aAAO;AAAA,IACT;AACA,UAAM,SAAS,OAAO,KAAK;AAC3B,WAAO,MAAM,MAAM,IAAI,eAAe;AAAA,EACxC;AAAA;AAAA;AAAA;AAAA,EAKA,iBAAiB,KAAqB;AACpC,UAAM,QAAQ,KAAK,UAAU,GAAG;AAChC,QAAI,UAAU,QAAW;AACvB,YAAM,IAAI,MAAM,sBAAsB,GAAG,2BAA2B;AAAA,IACtE;AACA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA;AAAA,EAMA,QAAQ,KAAa,cAA6C;AAChE,UAAM,QAAQ,KAAK,IAAI,KAAK,YAAY;AACxC,QAAI,UAAU,QAAW;AACvB,aAAO;AAAA,IACT;AACA,QAAI,OAAO,UAAU,WAAW;AAC9B,aAAO;AAAA,IACT;AACA,WAAO,CAAC,QAAQ,KAAK,OAAO,IAAI,EAAE,SAAS,OAAO,KAAK,EAAE,YAAY,CAAC;AAAA,EACxE;AAAA;AAAA;AAAA;AAAA,EAKA,eAAe,KAAsB;AACnC,UAAM,QAAQ,KAAK,QAAQ,GAAG;AAC9B,QAAI,UAAU,QAAW;AACvB,YAAM,IAAI,MAAM,sBAAsB,GAAG,2BAA2B;AAAA,IACtE;AACA,WAAO;AAAA,EACT;AAAA;AAAA;AAAA;AAAA,EAKA,SAAS,KAAa,cAA+C;AACnE,UAAM,QAAQ,KAAK,IAAI,KAAK,YAAY;AACxC,QAAI,UAAU,QAAW;AACvB,aAAO;AAAA,IACT;AACA,QAAI,MAAM,QAAQ,KAAK,GAAG;AACxB,aAAO,MAAM,IAAI,MAAM;AAAA,IACzB;AACA,WAAO,OAAO,KAAK,EAChB,MAAM,GAAG,EACT,IAAI,CAAC,MAAM,EAAE,KAAK,CAAC,EACnB,OAAO,OAAO;AAAA,EACnB;AAAA;AAAA;AAAA;AAAA,EAKA,QAAiB,KAAa,cAAiC;AAC7D,UAAM,QAAQ,KAAK,IAAI,KAAK,YAAY;AACxC,QAAI,UAAU,QAAW;AACvB,aAAO;AAAA,IACT;AACA,QAAI,OAAO,UAAU,UAAU;AAC7B,aAAO;AAAA,IACT;AACA,QAAI;AACF,aAAO,KAAK,MAAM,OAAO,KAAK,CAAC;AAAA,IACjC,QAAQ;AACN,aAAO;AAAA,IACT;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,IAAI,KAAsB;AACxB,WAAO,KAAK,OAAO,IAAI,GAAG;AAAA,EAC5B;AAAA;AAAA;AAAA;AAAA,EAKA,MAA2B;AACzB,WAAO,KAAK,OAAO,IAAI;AAAA,EACzB;AAAA;AAAA;AAAA;AAAA,EAKA,aAAmB;AAAA,EAEnB;AACF;AA/Ia,gBAAN;AAAA,EADN,WAAW;AAAA,EAGP,0BAAO,aAAa;AAAA,GAFZ;;;AJiBN,IAAM,eAAN,MAAmB;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAOxB,OAAO,QAAQ,UAA+B,CAAC,GAAkB;AAC/D,UAAM,SAAS,KAAK,aAAa,OAAO;AAExC,UAAM,WAAW,QAAQ,YAAY;AAErC,UAAM,YAAY;AAAA,MAChB;AAAA,QACE,SAAS;AAAA,QACT,UAAU;AAAA,QACV;AAAA,MACF;AAAA,MACA;AAAA,QACE,SAAS;AAAA,QACT,UAAU;AAAA,QACV;AAAA,MACF;AAAA,MACA;AAAA,MACA;AAAA,QACE,SAAS;AAAA,QACT,aAAa;AAAA,QACb;AAAA,MACF;AAAA,IACF;AAEA,WAAO;AAAA,MACL,QAAQ;AAAA,MACR;AAAA,MACA,SAAS,CAAC,aAAa;AAAA,IACzB;AAAA,EACF;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA;AAAA,EAQA,aAAa,aACX,SAGwB;AACxB,UAAM,kBAAkB,QAAQ,aAC5B,MAAM,QAAQ,WAAW,IACzB;AAEJ,WAAO,KAAK,QAAQ,eAAe;AAAA,EACrC;AAAA;AAAA;AAAA;AAAA,EAKA,OAAe,aAAa,SAA4C;AACtE,UAAM,aAAa,QAAQ,UAAU;AAErC,YAAQ,YAAY;AAAA,MAClB,KAAK;AACH,cAAM,YAAY,IAAI,UAAU;AAAA,UAC9B,aAAa,QAAQ;AAAA,UACrB,eAAe,QAAQ;AAAA,UACvB,iBAAiB,QAAQ;AAAA,UACzB,WAAW,QAAQ;AAAA,UACnB,YAAY,QAAQ;AAAA,QACtB,CAAC;AACD,kBAAU,KAAK;AAGf,YAAI,QAAQ,MAAM;AAChB,eAAK,kBAAkB,WAAW,QAAQ,IAAI;AAAA,QAChD;AAEA,eAAO;AAAA,MAET,KAAK;AACH,cAAM,aAAa,IAAI,WAAW;AAAA,UAChC,QAAQ,OAAO,QAAQ,SAAS,WAAW,QAAQ,OAAO;AAAA,QAC5D,CAAC;AACD,eAAO;AAAA,MAET;AACE,cAAM,IAAI,MAAM,iCAAiC,UAAU,EAAE;AAAA,IACjE;AAAA,EACF;AAAA;AAAA;AAAA;AAAA,EAKA,OAAe,kBACb,QACA,MACM;AACN,UAAM,eAAe,OAAO,SAAS,aAAa,KAAK,IAAI;AAE3D,QAAI,wBAAwB,SAAS;AACnC,mBAAa,KAAK,YAAU;AAC1B,eAAO,OAAO,OAAO,IAAI,GAAG,MAAM;AAAA,MACpC,CAAC;AAAA,IACH,OAAO;AACL,aAAO,OAAO,OAAO,IAAI,GAAG,YAAY;AAAA,IAC1C;AAAA,EACF;AACF;AA5Ga,eAAN;AAAA,EADN,OAAO,CAAC,CAAC;AAAA,GACG;;;AMjCb,eAAsB,eACpB,UAC8B;AAC9B,MAAI;AAGF,UAAM,SAAS,MAAM;AAAA;AAAA,MAA0B;AAAA;AAG/C,UAAM,SAAS,OAAO,WAAW;AAGjC,QAAI,OAAO,WAAW,YAAY;AAChC,aAAO,MAAM,OAAO;AAAA,IACtB;AAEA,WAAO;AAAA,EACT,SAAS,OAAO;AACd,YAAQ;AAAA,MACN,oDAAoD,QAAQ;AAAA,MAC5D;AAAA,IACF;AACA,WAAO,CAAC;AAAA,EACV;AACF;","names":[]}
@@ -0,0 +1,13 @@
1
+ import nesvelConfig from '@nesvel/eslint-config';
2
+
3
+ export default [
4
+ ...nesvelConfig,
5
+ {
6
+ ignores: ['dist/**', 'node_modules/**', '*.config.js', '*.config.ts'],
7
+ },
8
+ {
9
+ rules: {
10
+ // Add any package-specific rule overrides here
11
+ },
12
+ },
13
+ ];
package/package.json ADDED
@@ -0,0 +1,77 @@
1
+ {
2
+ "name": "@abdokouta/react-config",
3
+ "version": "1.0.0",
4
+ "description": "NestJS-inspired configuration management with multiple drivers (Env, File, Firebase) for React",
5
+ "keywords": [
6
+ "react",
7
+ "config",
8
+ "configuration",
9
+ "environment",
10
+ "nestjs",
11
+ "env",
12
+ "dotenv"
13
+ ],
14
+ "repository": {
15
+ "type": "git",
16
+ "url": "https://github.com/pixielity-inc/react-config.git"
17
+ },
18
+ "license": "MIT",
19
+ "author": "Refine",
20
+ "main": "dist/index.js",
21
+ "module": "dist/index.mjs",
22
+ "types": "dist/index.d.ts",
23
+ "exports": {
24
+ ".": {
25
+ "types": "./dist/index.d.ts",
26
+ "import": "./dist/index.mjs",
27
+ "require": "./dist/index.js"
28
+ },
29
+ "./vite-plugin": {
30
+ "types": "./dist/vite-plugin.d.ts",
31
+ "import": "./dist/vite-plugin.mjs",
32
+ "require": "./dist/vite-plugin.js"
33
+ }
34
+ },
35
+ "scripts": {
36
+ "build": "tsup",
37
+ "dev": "tsup --watch",
38
+ "test": "vitest --run",
39
+ "test:watch": "vitest",
40
+ "test:coverage": "vitest --run --coverage",
41
+ "test:ui": "vitest --ui",
42
+ "format": "prettier --write \"src/**/*.{ts,tsx,js,jsx,json}\""
43
+ },
44
+ "dependencies": {
45
+ "@abdokouta/react-support": "^1.1.0",
46
+ "@abdokouta/react-di": "^2.1.0"
47
+ },
48
+ "peerDependencies": {
49
+ "dotenv": "^16.0.0",
50
+ "vite": "^5.0.0"
51
+ },
52
+ "peerDependenciesMeta": {
53
+ "dotenv": {
54
+ "optional": true
55
+ },
56
+ "vite": {
57
+ "optional": true
58
+ }
59
+ },
60
+ "devDependencies": {
61
+ "@nesvel/eslint-config": "^1.0.5",
62
+ "@nesvel/prettier-config": "^1.0.3",
63
+ "@nesvel/tsup-config": "^1.0.3",
64
+ "@nesvel/typescript-config": "^1.0.4",
65
+ "@types/node": "^25.5.0",
66
+ "@vitest/ui": "^4.1.2",
67
+ "dotenv": "^17.3.1",
68
+ "prettier": "^3.8.1",
69
+ "tsup": "^8.5.1",
70
+ "typescript": "^6.0.2",
71
+ "vite": "^8.0.3",
72
+ "vitest": "^4.1.2"
73
+ },
74
+ "publishConfig": {
75
+ "access": "public"
76
+ }
77
+ }