@dollhousemcp/mcp-server 2.0.11 → 2.0.12-rc.2

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 (93) hide show
  1. package/CHANGELOG.md +33 -0
  2. package/dist/config/ConfigManager.d.ts +34 -0
  3. package/dist/config/ConfigManager.d.ts.map +1 -1
  4. package/dist/config/ConfigManager.js +46 -1
  5. package/dist/config/env.d.ts +18 -0
  6. package/dist/config/env.d.ts.map +1 -1
  7. package/dist/config/env.js +89 -2
  8. package/dist/di/Container.d.ts +1 -0
  9. package/dist/di/Container.d.ts.map +1 -1
  10. package/dist/di/Container.js +38 -2
  11. package/dist/elements/agents/AgentManager.js +2 -2
  12. package/dist/generated/version.d.ts +2 -2
  13. package/dist/generated/version.d.ts.map +1 -1
  14. package/dist/generated/version.js +3 -3
  15. package/dist/handlers/mcp-aql/GatekeeperSession.d.ts.map +1 -1
  16. package/dist/handlers/mcp-aql/GatekeeperSession.js +3 -2
  17. package/dist/handlers/mcp-aql/MCPAQLHandler.d.ts.map +1 -1
  18. package/dist/handlers/mcp-aql/MCPAQLHandler.js +11 -3
  19. package/dist/handlers/mcp-aql/OperationSchema.d.ts.map +1 -1
  20. package/dist/handlers/mcp-aql/OperationSchema.js +11 -6
  21. package/dist/handlers/mcp-aql/policies/ToolClassification.d.ts.map +1 -1
  22. package/dist/handlers/mcp-aql/policies/ToolClassification.js +8 -1
  23. package/dist/index.js +53 -3
  24. package/dist/security/InputNormalizer.d.ts +4 -6
  25. package/dist/security/InputNormalizer.d.ts.map +1 -1
  26. package/dist/security/InputNormalizer.js +10 -16
  27. package/dist/security/audit/config/suppressions.d.ts.map +1 -1
  28. package/dist/security/audit/config/suppressions.js +26 -1
  29. package/dist/security/constants.d.ts +7 -0
  30. package/dist/security/constants.d.ts.map +1 -1
  31. package/dist/security/constants.js +10 -1
  32. package/dist/security/contentValidator.d.ts.map +1 -1
  33. package/dist/security/contentValidator.js +14 -6
  34. package/dist/security/encryption/PatternEncryptor.d.ts.map +1 -1
  35. package/dist/security/encryption/PatternEncryptor.js +18 -10
  36. package/dist/security/securityMonitor.d.ts +1 -1
  37. package/dist/security/securityMonitor.d.ts.map +1 -1
  38. package/dist/security/securityMonitor.js +1 -1
  39. package/dist/security/tokenManager.d.ts +18 -2
  40. package/dist/security/tokenManager.d.ts.map +1 -1
  41. package/dist/security/tokenManager.js +47 -15
  42. package/dist/security/validators/unicodeValidator.d.ts +4 -5
  43. package/dist/security/validators/unicodeValidator.d.ts.map +1 -1
  44. package/dist/security/validators/unicodeValidator.js +19 -20
  45. package/dist/services/VerificationNotifier.d.ts.map +1 -1
  46. package/dist/services/VerificationNotifier.js +15 -13
  47. package/dist/services/validation/GenericElementValidator.js +5 -5
  48. package/dist/web/console/IngestRoutes.d.ts +14 -2
  49. package/dist/web/console/IngestRoutes.d.ts.map +1 -1
  50. package/dist/web/console/IngestRoutes.js +69 -5
  51. package/dist/web/console/LeaderElection.d.ts +37 -3
  52. package/dist/web/console/LeaderElection.d.ts.map +1 -1
  53. package/dist/web/console/LeaderElection.js +67 -7
  54. package/dist/web/console/LeaderForwardingSink.d.ts +19 -3
  55. package/dist/web/console/LeaderForwardingSink.d.ts.map +1 -1
  56. package/dist/web/console/LeaderForwardingSink.js +44 -9
  57. package/dist/web/console/PromotionManager.d.ts +29 -0
  58. package/dist/web/console/PromotionManager.d.ts.map +1 -0
  59. package/dist/web/console/PromotionManager.js +94 -0
  60. package/dist/web/console/SessionNames.d.ts +7 -0
  61. package/dist/web/console/SessionNames.d.ts.map +1 -1
  62. package/dist/web/console/SessionNames.js +10 -1
  63. package/dist/web/console/UnifiedConsole.d.ts +26 -1
  64. package/dist/web/console/UnifiedConsole.d.ts.map +1 -1
  65. package/dist/web/console/UnifiedConsole.js +143 -20
  66. package/dist/web/portDiscovery.d.ts +4 -3
  67. package/dist/web/portDiscovery.d.ts.map +1 -1
  68. package/dist/web/portDiscovery.js +7 -5
  69. package/dist/web/public/app.js +35 -18
  70. package/dist/web/public/consoleAuth.js +152 -0
  71. package/dist/web/public/index.html +243 -9
  72. package/dist/web/public/logs.js +1 -1
  73. package/dist/web/public/metrics.js +3 -3
  74. package/dist/web/public/permissions.js +1 -1
  75. package/dist/web/public/security.css +396 -0
  76. package/dist/web/public/security.js +393 -0
  77. package/dist/web/public/sessions.css +54 -4
  78. package/dist/web/public/sessions.js +46 -6
  79. package/dist/web/public/setup.css +439 -0
  80. package/dist/web/public/setup.js +490 -18
  81. package/dist/web/public/styles.css +10 -0
  82. package/dist/web/routes/permissionRoutes.d.ts.map +1 -1
  83. package/dist/web/routes/permissionRoutes.js +8 -3
  84. package/dist/web/routes/setupRoutes.d.ts +4 -0
  85. package/dist/web/routes/setupRoutes.d.ts.map +1 -1
  86. package/dist/web/routes/setupRoutes.js +418 -7
  87. package/dist/web/routes.d.ts.map +1 -1
  88. package/dist/web/routes.js +9 -5
  89. package/dist/web/server.d.ts +49 -3
  90. package/dist/web/server.d.ts.map +1 -1
  91. package/dist/web/server.js +282 -82
  92. package/package.json +11 -2
  93. package/server.json +2 -2
package/CHANGELOG.md CHANGED
@@ -1,5 +1,38 @@
1
1
  # Changelog
2
2
 
3
+ ## [2.0.11-rc.1] - 2026-04-08
4
+
5
+ Release candidate for v2.0.11 — console auth, permissions, licensing, port 41715, channel selector
6
+
7
+ ## [Unreleased] — Phase 2 authenticated console
8
+
9
+ ### Breaking: Web console default port moved from 3939 → 41715
10
+
11
+ The authenticated web console (Phase 2 and later) binds to port **41715** by default, not 3939. The associated state files now live at:
12
+
13
+ - `~/.dollhouse/run/console-leader.auth.lock` (was `console-leader.lock`)
14
+ - `~/.dollhouse/run/console-token.auth.json` (was `console-token.json`)
15
+
16
+ **Why:** Pre-authentication DollhouseMCP installations (≤ 2.0.x) continue to use port 3939 and the legacy file paths. The port + filename separation lets a legacy installation and an authenticated installation coexist on the same machine with zero interference — different ports, different lock files, different token files, independent leader-election spaces. This avoids a confusing "security popping on and off" UX when a user has both a pre-auth Claude Desktop install and an authenticated Claude Code install running simultaneously.
17
+
18
+ **Configurability:** The port, lock file path, and token file path are now all driven by env vars with a single source of truth in `src/config/env.ts`. Changing any of them is a one-line edit — no hunt-and-peck across the codebase:
19
+
20
+ | Env var | Default |
21
+ |---|---|
22
+ | `DOLLHOUSE_WEB_CONSOLE_PORT` | `41715` |
23
+ | `DOLLHOUSE_CONSOLE_LEADER_LOCK_FILE` | `~/.dollhouse/run/console-leader.auth.lock` |
24
+ | `DOLLHOUSE_CONSOLE_TOKEN_FILE` | `~/.dollhouse/run/console-token.auth.json` |
25
+
26
+ Port 41715 spells "AILIS" on a phone keypad — the AI Layer Interface Specification.
27
+
28
+ **Legacy detection:** When the authenticated console starts, it checks for an active legacy (pre-auth) DollhouseMCP process on port 3939 and logs a warning if one is found, explaining the coexistence and the differing security posture.
29
+
30
+ **Consumer impact:**
31
+ - **DollhouseBridge**: will need a matching port update when it consumes Phase 2 features — tracking issue filed
32
+ - **User bookmarks / shell scripts**: update `localhost:3939` references to `localhost:41715` (or set `DOLLHOUSE_WEB_CONSOLE_PORT` to whatever you prefer)
33
+ - **Docs**: all `docs/guides/*.md` references updated
34
+ - **Nothing breaks silently**: the auth console simply isn't on 3939 anymore; the legacy console continues to work there untouched
35
+
3
36
  ## [2.0.7] - 2026-04-02
4
37
 
5
38
  ### Clean terminal output for `--web` mode
@@ -191,6 +191,32 @@ export interface WizardConfig {
191
191
  lastSeenVersion?: string;
192
192
  skippedSections?: string[];
193
193
  }
194
+ export type LicenseTier = 'agpl' | 'free-commercial' | 'paid-commercial';
195
+ export interface LicenseConfig {
196
+ tier: LicenseTier;
197
+ email?: string;
198
+ attestedAt?: string;
199
+ telemetryRequired?: boolean;
200
+ revenueScale?: string;
201
+ companyName?: string;
202
+ useCase?: string;
203
+ }
204
+ /** Branded type for validated port numbers (1024–65535). */
205
+ export type PortNumber = number & {
206
+ readonly __brand: 'PortNumber';
207
+ };
208
+ /** Validate and brand a port number. Returns undefined if invalid. */
209
+ export declare function validatePort(value: unknown): PortNumber | undefined;
210
+ export interface ConsoleConfig {
211
+ /**
212
+ * Web console port (1024–65535). Resolution hierarchy:
213
+ * 1. --port CLI flag (standalone --web mode only)
214
+ * 2. This config value (~/.dollhouse/config.yml → console.port)
215
+ * 3. DOLLHOUSE_WEB_CONSOLE_PORT env var
216
+ * 4. Default: 41715
217
+ */
218
+ port: number;
219
+ }
194
220
  export interface SourcePriorityConfigData {
195
221
  order: string[];
196
222
  stop_on_first: boolean;
@@ -208,6 +234,8 @@ export interface DollhouseConfig {
208
234
  wizard: WizardConfig;
209
235
  autoLoad: AutoLoadConfig;
210
236
  retentionPolicy: RetentionPolicyConfig;
237
+ license: LicenseConfig;
238
+ console: ConsoleConfig;
211
239
  source_priority?: SourcePriorityConfigData;
212
240
  }
213
241
  export interface ConfigUpdateResult {
@@ -228,6 +256,12 @@ export declare class ConfigManager {
228
256
  private config;
229
257
  private readonly fileOperations;
230
258
  private os;
259
+ /**
260
+ * Extract console.port from raw YAML config content without full
261
+ * ConfigManager initialization. Uses FAILSAFE_SCHEMA for security
262
+ * (no code execution). Returns undefined if not found or invalid.
263
+ */
264
+ static readPortFromYaml(yamlContent: string): number | undefined;
231
265
  constructor(fileOperations: IFileOperationsService, osModule: typeof os);
232
266
  /**
233
267
  * Get default configuration
@@ -1 +1 @@
1
- {"version":3,"file":"ConfigManager.d.ts","sourceRoot":"","sources":["../../src/config/ConfigManager.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;;GAWG;AAGH,OAAO,KAAK,EAAE,MAAM,IAAI,CAAC;AAWzB,OAAO,EAAE,sBAAsB,EAAE,MAAM,sCAAsC,CAAC;AAE9E,MAAM,WAAW,UAAU;IACzB,QAAQ,EAAE,MAAM,GAAG,IAAI,CAAC;IACxB,KAAK,EAAE,MAAM,GAAG,IAAI,CAAC;IACrB,YAAY,EAAE,MAAM,GAAG,IAAI,CAAC;CAC7B;AAED,MAAM,WAAW,qBAAqB;IACpC,cAAc,EAAE,MAAM,GAAG,IAAI,CAAC;IAC9B,eAAe,EAAE,MAAM,CAAC;IACxB,cAAc,EAAE,MAAM,CAAC;IACvB,WAAW,EAAE,OAAO,CAAC;CACtB;AAED,MAAM,WAAW,gBAAgB;IAC/B,SAAS,EAAE,OAAO,CAAC;IACnB,YAAY,EAAE,aAAa,GAAG,OAAO,GAAG,QAAQ,CAAC;IACjD,SAAS,CAAC,EAAE,MAAM,CAAC;CACpB;AAED,MAAM,WAAW,YAAY;IAC3B,SAAS,EAAE,qBAAqB,CAAC;IACjC,IAAI,EAAE,gBAAgB,CAAC;CACxB;AAED,MAAM,WAAW,oBAAoB;IACnC,oBAAoB,EAAE,OAAO,CAAC;IAC9B,qBAAqB,EAAE,OAAO,CAAC;IAC/B,cAAc,EAAE,OAAO,CAAC;IACxB,YAAY,EAAE,MAAM,CAAC;CACtB;AAED,MAAM,WAAW,cAAc;IAC7B,cAAc,EAAE,OAAO,CAAC;IACxB,gBAAgB,EAAE,OAAO,CAAC;IAC1B,eAAe,EAAE,OAAO,CAAC;IACzB,kBAAkB,EAAE,OAAO,CAAC;CAC7B;AAED,MAAM,WAAW,iBAAiB;IAChC,gBAAgB,EAAE,OAAO,CAAC;IAC1B,YAAY,EAAE,OAAO,CAAC;IACtB,iBAAiB,EAAE,OAAO,CAAC;IAC3B,iBAAiB,EAAE,MAAM,EAAE,CAAC;CAC7B;AAED,MAAM,WAAW,UAAU;IACzB,OAAO,EAAE,OAAO,CAAC;IACjB,UAAU,EAAE,oBAAoB,CAAC;IACjC,IAAI,EAAE,cAAc,CAAC;IACrB,OAAO,EAAE,iBAAiB,CAAC;CAC5B;AAED,MAAM,WAAW,gBAAgB;IAC/B,WAAW,EAAE,OAAO,CAAC;IACrB,cAAc,EAAE,OAAO,CAAC;IACxB,eAAe,EAAE,OAAO,CAAC;CAC1B;AAED,MAAM,WAAW,cAAc;IAC7B,OAAO,EAAE,OAAO,CAAC;IACjB,cAAc,EAAE,MAAM,CAAC;IACvB,qBAAqB,CAAC,EAAE,MAAM,CAAC;IAC/B,2BAA2B,CAAC,EAAE,OAAO,CAAC;IACtC,QAAQ,EAAE,MAAM,EAAE,CAAC;CACpB;AAED;;;;;;;;;;;GAWG;AACH,MAAM,WAAW,qBAAqB;IACpC;;;;OAIG;IACH,OAAO,EAAE,OAAO,CAAC;IAEjB;;;;;;OAMG;IACH,gBAAgB,EAAE,UAAU,GAAG,QAAQ,GAAG,SAAS,GAAG,WAAW,CAAC;IAElE;;OAEG;IACH,MAAM,EAAE;QACN,wDAAwD;QACxD,oBAAoB,EAAE,OAAO,CAAC;QAC9B,2DAA2D;QAC3D,aAAa,EAAE,OAAO,CAAC;QACvB,2DAA2D;QAC3D,gBAAgB,EAAE,OAAO,CAAC;QAC1B,8CAA8C;QAC9C,sBAAsB,EAAE,MAAM,CAAC;KAChC,CAAC;IAEF;;OAEG;IACH,KAAK,EAAE;QACL,kDAAkD;QAClD,aAAa,EAAE,OAAO,CAAC;QACvB,gEAAgE;QAChE,oBAAoB,EAAE,OAAO,CAAC;QAC9B,8CAA8C;QAC9C,qBAAqB,EAAE,MAAM,CAAC;KAC/B,CAAC;IAEF;;;OAGG;IACH,QAAQ,EAAE;QACR,iDAAiD;QACjD,QAAQ,EAAE,MAAM,CAAC;QACjB,kDAAkD;QAClD,WAAW,EAAE,MAAM,CAAC;KACrB,CAAC;CACH;AAED,MAAM,WAAW,8BAA8B;IAC7C,mBAAmB,EAAE,OAAO,CAAC;IAC7B,QAAQ,EAAE;QACR,OAAO,EAAE,OAAO,CAAC;QACjB,IAAI,EAAE,OAAO,CAAC;QACd,KAAK,EAAE,OAAO,CAAC;KAChB,CAAC;CACH;AAED,MAAM,WAAW,mBAAmB;IAClC,OAAO,EAAE,OAAO,CAAC;IACjB,MAAM,EAAE;QACN,qBAAqB,EAAE,MAAM,CAAC;QAC9B,gBAAgB,EAAE,MAAM,CAAC;QACzB,kBAAkB,EAAE,MAAM,CAAC;KAC5B,CAAC;IACF,SAAS,EAAE;QACT,OAAO,EAAE,OAAO,CAAC;QACjB,UAAU,EAAE,MAAM,CAAC;QACnB,eAAe,EAAE,MAAM,CAAC;KACzB,CAAC;IACF,YAAY,CAAC,EAAE;QACb,cAAc,CAAC,EAAE,MAAM,EAAE,CAAC;QAC1B,cAAc,CAAC,EAAE,MAAM,EAAE,CAAC;QAC1B,aAAa,CAAC,EAAE,MAAM,EAAE,CAAC;KAC1B,CAAC;IACF,kBAAkB,CAAC,EAAE;QACnB,OAAO,EAAE,OAAO,CAAC;QACjB,YAAY,EAAE,MAAM,CAAC;QACrB,kBAAkB,EAAE,MAAM,CAAC;KAC5B,CAAC;IACF,SAAS,CAAC,EAAE,8BAA8B,CAAC;CAC5C;AAED,MAAM,WAAW,cAAc;IAC7B,aAAa,EAAE;QACb,QAAQ,CAAC,EAAE,MAAM,EAAE,CAAC;QACpB,MAAM,CAAC,EAAE,MAAM,EAAE,CAAC;QAClB,SAAS,CAAC,EAAE,MAAM,EAAE,CAAC;QACrB,MAAM,CAAC,EAAE,MAAM,EAAE,CAAC;QAClB,QAAQ,CAAC,EAAE,MAAM,EAAE,CAAC;QACpB,SAAS,CAAC,EAAE,MAAM,EAAE,CAAC;KACtB,CAAC;IACF,mBAAmB,EAAE,MAAM,CAAC;IAC5B,cAAc,CAAC,EAAE,mBAAmB,CAAC;CACtC;AAED,MAAM,WAAW,aAAa;IAC5B,kBAAkB,EAAE;QAClB,OAAO,EAAE,OAAO,CAAC;QACjB,KAAK,EAAE,MAAM,GAAG,SAAS,GAAG,SAAS,GAAG,QAAQ,CAAC;QACjD,aAAa,EAAE,OAAO,CAAC;KACxB,CAAC;IACF,eAAe,EAAE,OAAO,CAAC;IACzB,aAAa,EAAE,OAAO,CAAC;CACxB;AAED,MAAM,WAAW,YAAY;IAC3B,SAAS,EAAE,OAAO,CAAC;IACnB,SAAS,EAAE,OAAO,CAAC;IACnB,WAAW,CAAC,EAAE,MAAM,CAAC;IACrB,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,eAAe,CAAC,EAAE,MAAM,CAAC;IACzB,eAAe,CAAC,EAAE,MAAM,EAAE,CAAC;CAC5B;AAED,MAAM,WAAW,wBAAwB;IACvC,KAAK,EAAE,MAAM,EAAE,CAAC;IAChB,aAAa,EAAE,OAAO,CAAC;IACvB,qBAAqB,EAAE,OAAO,CAAC;IAC/B,iBAAiB,EAAE,OAAO,CAAC;CAC5B;AAED,MAAM,WAAW,eAAe;IAC9B,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,EAAE,UAAU,CAAC;IACjB,MAAM,EAAE,YAAY,CAAC;IACrB,IAAI,EAAE,UAAU,CAAC;IACjB,UAAU,EAAE,gBAAgB,CAAC;IAC7B,QAAQ,EAAE,cAAc,CAAC;IACzB,OAAO,EAAE,aAAa,CAAC;IACvB,MAAM,EAAE,YAAY,CAAC;IACrB,QAAQ,EAAE,cAAc,CAAC;IACzB,eAAe,EAAE,qBAAqB,CAAC;IACvC,eAAe,CAAC,EAAE,wBAAwB,CAAC;CAC5C;AAED,MAAM,WAAW,kBAAkB;IACjC,OAAO,EAAE,OAAO,CAAC;IACjB,OAAO,EAAE,MAAM,CAAC;IAChB,aAAa,CAAC,EAAE,GAAG,CAAC;IACpB,QAAQ,CAAC,EAAE,GAAG,CAAC;CAChB;AAED,MAAM,WAAW,kBAAkB;IACjC,OAAO,EAAE,OAAO,CAAC;IACjB,OAAO,EAAE,MAAM,CAAC;IAChB,IAAI,CAAC,EAAE,GAAG,CAAC;CACZ;AAED,qBAAa,aAAa;IACxB,OAAO,CAAC,SAAS,CAAS;IAC1B,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,UAAU,CAAS;IAC3B,OAAO,CAAC,MAAM,CAAgC;IAC9C,OAAO,CAAC,QAAQ,CAAC,cAAc,CAAyB;IACxD,OAAO,CAAC,EAAE,CAAY;gBAEV,cAAc,EAAE,sBAAsB,EAAE,QAAQ,EAAE,OAAO,EAAE;IAgBvE;;OAEG;IACH,OAAO,CAAC,gBAAgB;IA2HxB;;OAEG;IACU,UAAU,IAAI,OAAO,CAAC,IAAI,CAAC;IAsCxC;;OAEG;YACW,UAAU;IA+ExB;;OAEG;YACW,YAAY;IAI1B;;;OAGG;IACI,iBAAiB,IAAI,MAAM,GAAG,IAAI;IAYzC;;OAEG;IACU,iBAAiB,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAuB/D;;OAEG;IACI,SAAS,IAAI,eAAe;IAOnC;;OAEG;IACI,UAAU,CAAC,CAAC,EAAE,IAAI,EAAE,MAAM,EAAE,YAAY,CAAC,EAAE,CAAC,GAAG,CAAC,GAAG,SAAS;IAmBnE;;;;;OAKG;IACU,aAAa,CAAC,IAAI,EAAE,MAAM,EAAE,KAAK,EAAE,GAAG,GAAG,OAAO,CAAC,kBAAkB,CAAC;IA2CjF;;;;;;;;;;;;OAYG;WACW,gBAAgB,CAAC,QAAQ,EAAE,GAAG,GAAG,OAAO;IAUtD;;OAEG;YACW,UAAU;IA4DxB;;OAEG;YACW,YAAY;IAI1B;;OAEG;IACH,OAAO,CAAC,cAAc;IAuGtB;;;;;OAKG;IACH,OAAO,CAAC,iBAAiB;IAyGzB;;OAEG;YACW,sBAAsB;IAsCpC;;;;;OAKG;IACU,WAAW,CAAC,OAAO,CAAC,EAAE,MAAM,GAAG,OAAO,CAAC,kBAAkB,CAAC;IA2CvE;;OAEG;IACU,YAAY,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,kBAAkB,CAAC;IAmCxE;;OAEG;IACU,YAAY,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,kBAAkB,CAAC;IAsCxE;;OAEG;IACI,kBAAkB,CAAC,OAAO,CAAC,EAAE,MAAM,GAAG,MAAM;CA6BpD"}
1
+ {"version":3,"file":"ConfigManager.d.ts","sourceRoot":"","sources":["../../src/config/ConfigManager.ts"],"names":[],"mappings":"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"}
@@ -17,6 +17,16 @@ import { logger } from '../utils/logger.js';
17
17
  import { SecureYamlParser } from '../security/secureYamlParser.js';
18
18
  import { validatePropertyPath, safeSetProperty, createSafeObject, safeHasOwnProperty } from '../utils/securityUtils.js';
19
19
  import { env } from './env.js';
20
+ /** Validate and brand a port number. Returns undefined if invalid. */
21
+ export function validatePort(value) {
22
+ const num = typeof value === 'string' ? Number(value) : value;
23
+ if (typeof num !== 'number' || !Number.isFinite(num))
24
+ return undefined;
25
+ const port = Math.floor(num);
26
+ if (port < 1024 || port > 65535)
27
+ return undefined;
28
+ return port;
29
+ }
20
30
  export class ConfigManager {
21
31
  configDir;
22
32
  configPath;
@@ -24,6 +34,24 @@ export class ConfigManager {
24
34
  config = null;
25
35
  fileOperations;
26
36
  os;
37
+ /**
38
+ * Extract console.port from raw YAML config content without full
39
+ * ConfigManager initialization. Uses FAILSAFE_SCHEMA for security
40
+ * (no code execution). Returns undefined if not found or invalid.
41
+ */
42
+ static readPortFromYaml(yamlContent) {
43
+ try {
44
+ const parsed = yaml.load(yamlContent, { schema: yaml.FAILSAFE_SCHEMA });
45
+ const raw = parsed?.console?.port;
46
+ if (raw === undefined || raw === null)
47
+ return undefined;
48
+ const port = Number(raw);
49
+ return Number.isFinite(port) ? port : undefined;
50
+ }
51
+ catch {
52
+ return undefined;
53
+ }
54
+ }
27
55
  constructor(fileOperations, osModule) {
28
56
  this.fileOperations = fileOperations;
29
57
  this.os = osModule;
@@ -138,6 +166,12 @@ export class ConfigManager {
138
166
  completed: false,
139
167
  dismissed: false
140
168
  },
169
+ license: {
170
+ tier: 'agpl'
171
+ },
172
+ console: {
173
+ port: 41715
174
+ },
141
175
  /**
142
176
  * Retention Policy Configuration (Issue #51)
143
177
  * IMPORTANT: Disabled by default - nothing is auto-deleted without explicit consent.
@@ -358,6 +392,17 @@ export class ConfigManager {
358
392
  }
359
393
  // SECURITY: Validate path to prevent prototype pollution
360
394
  validatePropertyPath(path, 'path');
395
+ // Runtime validation for known typed settings (#1840)
396
+ if (path === 'console.port') {
397
+ const validated = validatePort(value);
398
+ if (!validated) {
399
+ return {
400
+ success: false,
401
+ message: `Invalid port: ${value}. Must be an integer between 1024 and 65535.`,
402
+ };
403
+ }
404
+ value = validated;
405
+ }
361
406
  const keys = path.split('.');
362
407
  let current = this.config;
363
408
  const previousValue = this.getSetting(path);
@@ -849,4 +894,4 @@ export class ConfigManager {
849
894
  });
850
895
  }
851
896
  }
852
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ConfigManager.js","sourceRoot":"","sources":["../../src/config/ConfigManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ConfigManager - Centralized configuration management for DollhouseMCP\n * \n * Features:\n * - YAML-based configuration file\n * - Default values with user overrides\n * - Migration from environment variables\n * - Validation and type safety\n * - Atomic updates with backup\n * - Privacy-first defaults\n * - OAuth client ID storage for MCP client integration\n */\n\nimport * as path from 'path';\nimport * as os from 'os';\nimport * as yaml from 'js-yaml';\nimport { logger } from '../utils/logger.js';\nimport { SecureYamlParser } from '../security/secureYamlParser.js';\nimport {\n  validatePropertyPath,\n  safeSetProperty,\n  createSafeObject,\n  safeHasOwnProperty\n} from '../utils/securityUtils.js';\nimport { env } from './env.js';\nimport { IFileOperationsService } from '../services/FileOperationsService.js';\n\nexport interface UserConfig {\n  username: string | null;\n  email: string | null;\n  display_name: string | null;\n}\n\nexport interface GitHubPortfolioConfig {\n  repository_url: string | null;\n  repository_name: string;\n  default_branch: string;\n  auto_create: boolean;\n}\n\nexport interface GitHubAuthConfig {\n  use_oauth: boolean;\n  token_source: 'environment' | 'oauth' | 'config';\n  client_id?: string; // OAuth client ID for GitHub App\n}\n\nexport interface GitHubConfig {\n  portfolio: GitHubPortfolioConfig;\n  auth: GitHubAuthConfig;\n}\n\nexport interface SyncIndividualConfig {\n  require_confirmation: boolean;\n  show_diff_before_sync: boolean;\n  track_versions: boolean;\n  keep_history: number;\n}\n\nexport interface SyncBulkConfig {\n  upload_enabled: boolean;\n  download_enabled: boolean;\n  require_preview: boolean;\n  respect_local_only: boolean;\n}\n\nexport interface SyncPrivacyConfig {\n  scan_for_secrets: boolean;\n  scan_for_pii: boolean;\n  warn_on_sensitive: boolean;\n  excluded_patterns: string[];\n}\n\nexport interface SyncConfig {\n  enabled: boolean;\n  individual: SyncIndividualConfig;\n  bulk: SyncBulkConfig;\n  privacy: SyncPrivacyConfig;\n}\n\nexport interface CollectionConfig {\n  auto_submit: boolean;\n  require_review: boolean;\n  add_attribution: boolean;\n}\n\nexport interface AutoLoadConfig {\n  enabled: boolean;\n  maxTokenBudget: number;\n  maxSingleMemoryTokens?: number;\n  suppressLargeMemoryWarnings?: boolean;\n  memories: string[];\n}\n\n/**\n * Retention Policy Configuration (Issue #51)\n *\n * Controls automatic deletion of expired memory entries.\n * IMPORTANT: Disabled by default - nothing is auto-deleted without explicit consent.\n *\n * Use cases:\n * - Legal/compliance: Law firms, accountants (7-year retention)\n * - Privacy-focused: Signal-like auto-expiring messages\n * - Storage management: Cleanup of old logs\n * - GDPR: Right-to-be-forgotten implementations\n */\nexport interface RetentionPolicyConfig {\n  /**\n   * Master switch for retention enforcement.\n   * If false, no automatic retention enforcement happens anywhere.\n   * Default: false (nothing auto-deleted)\n   */\n  enabled: boolean;\n\n  /**\n   * When retention enforcement happens:\n   * - 'disabled': Never enforce automatically (only via explicit command)\n   * - 'manual': Only when user explicitly requests enforcement\n   * - 'on_load': Enforce when memory is loaded (CURRENT BEHAVIOR - not recommended)\n   * - 'scheduled': Run on a schedule (future implementation)\n   */\n  enforcement_mode: 'disabled' | 'manual' | 'on_load' | 'scheduled';\n\n  /**\n   * Safety controls to prevent accidental data loss\n   */\n  safety: {\n    /** Require explicit confirmation before any deletion */\n    require_confirmation: boolean;\n    /** Always preview what would be deleted before doing it */\n    dry_run_first: boolean;\n    /** Show warning when entries are approaching expiration */\n    warn_on_expiring: boolean;\n    /** Days before expiration to start warning */\n    warning_threshold_days: number;\n  };\n\n  /**\n   * Audit and logging\n   */\n  audit: {\n    /** Log all retention deletions for audit trail */\n    log_deletions: boolean;\n    /** Keep deleted entries in a backup before permanent removal */\n    backup_before_delete: boolean;\n    /** Days to keep backups of deleted entries */\n    backup_retention_days: number;\n  };\n\n  /**\n   * Default TTL settings when creating new memories\n   * Note: These are defaults only - individual memories can override\n   */\n  defaults: {\n    /** Default TTL in days for new memory entries */\n    ttl_days: number;\n    /** Maximum entries before capacity enforcement */\n    max_entries: number;\n  };\n}\n\nexport interface CapabilityIndexResourcesConfig {\n  advertise_resources: boolean;\n  variants: {\n    summary: boolean;\n    full: boolean;\n    stats: boolean;\n  };\n}\n\nexport interface EnhancedIndexConfig {\n  enabled: boolean;\n  limits: {\n    maxTriggersPerElement: number;\n    maxTriggerLength: number;\n    maxKeywordsToCheck: number;\n  };\n  telemetry: {\n    enabled: boolean;\n    sampleRate: number;\n    metricsInterval: number;\n  };\n  verbPatterns?: {\n    customPrefixes?: string[];\n    customSuffixes?: string[];\n    excludedNouns?: string[];\n  };\n  backgroundAnalysis?: {\n    enabled: boolean;\n    scanInterval: number;\n    maxConcurrentScans: number;\n  };\n  resources?: CapabilityIndexResourcesConfig;\n}\n\nexport interface ElementsConfig {\n  auto_activate: {\n    personas?: string[];\n    skills?: string[];\n    templates?: string[];\n    agents?: string[];\n    memories?: string[];\n    ensembles?: string[];\n  };\n  default_element_dir: string;\n  enhanced_index?: EnhancedIndexConfig;\n}\n\nexport interface DisplayConfig {\n  persona_indicators: {\n    enabled: boolean;\n    style: 'full' | 'minimal' | 'compact' | 'custom';\n    include_emoji: boolean;\n  };\n  verbose_logging: boolean;\n  show_progress: boolean;\n}\n\nexport interface WizardConfig {\n  completed: boolean;      // Wizard was successfully completed\n  dismissed: boolean;      // User chose \"Don't show again\"\n  completedAt?: string;    // ISO timestamp when completed\n  version?: string;        // Wizard version for future updates (deprecated - use lastSeenVersion)\n  lastSeenVersion?: string; // Last version where user saw the wizard\n  skippedSections?: string[]; // Track which sections were skipped\n}\n\nexport interface SourcePriorityConfigData {\n  order: string[];  // Array of 'local' | 'github' | 'collection'\n  stop_on_first: boolean;\n  check_all_for_updates: boolean;\n  fallback_on_error: boolean;\n}\n\nexport interface DollhouseConfig {\n  version: string;\n  user: UserConfig;\n  github: GitHubConfig;\n  sync: SyncConfig;\n  collection: CollectionConfig;\n  elements: ElementsConfig;\n  display: DisplayConfig;\n  wizard: WizardConfig;\n  autoLoad: AutoLoadConfig;\n  retentionPolicy: RetentionPolicyConfig;\n  source_priority?: SourcePriorityConfigData;\n}\n\nexport interface ConfigUpdateResult {\n  success: boolean;\n  message: string;\n  previousValue?: any;\n  newValue?: any;\n}\n\nexport interface ConfigActionResult {\n  success: boolean;\n  message: string;\n  data?: any;\n}\n\nexport class ConfigManager {\n  private configDir: string;\n  private configPath: string;\n  private backupPath: string;\n  private config: DollhouseConfig | null = null;\n  private readonly fileOperations: IFileOperationsService;\n  private os: typeof os;\n\n  constructor(fileOperations: IFileOperationsService, osModule: typeof os) {\n    this.fileOperations = fileOperations;\n    this.os = osModule;\n\n    // Initialize paths - use test directory if in test environment\n    // NOTE: Using process.env directly here (not cached env.TEST_CONFIG_DIR)\n    // to allow tests to dynamically override the config directory\n    if (env.NODE_ENV === 'test' && process.env.TEST_CONFIG_DIR) {\n      this.configDir = process.env.TEST_CONFIG_DIR;\n    } else {\n      this.configDir = path.join(this.os.homedir(), '.dollhouse');\n    }\n    this.configPath = path.join(this.configDir, 'config.yml');\n    this.backupPath = path.join(this.configDir, 'config.yml.backup');\n  }\n\n  /**\n   * Get default configuration\n   */\n  private getDefaultConfig(): DollhouseConfig {\n    return {\n      version: '1.0.0',\n      user: {\n        username: null,\n        email: null,\n        display_name: null\n      },\n      github: {\n        portfolio: {\n          repository_url: null,\n          repository_name: env.GITHUB_REPOSITORY || 'dollhouse-portfolio',\n          default_branch: 'main',\n          auto_create: true\n        },\n        auth: {\n          use_oauth: true,\n          token_source: 'environment'\n        }\n      },\n      sync: {\n        enabled: false, // Privacy first - off by default\n        individual: {\n          require_confirmation: true,\n          show_diff_before_sync: true,\n          track_versions: true,\n          keep_history: 10\n        },\n        bulk: {\n          upload_enabled: false, // Requires explicit enablement\n          download_enabled: false,\n          require_preview: true,\n          respect_local_only: true\n        },\n        privacy: {\n          scan_for_secrets: true,\n          scan_for_pii: true,\n          warn_on_sensitive: true,\n          excluded_patterns: [\n            '*.secret',\n            '*-private.*',\n            'credentials/**',\n            'personal/**'\n          ]\n        }\n      },\n      collection: {\n        auto_submit: false, // Never auto-submit\n        require_review: true,\n        add_attribution: true\n      },\n      autoLoad: {\n        enabled: true, // Auto-load baseline memories by default\n        maxTokenBudget: 5000,\n        maxSingleMemoryTokens: undefined,\n        suppressLargeMemoryWarnings: false,\n        memories: [] // Configured via memory files with autoLoad flag\n      },\n      elements: {\n        auto_activate: {},\n        default_element_dir: path.join(os.homedir(), '.dollhouse', 'portfolio'),\n        enhanced_index: {\n          enabled: true,\n          limits: {\n            maxTriggersPerElement: 50,\n            maxTriggerLength: 50,\n            maxKeywordsToCheck: 100\n          },\n          telemetry: {\n            enabled: false,  // Opt-in only\n            sampleRate: 0.1,\n            metricsInterval: 60000\n          },\n          resources: {\n            advertise_resources: false, // Default: safe, disabled\n            variants: {\n              summary: false,  // ~1,254 tokens - Opt-in required\n              full: false,     // ~48,306 tokens - Opt-in required\n              stats: true      // ~50 tokens - Safe by default\n            }\n          }\n        }\n      },\n      display: {\n        persona_indicators: {\n          enabled: true,\n          style: 'minimal',\n          include_emoji: true\n        },\n        verbose_logging: false,\n        show_progress: true\n      },\n      wizard: {\n        completed: false,\n        dismissed: false\n      },\n      /**\n       * Retention Policy Configuration (Issue #51)\n       * IMPORTANT: Disabled by default - nothing is auto-deleted without explicit consent.\n       * Users must explicitly enable retention enforcement if they want automatic cleanup.\n       */\n      retentionPolicy: {\n        enabled: false,                    // DISABLED by default - no auto-deletion\n        enforcement_mode: 'disabled',       // Only manual enforcement allowed\n        safety: {\n          require_confirmation: true,       // Always require confirmation\n          dry_run_first: true,              // Always preview before deleting\n          warn_on_expiring: true,           // Warn when entries approaching expiration\n          warning_threshold_days: 7         // Warn 7 days before expiration\n        },\n        audit: {\n          log_deletions: true,              // Log all deletions\n          backup_before_delete: true,       // Backup deleted entries\n          backup_retention_days: 30         // Keep backups for 30 days\n        },\n        defaults: {\n          ttl_days: 30,                     // Default 30-day TTL (if enabled)\n          max_entries: 1000                 // Default max entries per memory\n        }\n      }\n    };\n  }\n\n  /**\n   * Initialize configuration\n   */\n  public async initialize(): Promise<void> {\n    // Always reload config from disk if it exists, even if we have defaults in memory\n    // This ensures we pick up any manual edits or saved settings\n    \n    try {\n      // Ensure config directory exists with proper permissions (0o700 = owner only)\n      await this.fileOperations.createDirectory(this.configDir);\n      // Set permissions on the directory\n      await this.fileOperations.chmod(this.configDir, 0o700, {\n        source: 'ConfigManager.initialize'\n      });\n      \n      // Load or create config\n      if (await this.configExists()) {\n        await this.loadConfig();\n      } else {\n        // Create default config\n        this.config = this.getDefaultConfig();\n        \n        // Try to migrate from environment variables\n        await this.migrateFromEnvironment();\n        \n        // Save the config\n        await this.saveConfig();\n        \n        logger.info('Created new configuration file', {\n          path: this.configPath\n        });\n      }\n    } catch (error) {\n      logger.error('Failed to initialize configuration', {\n        error: error instanceof Error ? error.message : String(error)\n      });\n      // Use defaults in memory\n      this.config = this.getDefaultConfig();\n    }\n  }\n\n  /**\n   * Load configuration from file\n   */\n  private async loadConfig(): Promise<void> {\n    try {\n      const content = await this.fileOperations.readFile(this.configPath, {\n        source: 'ConfigManager.loadConfig'\n      });\n      \n      /**\n       * IMPORTANT: Parser Selection for Different File Types\n       * \n       * We use DIFFERENT parsers for different file types:\n       * \n       * 1. js-yaml (used here) - For PURE YAML files:\n       *    - Configuration files (config.yml)\n       *    - Data files without markdown content\n       *    - Any .yml or .yaml file that's just YAML\n       *    Example format:\n       *    ```yaml\n       *    version: 1.0.0\n       *    user:\n       *      username: johndoe\n       *      email: john@example.com\n       *    ```\n       * \n       * 2. SecureYamlParser - For MARKDOWN files with YAML frontmatter:\n       *    - Persona files (*.md in personas/)\n       *    - Skill files (*.md in skills/)\n       *    - Template files (*.md in templates/)\n       *    - Any .md file with frontmatter between --- markers\n       *    Example format:\n       *    ```markdown\n       *    ---\n       *    name: Creative Writer\n       *    description: A creative assistant\n       *    ---\n       *    # Instructions\n       *    You are a creative writer...\n       *    ```\n       * \n       * The config file is PURE YAML, so we use js-yaml directly with FAILSAFE_SCHEMA\n       * for security (prevents code execution via YAML tags).\n       * SECURITY: This is NOT a vulnerability - FAILSAFE_SCHEMA prevents code execution\n       */\n      let loadedData: any;\n      try {\n        // Using yaml with FAILSAFE_SCHEMA is secure - prevents code execution\n        loadedData = yaml.load(content, {\n          schema: yaml.FAILSAFE_SCHEMA // Safe schema prevents code execution\n        });\n      } catch (yamlError) {\n        throw new Error(`Invalid YAML in configuration file: ${yamlError instanceof Error ? yamlError.message : String(yamlError)}`);\n      }\n      \n      if (!loadedData || typeof loadedData !== 'object') {\n        throw new Error('Invalid configuration format');\n      }\n      logger.debug('Loaded config from file', {\n        username: loadedData.user?.username,\n        email: loadedData.user?.email,\n        syncEnabled: loadedData.sync?.enabled\n      });\n      \n      this.config = this.mergeWithDefaults(loadedData);\n      \n      // Fix any string booleans that might have been saved incorrectly\n      this.fixConfigTypes();\n      \n      logger.debug('Configuration loaded successfully', {\n        username: this.config.user.username,\n        syncEnabled: this.config.sync.enabled\n      });\n      \n    } catch (error) {\n      logger.error('Failed to load configuration', {\n        error: error instanceof Error ? error.message : String(error)\n      });\n      throw error;\n    }\n  }\n\n  /**\n   * Check if config file exists\n   */\n  private async configExists(): Promise<boolean> {\n    return this.fileOperations.exists(this.configPath);\n  }\n\n  /**\n   * Get GitHub OAuth client ID\n   * Environment variable takes precedence over config file\n   */\n  public getGitHubClientId(): string | null {\n    // NOTE: DOLLHOUSE_GITHUB_CLIENT_ID is not in centralized env config\n    // as it's an optional feature flag, so we still use process.env here\n    const envClientId = process.env.DOLLHOUSE_GITHUB_CLIENT_ID;\n    if (envClientId) {\n      return envClientId;\n    }\n\n    // Fall back to config file\n    return this.config?.github?.auth?.client_id || null;\n  }\n\n  /**\n   * Set GitHub OAuth client ID in config file\n   */\n  public async setGitHubClientId(clientId: string): Promise<void> {\n    if (!ConfigManager.validateClientId(clientId)) {\n      throw new Error(\n        `Invalid GitHub client ID format. Expected format: Ov23li followed by at least 14 alphanumeric characters (e.g., Ov23liABCDEFGHIJKLMN)`\n      );\n    }\n\n    if (!this.config) {\n      this.config = this.getDefaultConfig();\n    }\n\n    // Ensure github.auth object exists\n    if (!this.config.github) {\n      this.config.github = this.getDefaultConfig().github;\n    }\n    if (!this.config.github.auth) {\n      this.config.github.auth = this.getDefaultConfig().github.auth;\n    }\n\n    this.config.github.auth.client_id = clientId;\n    await this.saveConfig();\n  }\n\n  /**\n   * Get the current configuration\n   */\n  public getConfig(): DollhouseConfig {\n    if (!this.config) {\n      throw new Error('Configuration not initialized');\n    }\n    return this.config;\n  }\n\n  /**\n   * Get a specific setting using dot notation\n   */\n  public getSetting<T>(path: string, defaultValue?: T): T | undefined {\n    if (!this.config) {\n      return defaultValue;\n    }\n    \n    const keys = path.split('.');\n    let value: any = this.config;\n    \n    for (const key of keys) {\n      if (value && typeof value === 'object' && key in value) {\n        value = value[key];\n      } else {\n        return defaultValue;\n      }\n    }\n    \n    return value as T;\n  }\n\n  /**\n   * Update a specific setting using dot notation\n   * SECURITY FIX (PR #895): Added prototype pollution protection\n   * Previously: Direct property assignment allowed __proto__ injection\n   * Now: Validates keys against forbidden properties before assignment\n   */\n  public async updateSetting(path: string, value: any): Promise<ConfigUpdateResult> {\n    if (!this.config) {\n      await this.initialize();\n    }\n    \n    // SECURITY: Validate path to prevent prototype pollution\n    validatePropertyPath(path, 'path');\n\n    const keys = path.split('.');\n    let current: any = this.config;\n    const previousValue = this.getSetting(path);\n\n    // Navigate to the parent object using security utilities\n    for (let i = 0; i < keys.length - 1; i++) {\n      const key = keys[i];\n      // SECURITY: Use safe property check and create prototype-less objects\n      if (!safeHasOwnProperty(current, key)) {\n        current[key] = createSafeObject();\n      }\n      current = current[key];\n    }\n\n    // Set the value using secure property setter\n    const lastKey = keys[keys.length - 1];\n    safeSetProperty(current, lastKey, value);\n    \n    // Save the configuration\n    await this.saveConfig();\n    \n    logger.info('Configuration setting updated', {\n      path,\n      previousValue,\n      newValue: value\n    });\n    \n    return {\n      success: true,\n      message: `Setting '${path}' updated successfully`,\n      previousValue,\n      newValue: value\n    };\n  }\n\n  /**\n   * Validate GitHub OAuth client ID format\n   * Client IDs start with \"Ov23li\" followed by at least 14 alphanumeric characters\n   * \n   * @param clientId - The client ID to validate\n   * @returns true if valid, false otherwise\n   * \n   * @example\n   * ConfigManager.validateClientId(\"Ov23liABCDEFGHIJKLMN123456\") // true\n   * ConfigManager.validateClientId(\"invalid\") // false\n   * ConfigManager.validateClientId(\"Ov23li\") // false (too short)\n   * ConfigManager.validateClientId(\"Xv23liABCDEFGHIJKLMN\") // false (wrong prefix)\n   */\n  public static validateClientId(clientId: any): boolean {\n    if (typeof clientId !== 'string' || !clientId) {\n      return false;\n    }\n\n    // GitHub OAuth client IDs follow the pattern: Ov23li[A-Za-z0-9]{14,}\n    const clientIdPattern = /^Ov23li[A-Za-z0-9]{14,}$/;\n    return clientIdPattern.test(clientId);\n  }\n\n  /**\n   * Save configuration to file\n   */\n  private async saveConfig(): Promise<void> {\n    if (!this.config) {\n      throw new Error('No configuration to save');\n    }\n    \n    try {\n      // Create backup of existing config\n      if (await this.configExists()) {\n        await this.fileOperations.copyFile(this.configPath, this.backupPath, {\n          source: 'ConfigManager.saveConfig'\n        });\n      }\n\n      // Convert to YAML\n      // Note: We use js-yaml's dump() for pure YAML output (no frontmatter markers)\n      // This creates a standard YAML file, not a markdown file with frontmatter\n      const yamlContent = yaml.dump(this.config, {\n        indent: 2,\n        lineWidth: 120,\n        noRefs: true,\n        sortKeys: false\n        // Using default schema (not FAILSAFE) for dump to preserve types like booleans\n      });\n\n      // Write atomically with proper permissions (0o600 = owner read/write only)\n      const tempPath = `${this.configPath}.tmp`;\n      await this.fileOperations.writeFile(tempPath, yamlContent, {\n        source: 'ConfigManager.saveConfig'\n      });\n      await this.fileOperations.chmod(tempPath, 0o600, {\n        source: 'ConfigManager.saveConfig'\n      });\n      await this.fileOperations.renameFile(tempPath, this.configPath);\n      \n      logger.debug('Configuration saved successfully');\n      \n      // Log audit event for configuration update\n      logger.debug('Configuration update audit', {\n        event: 'CONFIG_UPDATED',\n        source: 'ConfigManager.saveConfig',\n        timestamp: new Date().toISOString()\n      });\n      \n    } catch (error) {\n      logger.error('Failed to save configuration', {\n        error: error instanceof Error ? error.message : String(error)\n      });\n      \n      // Try to restore backup\n      if (await this.backupExists()) {\n        await this.fileOperations.copyFile(this.backupPath, this.configPath, {\n          source: 'ConfigManager.saveConfig'\n        });\n        logger.info('Restored configuration from backup');\n      }\n      \n      throw error;\n    }\n  }\n\n  /**\n   * Check if backup exists\n   */\n  private async backupExists(): Promise<boolean> {\n    return this.fileOperations.exists(this.backupPath);\n  }\n\n  /**\n   * Fix incorrect types in config (e.g., string booleans, string \"null\")\n   */\n  private fixConfigTypes(): void {\n    if (!this.config) return;\n    \n    // Helper to convert string \"null\" to actual null\n    const fixNull = (value: any): any => {\n      if (value === 'null' || value === 'NULL') return null;\n      return value;\n    };\n    \n    // Helper to convert string booleans to actual booleans\n    const fixBoolean = (value: any): any => {\n      if (typeof value === 'string') {\n        const lower = value.toLowerCase();\n        if (lower === 'true') return true;\n        if (lower === 'false') return false;\n      }\n      return value;\n    };\n    \n    // Fix user fields - handle string \"null\" values\n    if (this.config.user) {\n      this.config.user.username = fixNull(this.config.user.username);\n      this.config.user.email = fixNull(this.config.user.email);\n      this.config.user.display_name = fixNull(this.config.user.display_name);\n    }\n    \n    // Fix sync settings\n    if (this.config.sync) {\n      this.config.sync.enabled = fixBoolean(this.config.sync.enabled);\n      \n      if (this.config.sync.individual) {\n        this.config.sync.individual.require_confirmation = fixBoolean(this.config.sync.individual.require_confirmation);\n        this.config.sync.individual.show_diff_before_sync = fixBoolean(this.config.sync.individual.show_diff_before_sync);\n        this.config.sync.individual.track_versions = fixBoolean(this.config.sync.individual.track_versions);\n      }\n      \n      if (this.config.sync.bulk) {\n        this.config.sync.bulk.upload_enabled = fixBoolean(this.config.sync.bulk.upload_enabled);\n        this.config.sync.bulk.download_enabled = fixBoolean(this.config.sync.bulk.download_enabled);\n        this.config.sync.bulk.require_preview = fixBoolean(this.config.sync.bulk.require_preview);\n        this.config.sync.bulk.respect_local_only = fixBoolean(this.config.sync.bulk.respect_local_only);\n      }\n      \n      if (this.config.sync.privacy) {\n        this.config.sync.privacy.scan_for_secrets = fixBoolean(this.config.sync.privacy.scan_for_secrets);\n        this.config.sync.privacy.scan_for_pii = fixBoolean(this.config.sync.privacy.scan_for_pii);\n        this.config.sync.privacy.warn_on_sensitive = fixBoolean(this.config.sync.privacy.warn_on_sensitive);\n      }\n    }\n    \n    // Fix collection settings\n    if (this.config.collection) {\n      this.config.collection.auto_submit = fixBoolean(this.config.collection.auto_submit);\n      this.config.collection.require_review = fixBoolean(this.config.collection.require_review);\n      this.config.collection.add_attribution = fixBoolean(this.config.collection.add_attribution);\n    }\n    \n    // Fix display settings\n    if (this.config.display) {\n      if (this.config.display.persona_indicators) {\n        this.config.display.persona_indicators.enabled = fixBoolean(this.config.display.persona_indicators.enabled);\n        this.config.display.persona_indicators.include_emoji = fixBoolean(this.config.display.persona_indicators.include_emoji);\n      }\n      this.config.display.verbose_logging = fixBoolean(this.config.display.verbose_logging);\n      this.config.display.show_progress = fixBoolean(this.config.display.show_progress);\n    }\n    \n    // Fix github settings\n    if (this.config.github) {\n      if (this.config.github.portfolio) {\n        this.config.github.portfolio.repository_url = fixNull(this.config.github.portfolio.repository_url);\n        this.config.github.portfolio.auto_create = fixBoolean(this.config.github.portfolio.auto_create);\n      }\n      if (this.config.github.auth) {\n        this.config.github.auth.use_oauth = fixBoolean(this.config.github.auth.use_oauth);\n        // Fix client_id if it's a string \"null\"\n        if (this.config.github.auth.client_id) {\n          this.config.github.auth.client_id = fixNull(this.config.github.auth.client_id) || undefined;\n        }\n      }\n    }\n    \n    // Fix wizard settings\n    if (this.config.wizard) {\n      this.config.wizard.completed = fixBoolean(this.config.wizard.completed);\n      this.config.wizard.dismissed = fixBoolean(this.config.wizard.dismissed);\n    }\n\n    // Fix retention policy settings (Issue #51)\n    if (this.config.retentionPolicy) {\n      this.config.retentionPolicy.enabled = fixBoolean(this.config.retentionPolicy.enabled);\n      if (this.config.retentionPolicy.safety) {\n        this.config.retentionPolicy.safety.require_confirmation = fixBoolean(this.config.retentionPolicy.safety.require_confirmation);\n        this.config.retentionPolicy.safety.dry_run_first = fixBoolean(this.config.retentionPolicy.safety.dry_run_first);\n        this.config.retentionPolicy.safety.warn_on_expiring = fixBoolean(this.config.retentionPolicy.safety.warn_on_expiring);\n      }\n      if (this.config.retentionPolicy.audit) {\n        this.config.retentionPolicy.audit.log_deletions = fixBoolean(this.config.retentionPolicy.audit.log_deletions);\n        this.config.retentionPolicy.audit.backup_before_delete = fixBoolean(this.config.retentionPolicy.audit.backup_before_delete);\n      }\n    }\n  }\n\n  /**\n   * Merge partial config with defaults\n   * \n   * IMPORTANT: This function preserves unknown fields for forward compatibility.\n   * If a future version adds new config fields, older versions won't lose them.\n   */\n  private mergeWithDefaults(partial: Partial<DollhouseConfig>): DollhouseConfig {\n    const defaults = this.getDefaultConfig();\n    \n    // Start with a deep clone of partial to preserve all unknown fields\n    const result: any = JSON.parse(JSON.stringify(partial));\n    \n    // Ensure all required fields exist with defaults\n    result.version = result.version || defaults.version;\n    \n    // User section - preserve unknown fields while ensuring required fields\n    result.user = {\n      ...result.user,\n      username: result.user?.username ?? defaults.user.username,\n      email: result.user?.email ?? defaults.user.email,\n      display_name: result.user?.display_name ?? defaults.user.display_name\n    };\n    \n    // GitHub section - deep merge preserving unknown fields\n    if (!result.github) result.github = {};\n    result.github.portfolio = {\n      ...defaults.github.portfolio,\n      ...result.github.portfolio\n    };\n    result.github.auth = {\n      ...defaults.github.auth,\n      ...result.github.auth\n    };\n    \n    // Sync section - preserve unknown fields at all levels\n    if (!result.sync) result.sync = {};\n    result.sync.enabled = result.sync.enabled ?? defaults.sync.enabled;\n    result.sync.individual = {\n      ...defaults.sync.individual,\n      ...result.sync.individual\n    };\n    result.sync.bulk = {\n      ...defaults.sync.bulk,\n      ...result.sync.bulk\n    };\n    result.sync.privacy = {\n      ...defaults.sync.privacy,\n      ...result.sync.privacy,\n      // Special handling for arrays - use provided or default\n      excluded_patterns: result.sync.privacy?.excluded_patterns || defaults.sync.privacy.excluded_patterns\n    };\n    \n    // Collection section\n    result.collection = {\n      ...defaults.collection,\n      ...result.collection\n    };\n    \n    // Elements section\n    if (!result.elements) result.elements = {};\n    result.elements = {\n      ...result.elements,\n      auto_activate: result.elements.auto_activate || defaults.elements.auto_activate,\n      default_element_dir: result.elements.default_element_dir || defaults.elements.default_element_dir\n    };\n    \n    // Display section\n    if (!result.display) result.display = {};\n    result.display.persona_indicators = {\n      ...defaults.display.persona_indicators,\n      ...result.display.persona_indicators\n    };\n    result.display.verbose_logging = result.display.verbose_logging ?? defaults.display.verbose_logging;\n    result.display.show_progress = result.display.show_progress ?? defaults.display.show_progress;\n    \n    // Wizard section\n    result.wizard = {\n      ...defaults.wizard,\n      ...result.wizard\n    };\n\n    // AutoLoad section (Issue: regression from current-server)\n    // This was missing in mcp-server refactor, causing auto-load to fail for existing configs\n    result.autoLoad = {\n      ...defaults.autoLoad,\n      ...result.autoLoad\n    };\n\n    // Retention Policy section (Issue #51)\n    // IMPORTANT: Defaults are disabled - nothing auto-deleted without explicit consent\n    if (!result.retentionPolicy) result.retentionPolicy = {};\n    result.retentionPolicy = {\n      enabled: result.retentionPolicy.enabled ?? defaults.retentionPolicy.enabled,\n      enforcement_mode: result.retentionPolicy.enforcement_mode ?? defaults.retentionPolicy.enforcement_mode,\n      safety: {\n        ...defaults.retentionPolicy.safety,\n        ...result.retentionPolicy.safety\n      },\n      audit: {\n        ...defaults.retentionPolicy.audit,\n        ...result.retentionPolicy.audit\n      },\n      defaults: {\n        ...defaults.retentionPolicy.defaults,\n        ...result.retentionPolicy.defaults\n      }\n    };\n\n    return result as DollhouseConfig;\n  }\n\n  /**\n   * Migrate settings from environment variables\n   */\n  private async migrateFromEnvironment(): Promise<void> {\n    let migrated = false;\n\n    // NOTE: These are optional custom env vars not in centralized config\n    // They're used for backwards compatibility migration only\n\n    // Migrate user settings\n    if (process.env.DOLLHOUSE_USER && !this.config?.user.username) {\n      if (!this.config) this.config = this.getDefaultConfig();\n      this.config.user.username = process.env.DOLLHOUSE_USER;\n      migrated = true;\n    }\n\n    if (process.env.DOLLHOUSE_EMAIL && !this.config?.user.email) {\n      if (!this.config) this.config = this.getDefaultConfig();\n      this.config.user.email = process.env.DOLLHOUSE_EMAIL;\n      migrated = true;\n    }\n\n    // Migrate portfolio URL\n    if (process.env.DOLLHOUSE_PORTFOLIO_URL && !this.config?.github.portfolio.repository_url) {\n      if (!this.config) this.config = this.getDefaultConfig();\n      this.config.github.portfolio.repository_url = process.env.DOLLHOUSE_PORTFOLIO_URL;\n      migrated = true;\n    }\n\n    // Migrate collection auto-submit - use centralized env config\n    if (env.DOLLHOUSE_AUTO_SUBMIT_TO_COLLECTION !== false) {\n      if (!this.config) this.config = this.getDefaultConfig();\n      this.config.collection.auto_submit = env.DOLLHOUSE_AUTO_SUBMIT_TO_COLLECTION;\n      migrated = true;\n    }\n\n    if (migrated) {\n      logger.info('Migrated settings from environment variables');\n    }\n  }\n\n  /**\n   * Reset configuration to defaults\n   * SECURITY FIX (PR #895): Added prototype pollution protection\n   * Previously: Direct property assignment allowed __proto__ injection\n   * Now: Validates keys against forbidden properties before assignment\n   */\n  public async resetConfig(section?: string): Promise<ConfigActionResult> {\n    const defaults = this.getDefaultConfig();\n    \n    if (section) {\n      // Reset specific section\n      if (!this.config) {\n        this.config = defaults;\n      } else {\n        // SECURITY: Validate section path to prevent prototype pollution\n        validatePropertyPath(section, 'section');\n\n        const sectionKeys = section.split('.');\n        let current: any = this.config;\n        let defaultSection: any = defaults;\n\n        for (let i = 0; i < sectionKeys.length - 1; i++) {\n          current = current[sectionKeys[i]];\n          defaultSection = defaultSection[sectionKeys[i]];\n        }\n\n        const lastKey = sectionKeys[sectionKeys.length - 1];\n        // SECURITY: Use secure property setter to avoid prototype chain pollution\n        safeSetProperty(current, lastKey, defaultSection[lastKey]);\n      }\n      \n      await this.saveConfig();\n      \n      return {\n        success: true,\n        message: `Section '${section}' reset to defaults`\n      };\n    } else {\n      // Reset entire config\n      this.config = defaults;\n      await this.saveConfig();\n      \n      return {\n        success: true,\n        message: 'Configuration reset to defaults'\n      };\n    }\n  }\n\n  /**\n   * Export configuration to file\n   */\n  public async exportConfig(filePath: string): Promise<ConfigActionResult> {\n    if (!this.config) {\n      return {\n        success: false,\n        message: 'No configuration to export'\n      };\n    }\n    \n    try {\n      const yamlContent = yaml.dump(this.config, {\n        indent: 2,\n        lineWidth: 120,\n        noRefs: true,\n        sortKeys: false\n      });\n\n      await this.fileOperations.writeFile(filePath, yamlContent, {\n        source: 'ConfigManager.exportConfig'\n      });\n      await this.fileOperations.chmod(filePath, 0o600, {\n        source: 'ConfigManager.exportConfig'\n      });\n\n      return {\n        success: true,\n        message: `Configuration exported to ${filePath}`\n      };\n    } catch (error) {\n      return {\n        success: false,\n        message: `Failed to export configuration: ${error instanceof Error ? error.message : String(error)}`\n      };\n    }\n  }\n\n  /**\n   * Import configuration from file\n   */\n  public async importConfig(filePath: string): Promise<ConfigActionResult> {\n    try {\n      const content = await this.fileOperations.readFile(filePath, {\n        source: 'ConfigManager.importConfig'\n      });\n      \n      // Parse and validate\n      const parsed = SecureYamlParser.parse(content, {\n        maxYamlSize: 64 * 1024,\n        validateContent: false,\n        validateFields: false\n      });\n      \n      if (!parsed.data || typeof parsed.data !== 'object') {\n        return {\n          success: false,\n          message: 'Invalid configuration format in import file'\n        };\n      }\n      \n      // Merge with defaults\n      this.config = this.mergeWithDefaults(parsed.data as Partial<DollhouseConfig>);\n      \n      // Save the imported config\n      await this.saveConfig();\n      \n      return {\n        success: true,\n        message: `Configuration imported from ${filePath}`\n      };\n    } catch (error) {\n      return {\n        success: false,\n        message: `Failed to import configuration: ${error instanceof Error ? error.message : String(error)}`\n      };\n    }\n  }\n\n  /**\n   * Get formatted config for display\n   */\n  public getFormattedConfig(section?: string): string {\n    if (!this.config) {\n      return 'Configuration not initialized';\n    }\n    \n    let configToShow: any = this.config;\n    \n    if (section) {\n      configToShow = this.getSetting(section);\n      if (!configToShow) {\n        return `Section '${section}' not found`;\n      }\n    }\n    \n    // Remove sensitive data for display\n    const sanitized = JSON.parse(JSON.stringify(configToShow));\n    \n    // Don't show tokens if they exist\n    if (sanitized.github?.auth?.token) {\n      sanitized.github.auth.token = '***REDACTED***';\n    }\n    \n    return yaml.dump(sanitized, {\n      indent: 2,\n      lineWidth: 120,\n      noRefs: true,\n      sortKeys: false\n    });\n  }\n}\n"]}
897
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ConfigManager.js","sourceRoot":"","sources":["../../src/config/ConfigManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * ConfigManager - Centralized configuration management for DollhouseMCP\n * \n * Features:\n * - YAML-based configuration file\n * - Default values with user overrides\n * - Migration from environment variables\n * - Validation and type safety\n * - Atomic updates with backup\n * - Privacy-first defaults\n * - OAuth client ID storage for MCP client integration\n */\n\nimport * as path from 'path';\nimport * as os from 'os';\nimport * as yaml from 'js-yaml';\nimport { logger } from '../utils/logger.js';\nimport { SecureYamlParser } from '../security/secureYamlParser.js';\nimport {\n  validatePropertyPath,\n  safeSetProperty,\n  createSafeObject,\n  safeHasOwnProperty\n} from '../utils/securityUtils.js';\nimport { env } from './env.js';\nimport { IFileOperationsService } from '../services/FileOperationsService.js';\n\nexport interface UserConfig {\n  username: string | null;\n  email: string | null;\n  display_name: string | null;\n}\n\nexport interface GitHubPortfolioConfig {\n  repository_url: string | null;\n  repository_name: string;\n  default_branch: string;\n  auto_create: boolean;\n}\n\nexport interface GitHubAuthConfig {\n  use_oauth: boolean;\n  token_source: 'environment' | 'oauth' | 'config';\n  client_id?: string; // OAuth client ID for GitHub App\n}\n\nexport interface GitHubConfig {\n  portfolio: GitHubPortfolioConfig;\n  auth: GitHubAuthConfig;\n}\n\nexport interface SyncIndividualConfig {\n  require_confirmation: boolean;\n  show_diff_before_sync: boolean;\n  track_versions: boolean;\n  keep_history: number;\n}\n\nexport interface SyncBulkConfig {\n  upload_enabled: boolean;\n  download_enabled: boolean;\n  require_preview: boolean;\n  respect_local_only: boolean;\n}\n\nexport interface SyncPrivacyConfig {\n  scan_for_secrets: boolean;\n  scan_for_pii: boolean;\n  warn_on_sensitive: boolean;\n  excluded_patterns: string[];\n}\n\nexport interface SyncConfig {\n  enabled: boolean;\n  individual: SyncIndividualConfig;\n  bulk: SyncBulkConfig;\n  privacy: SyncPrivacyConfig;\n}\n\nexport interface CollectionConfig {\n  auto_submit: boolean;\n  require_review: boolean;\n  add_attribution: boolean;\n}\n\nexport interface AutoLoadConfig {\n  enabled: boolean;\n  maxTokenBudget: number;\n  maxSingleMemoryTokens?: number;\n  suppressLargeMemoryWarnings?: boolean;\n  memories: string[];\n}\n\n/**\n * Retention Policy Configuration (Issue #51)\n *\n * Controls automatic deletion of expired memory entries.\n * IMPORTANT: Disabled by default - nothing is auto-deleted without explicit consent.\n *\n * Use cases:\n * - Legal/compliance: Law firms, accountants (7-year retention)\n * - Privacy-focused: Signal-like auto-expiring messages\n * - Storage management: Cleanup of old logs\n * - GDPR: Right-to-be-forgotten implementations\n */\nexport interface RetentionPolicyConfig {\n  /**\n   * Master switch for retention enforcement.\n   * If false, no automatic retention enforcement happens anywhere.\n   * Default: false (nothing auto-deleted)\n   */\n  enabled: boolean;\n\n  /**\n   * When retention enforcement happens:\n   * - 'disabled': Never enforce automatically (only via explicit command)\n   * - 'manual': Only when user explicitly requests enforcement\n   * - 'on_load': Enforce when memory is loaded (CURRENT BEHAVIOR - not recommended)\n   * - 'scheduled': Run on a schedule (future implementation)\n   */\n  enforcement_mode: 'disabled' | 'manual' | 'on_load' | 'scheduled';\n\n  /**\n   * Safety controls to prevent accidental data loss\n   */\n  safety: {\n    /** Require explicit confirmation before any deletion */\n    require_confirmation: boolean;\n    /** Always preview what would be deleted before doing it */\n    dry_run_first: boolean;\n    /** Show warning when entries are approaching expiration */\n    warn_on_expiring: boolean;\n    /** Days before expiration to start warning */\n    warning_threshold_days: number;\n  };\n\n  /**\n   * Audit and logging\n   */\n  audit: {\n    /** Log all retention deletions for audit trail */\n    log_deletions: boolean;\n    /** Keep deleted entries in a backup before permanent removal */\n    backup_before_delete: boolean;\n    /** Days to keep backups of deleted entries */\n    backup_retention_days: number;\n  };\n\n  /**\n   * Default TTL settings when creating new memories\n   * Note: These are defaults only - individual memories can override\n   */\n  defaults: {\n    /** Default TTL in days for new memory entries */\n    ttl_days: number;\n    /** Maximum entries before capacity enforcement */\n    max_entries: number;\n  };\n}\n\nexport interface CapabilityIndexResourcesConfig {\n  advertise_resources: boolean;\n  variants: {\n    summary: boolean;\n    full: boolean;\n    stats: boolean;\n  };\n}\n\nexport interface EnhancedIndexConfig {\n  enabled: boolean;\n  limits: {\n    maxTriggersPerElement: number;\n    maxTriggerLength: number;\n    maxKeywordsToCheck: number;\n  };\n  telemetry: {\n    enabled: boolean;\n    sampleRate: number;\n    metricsInterval: number;\n  };\n  verbPatterns?: {\n    customPrefixes?: string[];\n    customSuffixes?: string[];\n    excludedNouns?: string[];\n  };\n  backgroundAnalysis?: {\n    enabled: boolean;\n    scanInterval: number;\n    maxConcurrentScans: number;\n  };\n  resources?: CapabilityIndexResourcesConfig;\n}\n\nexport interface ElementsConfig {\n  auto_activate: {\n    personas?: string[];\n    skills?: string[];\n    templates?: string[];\n    agents?: string[];\n    memories?: string[];\n    ensembles?: string[];\n  };\n  default_element_dir: string;\n  enhanced_index?: EnhancedIndexConfig;\n}\n\nexport interface DisplayConfig {\n  persona_indicators: {\n    enabled: boolean;\n    style: 'full' | 'minimal' | 'compact' | 'custom';\n    include_emoji: boolean;\n  };\n  verbose_logging: boolean;\n  show_progress: boolean;\n}\n\nexport interface WizardConfig {\n  completed: boolean;      // Wizard was successfully completed\n  dismissed: boolean;      // User chose \"Don't show again\"\n  completedAt?: string;    // ISO timestamp when completed\n  version?: string;        // Wizard version for future updates (deprecated - use lastSeenVersion)\n  lastSeenVersion?: string; // Last version where user saw the wizard\n  skippedSections?: string[]; // Track which sections were skipped\n}\n\nexport type LicenseTier = 'agpl' | 'free-commercial' | 'paid-commercial';\n\nexport interface LicenseConfig {\n  tier: LicenseTier;\n  email?: string;            // Required for commercial tiers\n  attestedAt?: string;       // ISO timestamp of attestation\n  telemetryRequired?: boolean; // true for commercial tiers (license condition)\n  revenueScale?: string;     // Paid commercial: \"$1M–$5M\", \"$5M–$25M\", etc.\n  companyName?: string;      // Paid commercial: required\n  useCase?: string;          // Paid commercial: required\n}\n\n/** Branded type for validated port numbers (1024–65535). */\nexport type PortNumber = number & { readonly __brand: 'PortNumber' };\n\n/** Validate and brand a port number. Returns undefined if invalid. */\nexport function validatePort(value: unknown): PortNumber | undefined {\n  const num = typeof value === 'string' ? Number(value) : value;\n  if (typeof num !== 'number' || !Number.isFinite(num)) return undefined;\n  const port = Math.floor(num);\n  if (port < 1024 || port > 65535) return undefined;\n  return port as PortNumber;\n}\n\nexport interface ConsoleConfig {\n  /**\n   * Web console port (1024–65535). Resolution hierarchy:\n   *   1. --port CLI flag (standalone --web mode only)\n   *   2. This config value (~/.dollhouse/config.yml → console.port)\n   *   3. DOLLHOUSE_WEB_CONSOLE_PORT env var\n   *   4. Default: 41715\n   */\n  port: number;\n}\n\nexport interface SourcePriorityConfigData {\n  order: string[];  // Array of 'local' | 'github' | 'collection'\n  stop_on_first: boolean;\n  check_all_for_updates: boolean;\n  fallback_on_error: boolean;\n}\n\nexport interface DollhouseConfig {\n  version: string;\n  user: UserConfig;\n  github: GitHubConfig;\n  sync: SyncConfig;\n  collection: CollectionConfig;\n  elements: ElementsConfig;\n  display: DisplayConfig;\n  wizard: WizardConfig;\n  autoLoad: AutoLoadConfig;\n  retentionPolicy: RetentionPolicyConfig;\n  license: LicenseConfig;\n  console: ConsoleConfig;\n  source_priority?: SourcePriorityConfigData;\n}\n\nexport interface ConfigUpdateResult {\n  success: boolean;\n  message: string;\n  previousValue?: any;\n  newValue?: any;\n}\n\nexport interface ConfigActionResult {\n  success: boolean;\n  message: string;\n  data?: any;\n}\n\nexport class ConfigManager {\n  private configDir: string;\n  private configPath: string;\n  private backupPath: string;\n  private config: DollhouseConfig | null = null;\n  private readonly fileOperations: IFileOperationsService;\n  private os: typeof os;\n\n  /**\n   * Extract console.port from raw YAML config content without full\n   * ConfigManager initialization. Uses FAILSAFE_SCHEMA for security\n   * (no code execution). Returns undefined if not found or invalid.\n   */\n  static readPortFromYaml(yamlContent: string): number | undefined {\n    try {\n      const parsed = yaml.load(yamlContent, { schema: yaml.FAILSAFE_SCHEMA }) as Record<string, any> | null;\n      const raw = parsed?.console?.port;\n      if (raw === undefined || raw === null) return undefined;\n      const port = Number(raw);\n      return Number.isFinite(port) ? port : undefined;\n    } catch {\n      return undefined;\n    }\n  }\n\n  constructor(fileOperations: IFileOperationsService, osModule: typeof os) {\n    this.fileOperations = fileOperations;\n    this.os = osModule;\n\n    // Initialize paths - use test directory if in test environment\n    // NOTE: Using process.env directly here (not cached env.TEST_CONFIG_DIR)\n    // to allow tests to dynamically override the config directory\n    if (env.NODE_ENV === 'test' && process.env.TEST_CONFIG_DIR) {\n      this.configDir = process.env.TEST_CONFIG_DIR;\n    } else {\n      this.configDir = path.join(this.os.homedir(), '.dollhouse');\n    }\n    this.configPath = path.join(this.configDir, 'config.yml');\n    this.backupPath = path.join(this.configDir, 'config.yml.backup');\n  }\n\n  /**\n   * Get default configuration\n   */\n  private getDefaultConfig(): DollhouseConfig {\n    return {\n      version: '1.0.0',\n      user: {\n        username: null,\n        email: null,\n        display_name: null\n      },\n      github: {\n        portfolio: {\n          repository_url: null,\n          repository_name: env.GITHUB_REPOSITORY || 'dollhouse-portfolio',\n          default_branch: 'main',\n          auto_create: true\n        },\n        auth: {\n          use_oauth: true,\n          token_source: 'environment'\n        }\n      },\n      sync: {\n        enabled: false, // Privacy first - off by default\n        individual: {\n          require_confirmation: true,\n          show_diff_before_sync: true,\n          track_versions: true,\n          keep_history: 10\n        },\n        bulk: {\n          upload_enabled: false, // Requires explicit enablement\n          download_enabled: false,\n          require_preview: true,\n          respect_local_only: true\n        },\n        privacy: {\n          scan_for_secrets: true,\n          scan_for_pii: true,\n          warn_on_sensitive: true,\n          excluded_patterns: [\n            '*.secret',\n            '*-private.*',\n            'credentials/**',\n            'personal/**'\n          ]\n        }\n      },\n      collection: {\n        auto_submit: false, // Never auto-submit\n        require_review: true,\n        add_attribution: true\n      },\n      autoLoad: {\n        enabled: true, // Auto-load baseline memories by default\n        maxTokenBudget: 5000,\n        maxSingleMemoryTokens: undefined,\n        suppressLargeMemoryWarnings: false,\n        memories: [] // Configured via memory files with autoLoad flag\n      },\n      elements: {\n        auto_activate: {},\n        default_element_dir: path.join(os.homedir(), '.dollhouse', 'portfolio'),\n        enhanced_index: {\n          enabled: true,\n          limits: {\n            maxTriggersPerElement: 50,\n            maxTriggerLength: 50,\n            maxKeywordsToCheck: 100\n          },\n          telemetry: {\n            enabled: false,  // Opt-in only\n            sampleRate: 0.1,\n            metricsInterval: 60000\n          },\n          resources: {\n            advertise_resources: false, // Default: safe, disabled\n            variants: {\n              summary: false,  // ~1,254 tokens - Opt-in required\n              full: false,     // ~48,306 tokens - Opt-in required\n              stats: true      // ~50 tokens - Safe by default\n            }\n          }\n        }\n      },\n      display: {\n        persona_indicators: {\n          enabled: true,\n          style: 'minimal',\n          include_emoji: true\n        },\n        verbose_logging: false,\n        show_progress: true\n      },\n      wizard: {\n        completed: false,\n        dismissed: false\n      },\n      license: {\n        tier: 'agpl'\n      },\n      console: {\n        port: 41715\n      },\n      /**\n       * Retention Policy Configuration (Issue #51)\n       * IMPORTANT: Disabled by default - nothing is auto-deleted without explicit consent.\n       * Users must explicitly enable retention enforcement if they want automatic cleanup.\n       */\n      retentionPolicy: {\n        enabled: false,                    // DISABLED by default - no auto-deletion\n        enforcement_mode: 'disabled',       // Only manual enforcement allowed\n        safety: {\n          require_confirmation: true,       // Always require confirmation\n          dry_run_first: true,              // Always preview before deleting\n          warn_on_expiring: true,           // Warn when entries approaching expiration\n          warning_threshold_days: 7         // Warn 7 days before expiration\n        },\n        audit: {\n          log_deletions: true,              // Log all deletions\n          backup_before_delete: true,       // Backup deleted entries\n          backup_retention_days: 30         // Keep backups for 30 days\n        },\n        defaults: {\n          ttl_days: 30,                     // Default 30-day TTL (if enabled)\n          max_entries: 1000                 // Default max entries per memory\n        }\n      }\n    };\n  }\n\n  /**\n   * Initialize configuration\n   */\n  public async initialize(): Promise<void> {\n    // Always reload config from disk if it exists, even if we have defaults in memory\n    // This ensures we pick up any manual edits or saved settings\n    \n    try {\n      // Ensure config directory exists with proper permissions (0o700 = owner only)\n      await this.fileOperations.createDirectory(this.configDir);\n      // Set permissions on the directory\n      await this.fileOperations.chmod(this.configDir, 0o700, {\n        source: 'ConfigManager.initialize'\n      });\n      \n      // Load or create config\n      if (await this.configExists()) {\n        await this.loadConfig();\n      } else {\n        // Create default config\n        this.config = this.getDefaultConfig();\n        \n        // Try to migrate from environment variables\n        await this.migrateFromEnvironment();\n        \n        // Save the config\n        await this.saveConfig();\n        \n        logger.info('Created new configuration file', {\n          path: this.configPath\n        });\n      }\n    } catch (error) {\n      logger.error('Failed to initialize configuration', {\n        error: error instanceof Error ? error.message : String(error)\n      });\n      // Use defaults in memory\n      this.config = this.getDefaultConfig();\n    }\n  }\n\n  /**\n   * Load configuration from file\n   */\n  private async loadConfig(): Promise<void> {\n    try {\n      const content = await this.fileOperations.readFile(this.configPath, {\n        source: 'ConfigManager.loadConfig'\n      });\n      \n      /**\n       * IMPORTANT: Parser Selection for Different File Types\n       * \n       * We use DIFFERENT parsers for different file types:\n       * \n       * 1. js-yaml (used here) - For PURE YAML files:\n       *    - Configuration files (config.yml)\n       *    - Data files without markdown content\n       *    - Any .yml or .yaml file that's just YAML\n       *    Example format:\n       *    ```yaml\n       *    version: 1.0.0\n       *    user:\n       *      username: johndoe\n       *      email: john@example.com\n       *    ```\n       * \n       * 2. SecureYamlParser - For MARKDOWN files with YAML frontmatter:\n       *    - Persona files (*.md in personas/)\n       *    - Skill files (*.md in skills/)\n       *    - Template files (*.md in templates/)\n       *    - Any .md file with frontmatter between --- markers\n       *    Example format:\n       *    ```markdown\n       *    ---\n       *    name: Creative Writer\n       *    description: A creative assistant\n       *    ---\n       *    # Instructions\n       *    You are a creative writer...\n       *    ```\n       * \n       * The config file is PURE YAML, so we use js-yaml directly with FAILSAFE_SCHEMA\n       * for security (prevents code execution via YAML tags).\n       * SECURITY: This is NOT a vulnerability - FAILSAFE_SCHEMA prevents code execution\n       */\n      let loadedData: any;\n      try {\n        // Using yaml with FAILSAFE_SCHEMA is secure - prevents code execution\n        loadedData = yaml.load(content, {\n          schema: yaml.FAILSAFE_SCHEMA // Safe schema prevents code execution\n        });\n      } catch (yamlError) {\n        throw new Error(`Invalid YAML in configuration file: ${yamlError instanceof Error ? yamlError.message : String(yamlError)}`);\n      }\n      \n      if (!loadedData || typeof loadedData !== 'object') {\n        throw new Error('Invalid configuration format');\n      }\n      logger.debug('Loaded config from file', {\n        username: loadedData.user?.username,\n        email: loadedData.user?.email,\n        syncEnabled: loadedData.sync?.enabled\n      });\n      \n      this.config = this.mergeWithDefaults(loadedData);\n      \n      // Fix any string booleans that might have been saved incorrectly\n      this.fixConfigTypes();\n      \n      logger.debug('Configuration loaded successfully', {\n        username: this.config.user.username,\n        syncEnabled: this.config.sync.enabled\n      });\n      \n    } catch (error) {\n      logger.error('Failed to load configuration', {\n        error: error instanceof Error ? error.message : String(error)\n      });\n      throw error;\n    }\n  }\n\n  /**\n   * Check if config file exists\n   */\n  private async configExists(): Promise<boolean> {\n    return this.fileOperations.exists(this.configPath);\n  }\n\n  /**\n   * Get GitHub OAuth client ID\n   * Environment variable takes precedence over config file\n   */\n  public getGitHubClientId(): string | null {\n    // NOTE: DOLLHOUSE_GITHUB_CLIENT_ID is not in centralized env config\n    // as it's an optional feature flag, so we still use process.env here\n    const envClientId = process.env.DOLLHOUSE_GITHUB_CLIENT_ID;\n    if (envClientId) {\n      return envClientId;\n    }\n\n    // Fall back to config file\n    return this.config?.github?.auth?.client_id || null;\n  }\n\n  /**\n   * Set GitHub OAuth client ID in config file\n   */\n  public async setGitHubClientId(clientId: string): Promise<void> {\n    if (!ConfigManager.validateClientId(clientId)) {\n      throw new Error(\n        `Invalid GitHub client ID format. Expected format: Ov23li followed by at least 14 alphanumeric characters (e.g., Ov23liABCDEFGHIJKLMN)`\n      );\n    }\n\n    if (!this.config) {\n      this.config = this.getDefaultConfig();\n    }\n\n    // Ensure github.auth object exists\n    if (!this.config.github) {\n      this.config.github = this.getDefaultConfig().github;\n    }\n    if (!this.config.github.auth) {\n      this.config.github.auth = this.getDefaultConfig().github.auth;\n    }\n\n    this.config.github.auth.client_id = clientId;\n    await this.saveConfig();\n  }\n\n  /**\n   * Get the current configuration\n   */\n  public getConfig(): DollhouseConfig {\n    if (!this.config) {\n      throw new Error('Configuration not initialized');\n    }\n    return this.config;\n  }\n\n  /**\n   * Get a specific setting using dot notation\n   */\n  public getSetting<T>(path: string, defaultValue?: T): T | undefined {\n    if (!this.config) {\n      return defaultValue;\n    }\n    \n    const keys = path.split('.');\n    let value: any = this.config;\n    \n    for (const key of keys) {\n      if (value && typeof value === 'object' && key in value) {\n        value = value[key];\n      } else {\n        return defaultValue;\n      }\n    }\n    \n    return value as T;\n  }\n\n  /**\n   * Update a specific setting using dot notation\n   * SECURITY FIX (PR #895): Added prototype pollution protection\n   * Previously: Direct property assignment allowed __proto__ injection\n   * Now: Validates keys against forbidden properties before assignment\n   */\n  public async updateSetting(path: string, value: any): Promise<ConfigUpdateResult> {\n    if (!this.config) {\n      await this.initialize();\n    }\n    \n    // SECURITY: Validate path to prevent prototype pollution\n    validatePropertyPath(path, 'path');\n\n    // Runtime validation for known typed settings (#1840)\n    if (path === 'console.port') {\n      const validated = validatePort(value);\n      if (!validated) {\n        return {\n          success: false,\n          message: `Invalid port: ${value}. Must be an integer between 1024 and 65535.`,\n        };\n      }\n      value = validated;\n    }\n\n    const keys = path.split('.');\n    let current: any = this.config;\n    const previousValue = this.getSetting(path);\n\n    // Navigate to the parent object using security utilities\n    for (let i = 0; i < keys.length - 1; i++) {\n      const key = keys[i];\n      // SECURITY: Use safe property check and create prototype-less objects\n      if (!safeHasOwnProperty(current, key)) {\n        current[key] = createSafeObject();\n      }\n      current = current[key];\n    }\n\n    // Set the value using secure property setter\n    const lastKey = keys[keys.length - 1];\n    safeSetProperty(current, lastKey, value);\n    \n    // Save the configuration\n    await this.saveConfig();\n    \n    logger.info('Configuration setting updated', {\n      path,\n      previousValue,\n      newValue: value\n    });\n    \n    return {\n      success: true,\n      message: `Setting '${path}' updated successfully`,\n      previousValue,\n      newValue: value\n    };\n  }\n\n  /**\n   * Validate GitHub OAuth client ID format\n   * Client IDs start with \"Ov23li\" followed by at least 14 alphanumeric characters\n   * \n   * @param clientId - The client ID to validate\n   * @returns true if valid, false otherwise\n   * \n   * @example\n   * ConfigManager.validateClientId(\"Ov23liABCDEFGHIJKLMN123456\") // true\n   * ConfigManager.validateClientId(\"invalid\") // false\n   * ConfigManager.validateClientId(\"Ov23li\") // false (too short)\n   * ConfigManager.validateClientId(\"Xv23liABCDEFGHIJKLMN\") // false (wrong prefix)\n   */\n  public static validateClientId(clientId: any): boolean {\n    if (typeof clientId !== 'string' || !clientId) {\n      return false;\n    }\n\n    // GitHub OAuth client IDs follow the pattern: Ov23li[A-Za-z0-9]{14,}\n    const clientIdPattern = /^Ov23li[A-Za-z0-9]{14,}$/;\n    return clientIdPattern.test(clientId);\n  }\n\n  /**\n   * Save configuration to file\n   */\n  private async saveConfig(): Promise<void> {\n    if (!this.config) {\n      throw new Error('No configuration to save');\n    }\n    \n    try {\n      // Create backup of existing config\n      if (await this.configExists()) {\n        await this.fileOperations.copyFile(this.configPath, this.backupPath, {\n          source: 'ConfigManager.saveConfig'\n        });\n      }\n\n      // Convert to YAML\n      // Note: We use js-yaml's dump() for pure YAML output (no frontmatter markers)\n      // This creates a standard YAML file, not a markdown file with frontmatter\n      const yamlContent = yaml.dump(this.config, {\n        indent: 2,\n        lineWidth: 120,\n        noRefs: true,\n        sortKeys: false\n        // Using default schema (not FAILSAFE) for dump to preserve types like booleans\n      });\n\n      // Write atomically with proper permissions (0o600 = owner read/write only)\n      const tempPath = `${this.configPath}.tmp`;\n      await this.fileOperations.writeFile(tempPath, yamlContent, {\n        source: 'ConfigManager.saveConfig'\n      });\n      await this.fileOperations.chmod(tempPath, 0o600, {\n        source: 'ConfigManager.saveConfig'\n      });\n      await this.fileOperations.renameFile(tempPath, this.configPath);\n      \n      logger.debug('Configuration saved successfully');\n      \n      // Log audit event for configuration update\n      logger.debug('Configuration update audit', {\n        event: 'CONFIG_UPDATED',\n        source: 'ConfigManager.saveConfig',\n        timestamp: new Date().toISOString()\n      });\n      \n    } catch (error) {\n      logger.error('Failed to save configuration', {\n        error: error instanceof Error ? error.message : String(error)\n      });\n      \n      // Try to restore backup\n      if (await this.backupExists()) {\n        await this.fileOperations.copyFile(this.backupPath, this.configPath, {\n          source: 'ConfigManager.saveConfig'\n        });\n        logger.info('Restored configuration from backup');\n      }\n      \n      throw error;\n    }\n  }\n\n  /**\n   * Check if backup exists\n   */\n  private async backupExists(): Promise<boolean> {\n    return this.fileOperations.exists(this.backupPath);\n  }\n\n  /**\n   * Fix incorrect types in config (e.g., string booleans, string \"null\")\n   */\n  private fixConfigTypes(): void {\n    if (!this.config) return;\n    \n    // Helper to convert string \"null\" to actual null\n    const fixNull = (value: any): any => {\n      if (value === 'null' || value === 'NULL') return null;\n      return value;\n    };\n    \n    // Helper to convert string booleans to actual booleans\n    const fixBoolean = (value: any): any => {\n      if (typeof value === 'string') {\n        const lower = value.toLowerCase();\n        if (lower === 'true') return true;\n        if (lower === 'false') return false;\n      }\n      return value;\n    };\n    \n    // Fix user fields - handle string \"null\" values\n    if (this.config.user) {\n      this.config.user.username = fixNull(this.config.user.username);\n      this.config.user.email = fixNull(this.config.user.email);\n      this.config.user.display_name = fixNull(this.config.user.display_name);\n    }\n    \n    // Fix sync settings\n    if (this.config.sync) {\n      this.config.sync.enabled = fixBoolean(this.config.sync.enabled);\n      \n      if (this.config.sync.individual) {\n        this.config.sync.individual.require_confirmation = fixBoolean(this.config.sync.individual.require_confirmation);\n        this.config.sync.individual.show_diff_before_sync = fixBoolean(this.config.sync.individual.show_diff_before_sync);\n        this.config.sync.individual.track_versions = fixBoolean(this.config.sync.individual.track_versions);\n      }\n      \n      if (this.config.sync.bulk) {\n        this.config.sync.bulk.upload_enabled = fixBoolean(this.config.sync.bulk.upload_enabled);\n        this.config.sync.bulk.download_enabled = fixBoolean(this.config.sync.bulk.download_enabled);\n        this.config.sync.bulk.require_preview = fixBoolean(this.config.sync.bulk.require_preview);\n        this.config.sync.bulk.respect_local_only = fixBoolean(this.config.sync.bulk.respect_local_only);\n      }\n      \n      if (this.config.sync.privacy) {\n        this.config.sync.privacy.scan_for_secrets = fixBoolean(this.config.sync.privacy.scan_for_secrets);\n        this.config.sync.privacy.scan_for_pii = fixBoolean(this.config.sync.privacy.scan_for_pii);\n        this.config.sync.privacy.warn_on_sensitive = fixBoolean(this.config.sync.privacy.warn_on_sensitive);\n      }\n    }\n    \n    // Fix collection settings\n    if (this.config.collection) {\n      this.config.collection.auto_submit = fixBoolean(this.config.collection.auto_submit);\n      this.config.collection.require_review = fixBoolean(this.config.collection.require_review);\n      this.config.collection.add_attribution = fixBoolean(this.config.collection.add_attribution);\n    }\n    \n    // Fix display settings\n    if (this.config.display) {\n      if (this.config.display.persona_indicators) {\n        this.config.display.persona_indicators.enabled = fixBoolean(this.config.display.persona_indicators.enabled);\n        this.config.display.persona_indicators.include_emoji = fixBoolean(this.config.display.persona_indicators.include_emoji);\n      }\n      this.config.display.verbose_logging = fixBoolean(this.config.display.verbose_logging);\n      this.config.display.show_progress = fixBoolean(this.config.display.show_progress);\n    }\n    \n    // Fix github settings\n    if (this.config.github) {\n      if (this.config.github.portfolio) {\n        this.config.github.portfolio.repository_url = fixNull(this.config.github.portfolio.repository_url);\n        this.config.github.portfolio.auto_create = fixBoolean(this.config.github.portfolio.auto_create);\n      }\n      if (this.config.github.auth) {\n        this.config.github.auth.use_oauth = fixBoolean(this.config.github.auth.use_oauth);\n        // Fix client_id if it's a string \"null\"\n        if (this.config.github.auth.client_id) {\n          this.config.github.auth.client_id = fixNull(this.config.github.auth.client_id) || undefined;\n        }\n      }\n    }\n    \n    // Fix wizard settings\n    if (this.config.wizard) {\n      this.config.wizard.completed = fixBoolean(this.config.wizard.completed);\n      this.config.wizard.dismissed = fixBoolean(this.config.wizard.dismissed);\n    }\n\n    // Fix retention policy settings (Issue #51)\n    if (this.config.retentionPolicy) {\n      this.config.retentionPolicy.enabled = fixBoolean(this.config.retentionPolicy.enabled);\n      if (this.config.retentionPolicy.safety) {\n        this.config.retentionPolicy.safety.require_confirmation = fixBoolean(this.config.retentionPolicy.safety.require_confirmation);\n        this.config.retentionPolicy.safety.dry_run_first = fixBoolean(this.config.retentionPolicy.safety.dry_run_first);\n        this.config.retentionPolicy.safety.warn_on_expiring = fixBoolean(this.config.retentionPolicy.safety.warn_on_expiring);\n      }\n      if (this.config.retentionPolicy.audit) {\n        this.config.retentionPolicy.audit.log_deletions = fixBoolean(this.config.retentionPolicy.audit.log_deletions);\n        this.config.retentionPolicy.audit.backup_before_delete = fixBoolean(this.config.retentionPolicy.audit.backup_before_delete);\n      }\n    }\n  }\n\n  /**\n   * Merge partial config with defaults\n   * \n   * IMPORTANT: This function preserves unknown fields for forward compatibility.\n   * If a future version adds new config fields, older versions won't lose them.\n   */\n  private mergeWithDefaults(partial: Partial<DollhouseConfig>): DollhouseConfig {\n    const defaults = this.getDefaultConfig();\n    \n    // Start with a deep clone of partial to preserve all unknown fields\n    const result: any = JSON.parse(JSON.stringify(partial));\n    \n    // Ensure all required fields exist with defaults\n    result.version = result.version || defaults.version;\n    \n    // User section - preserve unknown fields while ensuring required fields\n    result.user = {\n      ...result.user,\n      username: result.user?.username ?? defaults.user.username,\n      email: result.user?.email ?? defaults.user.email,\n      display_name: result.user?.display_name ?? defaults.user.display_name\n    };\n    \n    // GitHub section - deep merge preserving unknown fields\n    if (!result.github) result.github = {};\n    result.github.portfolio = {\n      ...defaults.github.portfolio,\n      ...result.github.portfolio\n    };\n    result.github.auth = {\n      ...defaults.github.auth,\n      ...result.github.auth\n    };\n    \n    // Sync section - preserve unknown fields at all levels\n    if (!result.sync) result.sync = {};\n    result.sync.enabled = result.sync.enabled ?? defaults.sync.enabled;\n    result.sync.individual = {\n      ...defaults.sync.individual,\n      ...result.sync.individual\n    };\n    result.sync.bulk = {\n      ...defaults.sync.bulk,\n      ...result.sync.bulk\n    };\n    result.sync.privacy = {\n      ...defaults.sync.privacy,\n      ...result.sync.privacy,\n      // Special handling for arrays - use provided or default\n      excluded_patterns: result.sync.privacy?.excluded_patterns || defaults.sync.privacy.excluded_patterns\n    };\n    \n    // Collection section\n    result.collection = {\n      ...defaults.collection,\n      ...result.collection\n    };\n    \n    // Elements section\n    if (!result.elements) result.elements = {};\n    result.elements = {\n      ...result.elements,\n      auto_activate: result.elements.auto_activate || defaults.elements.auto_activate,\n      default_element_dir: result.elements.default_element_dir || defaults.elements.default_element_dir\n    };\n    \n    // Display section\n    if (!result.display) result.display = {};\n    result.display.persona_indicators = {\n      ...defaults.display.persona_indicators,\n      ...result.display.persona_indicators\n    };\n    result.display.verbose_logging = result.display.verbose_logging ?? defaults.display.verbose_logging;\n    result.display.show_progress = result.display.show_progress ?? defaults.display.show_progress;\n    \n    // Wizard section\n    result.wizard = {\n      ...defaults.wizard,\n      ...result.wizard\n    };\n\n    // AutoLoad section (Issue: regression from current-server)\n    // This was missing in mcp-server refactor, causing auto-load to fail for existing configs\n    result.autoLoad = {\n      ...defaults.autoLoad,\n      ...result.autoLoad\n    };\n\n    // Retention Policy section (Issue #51)\n    // IMPORTANT: Defaults are disabled - nothing auto-deleted without explicit consent\n    if (!result.retentionPolicy) result.retentionPolicy = {};\n    result.retentionPolicy = {\n      enabled: result.retentionPolicy.enabled ?? defaults.retentionPolicy.enabled,\n      enforcement_mode: result.retentionPolicy.enforcement_mode ?? defaults.retentionPolicy.enforcement_mode,\n      safety: {\n        ...defaults.retentionPolicy.safety,\n        ...result.retentionPolicy.safety\n      },\n      audit: {\n        ...defaults.retentionPolicy.audit,\n        ...result.retentionPolicy.audit\n      },\n      defaults: {\n        ...defaults.retentionPolicy.defaults,\n        ...result.retentionPolicy.defaults\n      }\n    };\n\n    return result as DollhouseConfig;\n  }\n\n  /**\n   * Migrate settings from environment variables\n   */\n  private async migrateFromEnvironment(): Promise<void> {\n    let migrated = false;\n\n    // NOTE: These are optional custom env vars not in centralized config\n    // They're used for backwards compatibility migration only\n\n    // Migrate user settings\n    if (process.env.DOLLHOUSE_USER && !this.config?.user.username) {\n      if (!this.config) this.config = this.getDefaultConfig();\n      this.config.user.username = process.env.DOLLHOUSE_USER;\n      migrated = true;\n    }\n\n    if (process.env.DOLLHOUSE_EMAIL && !this.config?.user.email) {\n      if (!this.config) this.config = this.getDefaultConfig();\n      this.config.user.email = process.env.DOLLHOUSE_EMAIL;\n      migrated = true;\n    }\n\n    // Migrate portfolio URL\n    if (process.env.DOLLHOUSE_PORTFOLIO_URL && !this.config?.github.portfolio.repository_url) {\n      if (!this.config) this.config = this.getDefaultConfig();\n      this.config.github.portfolio.repository_url = process.env.DOLLHOUSE_PORTFOLIO_URL;\n      migrated = true;\n    }\n\n    // Migrate collection auto-submit - use centralized env config\n    if (env.DOLLHOUSE_AUTO_SUBMIT_TO_COLLECTION !== false) {\n      if (!this.config) this.config = this.getDefaultConfig();\n      this.config.collection.auto_submit = env.DOLLHOUSE_AUTO_SUBMIT_TO_COLLECTION;\n      migrated = true;\n    }\n\n    if (migrated) {\n      logger.info('Migrated settings from environment variables');\n    }\n  }\n\n  /**\n   * Reset configuration to defaults\n   * SECURITY FIX (PR #895): Added prototype pollution protection\n   * Previously: Direct property assignment allowed __proto__ injection\n   * Now: Validates keys against forbidden properties before assignment\n   */\n  public async resetConfig(section?: string): Promise<ConfigActionResult> {\n    const defaults = this.getDefaultConfig();\n    \n    if (section) {\n      // Reset specific section\n      if (!this.config) {\n        this.config = defaults;\n      } else {\n        // SECURITY: Validate section path to prevent prototype pollution\n        validatePropertyPath(section, 'section');\n\n        const sectionKeys = section.split('.');\n        let current: any = this.config;\n        let defaultSection: any = defaults;\n\n        for (let i = 0; i < sectionKeys.length - 1; i++) {\n          current = current[sectionKeys[i]];\n          defaultSection = defaultSection[sectionKeys[i]];\n        }\n\n        const lastKey = sectionKeys[sectionKeys.length - 1];\n        // SECURITY: Use secure property setter to avoid prototype chain pollution\n        safeSetProperty(current, lastKey, defaultSection[lastKey]);\n      }\n      \n      await this.saveConfig();\n      \n      return {\n        success: true,\n        message: `Section '${section}' reset to defaults`\n      };\n    } else {\n      // Reset entire config\n      this.config = defaults;\n      await this.saveConfig();\n      \n      return {\n        success: true,\n        message: 'Configuration reset to defaults'\n      };\n    }\n  }\n\n  /**\n   * Export configuration to file\n   */\n  public async exportConfig(filePath: string): Promise<ConfigActionResult> {\n    if (!this.config) {\n      return {\n        success: false,\n        message: 'No configuration to export'\n      };\n    }\n    \n    try {\n      const yamlContent = yaml.dump(this.config, {\n        indent: 2,\n        lineWidth: 120,\n        noRefs: true,\n        sortKeys: false\n      });\n\n      await this.fileOperations.writeFile(filePath, yamlContent, {\n        source: 'ConfigManager.exportConfig'\n      });\n      await this.fileOperations.chmod(filePath, 0o600, {\n        source: 'ConfigManager.exportConfig'\n      });\n\n      return {\n        success: true,\n        message: `Configuration exported to ${filePath}`\n      };\n    } catch (error) {\n      return {\n        success: false,\n        message: `Failed to export configuration: ${error instanceof Error ? error.message : String(error)}`\n      };\n    }\n  }\n\n  /**\n   * Import configuration from file\n   */\n  public async importConfig(filePath: string): Promise<ConfigActionResult> {\n    try {\n      const content = await this.fileOperations.readFile(filePath, {\n        source: 'ConfigManager.importConfig'\n      });\n      \n      // Parse and validate\n      const parsed = SecureYamlParser.parse(content, {\n        maxYamlSize: 64 * 1024,\n        validateContent: false,\n        validateFields: false\n      });\n      \n      if (!parsed.data || typeof parsed.data !== 'object') {\n        return {\n          success: false,\n          message: 'Invalid configuration format in import file'\n        };\n      }\n      \n      // Merge with defaults\n      this.config = this.mergeWithDefaults(parsed.data as Partial<DollhouseConfig>);\n      \n      // Save the imported config\n      await this.saveConfig();\n      \n      return {\n        success: true,\n        message: `Configuration imported from ${filePath}`\n      };\n    } catch (error) {\n      return {\n        success: false,\n        message: `Failed to import configuration: ${error instanceof Error ? error.message : String(error)}`\n      };\n    }\n  }\n\n  /**\n   * Get formatted config for display\n   */\n  public getFormattedConfig(section?: string): string {\n    if (!this.config) {\n      return 'Configuration not initialized';\n    }\n    \n    let configToShow: any = this.config;\n    \n    if (section) {\n      configToShow = this.getSetting(section);\n      if (!configToShow) {\n        return `Section '${section}' not found`;\n      }\n    }\n    \n    // Remove sensitive data for display\n    const sanitized = JSON.parse(JSON.stringify(configToShow));\n    \n    // Don't show tokens if they exist\n    if (sanitized.github?.auth?.token) {\n      sanitized.github.auth.token = '***REDACTED***';\n    }\n    \n    return yaml.dump(sanitized, {\n      indent: 2,\n      lineWidth: 120,\n      noRefs: true,\n      sortKeys: false\n    });\n  }\n}\n"]}