@dollhousemcp/mcp-server 1.7.1 → 1.7.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (42) hide show
  1. package/CHANGELOG.md +22 -0
  2. package/dist/auth/GitHubAuthManager.js +2 -2
  3. package/dist/config/ConfigManager.d.ts +158 -25
  4. package/dist/config/ConfigManager.d.ts.map +1 -1
  5. package/dist/config/ConfigManager.js +627 -88
  6. package/dist/generated/version.d.ts +2 -2
  7. package/dist/generated/version.js +3 -3
  8. package/dist/handlers/ConfigHandler.d.ts +32 -0
  9. package/dist/handlers/ConfigHandler.d.ts.map +1 -0
  10. package/dist/handlers/ConfigHandler.js +202 -0
  11. package/dist/handlers/SyncHandlerV2.d.ts +42 -0
  12. package/dist/handlers/SyncHandlerV2.d.ts.map +1 -0
  13. package/dist/handlers/SyncHandlerV2.js +231 -0
  14. package/dist/index.d.ts +18 -0
  15. package/dist/index.d.ts.map +1 -1
  16. package/dist/index.js +19 -3
  17. package/dist/portfolio/GitHubPortfolioIndexer.d.ts +0 -1
  18. package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -1
  19. package/dist/portfolio/GitHubPortfolioIndexer.js +36 -16
  20. package/dist/portfolio/PortfolioRepoManager.d.ts +2 -1
  21. package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -1
  22. package/dist/portfolio/PortfolioRepoManager.js +2 -1
  23. package/dist/portfolio/PortfolioSyncManager.d.ts +127 -0
  24. package/dist/portfolio/PortfolioSyncManager.d.ts.map +1 -0
  25. package/dist/portfolio/PortfolioSyncManager.js +818 -0
  26. package/dist/security/audit/config/suppressions.d.ts.map +1 -1
  27. package/dist/security/audit/config/suppressions.js +54 -2
  28. package/dist/security/secureYamlParser.d.ts +46 -2
  29. package/dist/security/secureYamlParser.d.ts.map +1 -1
  30. package/dist/security/secureYamlParser.js +47 -3
  31. package/dist/server/ServerSetup.d.ts.map +1 -1
  32. package/dist/server/ServerSetup.js +16 -10
  33. package/dist/server/tools/ConfigToolsV2.d.ts +10 -0
  34. package/dist/server/tools/ConfigToolsV2.d.ts.map +1 -0
  35. package/dist/server/tools/ConfigToolsV2.js +110 -0
  36. package/dist/server/types.d.ts +2 -0
  37. package/dist/server/types.d.ts.map +1 -1
  38. package/dist/server/types.js +1 -1
  39. package/dist/utils/logger.d.ts +45 -0
  40. package/dist/utils/logger.d.ts.map +1 -1
  41. package/dist/utils/logger.js +202 -9
  42. package/package.json +1 -1
@@ -2,8 +2,8 @@
2
2
  * Auto-generated file - DO NOT EDIT
3
3
  * Generated at build time by scripts/generate-version.js
4
4
  */
5
- export declare const PACKAGE_VERSION = "1.7.1";
6
- export declare const BUILD_TIMESTAMP = "2025-08-31T23:09:37.090Z";
5
+ export declare const PACKAGE_VERSION = "1.7.3";
6
+ export declare const BUILD_TIMESTAMP = "2025-09-09T19:23:32.095Z";
7
7
  export declare const BUILD_TYPE: 'npm' | 'git';
8
8
  export declare const PACKAGE_NAME = "@dollhousemcp/mcp-server";
9
9
  //# sourceMappingURL=version.d.ts.map
@@ -2,8 +2,8 @@
2
2
  * Auto-generated file - DO NOT EDIT
3
3
  * Generated at build time by scripts/generate-version.js
4
4
  */
5
- export const PACKAGE_VERSION = '1.7.1';
6
- export const BUILD_TIMESTAMP = '2025-08-31T23:09:37.090Z';
5
+ export const PACKAGE_VERSION = '1.7.3';
6
+ export const BUILD_TIMESTAMP = '2025-09-09T19:23:32.095Z';
7
7
  export const BUILD_TYPE = 'npm';
8
8
  export const PACKAGE_NAME = '@dollhousemcp/mcp-server';
9
- //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsT0FBTyxDQUFDO0FBQ3ZDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcxLjcuMSc7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjUtMDgtMzFUMjM6MDk6MzcuMDkwWic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
9
+ //# sourceMappingURL=data:application/json;base64,eyJ2ZXJzaW9uIjozLCJmaWxlIjoidmVyc2lvbi5qcyIsInNvdXJjZVJvb3QiOiIiLCJzb3VyY2VzIjpbIi4uLy4uL3NyYy9nZW5lcmF0ZWQvdmVyc2lvbi50cyJdLCJuYW1lcyI6W10sIm1hcHBpbmdzIjoiQUFBQTs7O0dBR0c7QUFFSCxNQUFNLENBQUMsTUFBTSxlQUFlLEdBQUcsT0FBTyxDQUFDO0FBQ3ZDLE1BQU0sQ0FBQyxNQUFNLGVBQWUsR0FBRywwQkFBMEIsQ0FBQztBQUMxRCxNQUFNLENBQUMsTUFBTSxVQUFVLEdBQWtCLEtBQUssQ0FBQztBQUMvQyxNQUFNLENBQUMsTUFBTSxZQUFZLEdBQUcsMEJBQTBCLENBQUMiLCJzb3VyY2VzQ29udGVudCI6WyIvKipcbiAqIEF1dG8tZ2VuZXJhdGVkIGZpbGUgLSBETyBOT1QgRURJVFxuICogR2VuZXJhdGVkIGF0IGJ1aWxkIHRpbWUgYnkgc2NyaXB0cy9nZW5lcmF0ZS12ZXJzaW9uLmpzXG4gKi9cblxuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfVkVSU0lPTiA9ICcxLjcuMyc7XG5leHBvcnQgY29uc3QgQlVJTERfVElNRVNUQU1QID0gJzIwMjUtMDktMDlUMTk6MjM6MzIuMDk1Wic7XG5leHBvcnQgY29uc3QgQlVJTERfVFlQRTogJ25wbScgfCAnZ2l0JyA9ICducG0nO1xuZXhwb3J0IGNvbnN0IFBBQ0tBR0VfTkFNRSA9ICdAZG9sbGhvdXNlbWNwL21jcC1zZXJ2ZXInO1xuIl19
@@ -0,0 +1,32 @@
1
+ /**
2
+ * Configuration handler for the dollhouse_config MCP tool
3
+ * Provides unified interface for all configuration management operations
4
+ */
5
+ export interface ConfigOperationOptions {
6
+ action: 'get' | 'set' | 'reset' | 'export' | 'import' | 'wizard';
7
+ setting?: string;
8
+ value?: any;
9
+ section?: string;
10
+ format?: 'yaml' | 'json';
11
+ data?: string;
12
+ }
13
+ export declare class ConfigHandler {
14
+ private configManager;
15
+ constructor();
16
+ /**
17
+ * Handle configuration operations via the dollhouse_config tool
18
+ */
19
+ handleConfigOperation(options: ConfigOperationOptions, indicator?: string): Promise<{
20
+ content: {
21
+ type: string;
22
+ text: string;
23
+ }[];
24
+ }>;
25
+ private handleGet;
26
+ private handleSet;
27
+ private handleReset;
28
+ private handleExport;
29
+ private handleImport;
30
+ private handleWizard;
31
+ }
32
+ //# sourceMappingURL=ConfigHandler.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"ConfigHandler.d.ts","sourceRoot":"","sources":["../../src/handlers/ConfigHandler.ts"],"names":[],"mappings":"AAAA;;;GAGG;AAMH,MAAM,WAAW,sBAAsB;IACrC,MAAM,EAAE,KAAK,GAAG,KAAK,GAAG,OAAO,GAAG,QAAQ,GAAG,QAAQ,GAAG,QAAQ,CAAC;IACjE,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,KAAK,CAAC,EAAE,GAAG,CAAC;IACZ,OAAO,CAAC,EAAE,MAAM,CAAC;IACjB,MAAM,CAAC,EAAE,MAAM,GAAG,MAAM,CAAC;IACzB,IAAI,CAAC,EAAE,MAAM,CAAC;CACf;AAED,qBAAa,aAAa;IACxB,OAAO,CAAC,aAAa,CAAgB;;IAMrC;;OAEG;IACG,qBAAqB,CAAC,OAAO,EAAE,sBAAsB,EAAE,SAAS,GAAE,MAAW;;;;;;YA4CrE,SAAS;YAkCT,SAAS;YA2CT,WAAW;YAyBX,YAAY;YAeZ,YAAY;IAuB1B,OAAO,CAAC,YAAY;CAsBrB"}
@@ -0,0 +1,202 @@
1
+ /**
2
+ * Configuration handler for the dollhouse_config MCP tool
3
+ * Provides unified interface for all configuration management operations
4
+ */
5
+ import { ConfigManager } from '../config/ConfigManager.js';
6
+ import { SecureErrorHandler } from '../security/errorHandler.js';
7
+ import * as yaml from 'js-yaml';
8
+ export class ConfigHandler {
9
+ configManager;
10
+ constructor() {
11
+ this.configManager = ConfigManager.getInstance();
12
+ }
13
+ /**
14
+ * Handle configuration operations via the dollhouse_config tool
15
+ */
16
+ async handleConfigOperation(options, indicator = '') {
17
+ try {
18
+ await this.configManager.initialize();
19
+ switch (options.action) {
20
+ case 'get':
21
+ return this.handleGet(options, indicator);
22
+ case 'set':
23
+ return this.handleSet(options, indicator);
24
+ case 'reset':
25
+ return this.handleReset(options, indicator);
26
+ case 'export':
27
+ return this.handleExport(options, indicator);
28
+ case 'import':
29
+ return this.handleImport(options, indicator);
30
+ case 'wizard':
31
+ return this.handleWizard(indicator);
32
+ default:
33
+ return {
34
+ content: [{
35
+ type: "text",
36
+ text: `${indicator}❌ Invalid action '${options.action}'.\n\n` +
37
+ `Valid actions: get, set, reset, export, import, wizard`
38
+ }]
39
+ };
40
+ }
41
+ }
42
+ catch (error) {
43
+ const sanitizedError = SecureErrorHandler.sanitizeError(error);
44
+ return {
45
+ content: [{
46
+ type: "text",
47
+ text: `${indicator}❌ Configuration operation failed: ${sanitizedError.message}`
48
+ }]
49
+ };
50
+ }
51
+ }
52
+ async handleGet(options, indicator) {
53
+ // Get a specific setting or all settings
54
+ if (options.setting) {
55
+ const value = this.configManager.getSetting(options.setting);
56
+ if (value === undefined) {
57
+ return {
58
+ content: [{
59
+ type: "text",
60
+ text: `${indicator}❌ Setting '${options.setting}' not found.\n\n` +
61
+ `Use \`dollhouse_config action: "get"\` to see all available settings.`
62
+ }]
63
+ };
64
+ }
65
+ return {
66
+ content: [{
67
+ type: "text",
68
+ text: `${indicator}⚙️ **Configuration Setting**\n\n` +
69
+ `**${options.setting}**: ${JSON.stringify(value, null, 2)}`
70
+ }]
71
+ };
72
+ }
73
+ // Get all settings
74
+ const config = this.configManager.getConfig();
75
+ return {
76
+ content: [{
77
+ type: "text",
78
+ text: `${indicator}⚙️ **DollhouseMCP Configuration**\n\n` +
79
+ `\`\`\`yaml\n${yaml.dump(config, { lineWidth: -1 })}\`\`\``
80
+ }]
81
+ };
82
+ }
83
+ async handleSet(options, indicator) {
84
+ // Set a configuration value
85
+ if (!options.setting || options.value === undefined) {
86
+ return {
87
+ content: [{
88
+ type: "text",
89
+ text: `${indicator}❌ Both 'setting' and 'value' are required for set operation.\n\n` +
90
+ `Example: \`dollhouse_config action: "set", setting: "sync.enabled", value: true\``
91
+ }]
92
+ };
93
+ }
94
+ // Type coercion for common string-to-type conversions
95
+ let coercedValue = options.value;
96
+ // Convert string booleans to actual booleans
97
+ if (typeof coercedValue === 'string') {
98
+ const lowerValue = coercedValue.toLowerCase();
99
+ if (lowerValue === 'true') {
100
+ coercedValue = true;
101
+ }
102
+ else if (lowerValue === 'false') {
103
+ coercedValue = false;
104
+ }
105
+ else if (/^\d+$/.test(coercedValue)) {
106
+ // Convert numeric strings to numbers
107
+ const numValue = parseInt(coercedValue, 10);
108
+ if (!isNaN(numValue)) {
109
+ coercedValue = numValue;
110
+ }
111
+ }
112
+ }
113
+ await this.configManager.updateSetting(options.setting, coercedValue);
114
+ return {
115
+ content: [{
116
+ type: "text",
117
+ text: `${indicator}✅ **Configuration Updated**\n\n` +
118
+ `**${options.setting}** set to: ${JSON.stringify(coercedValue, null, 2)}\n\n` +
119
+ `Changes have been saved to the configuration file.`
120
+ }]
121
+ };
122
+ }
123
+ async handleReset(options, indicator) {
124
+ // Reset configuration to defaults
125
+ if (options.section) {
126
+ await this.configManager.resetConfig(options.section);
127
+ return {
128
+ content: [{
129
+ type: "text",
130
+ text: `${indicator}🔄 **Configuration Reset**\n\n` +
131
+ `Section '${options.section}' has been reset to default values.`
132
+ }]
133
+ };
134
+ }
135
+ // Reset all configuration
136
+ await this.configManager.resetConfig();
137
+ return {
138
+ content: [{
139
+ type: "text",
140
+ text: `${indicator}🔄 **Configuration Reset**\n\n` +
141
+ `All settings have been reset to default values.\n\n` +
142
+ `Note: User identity and GitHub authentication are preserved.`
143
+ }]
144
+ };
145
+ }
146
+ async handleExport(options, indicator) {
147
+ // Export configuration
148
+ const format = options.format || 'yaml';
149
+ const exported = await this.configManager.exportConfig(format);
150
+ return {
151
+ content: [{
152
+ type: "text",
153
+ text: `${indicator}📤 **Configuration Export**\n\n` +
154
+ `\`\`\`${format}\n${exported}\`\`\`\n\n` +
155
+ `You can save this configuration and import it later.`
156
+ }]
157
+ };
158
+ }
159
+ async handleImport(options, indicator) {
160
+ // Import configuration
161
+ if (!options.data) {
162
+ return {
163
+ content: [{
164
+ type: "text",
165
+ text: `${indicator}❌ Configuration data is required for import.\n\n` +
166
+ `Provide YAML or JSON configuration in the 'data' parameter.`
167
+ }]
168
+ };
169
+ }
170
+ await this.configManager.importConfig(options.data);
171
+ return {
172
+ content: [{
173
+ type: "text",
174
+ text: `${indicator}✅ **Configuration Imported**\n\n` +
175
+ `Configuration has been successfully imported and saved.`
176
+ }]
177
+ };
178
+ }
179
+ handleWizard(indicator) {
180
+ // Interactive configuration wizard
181
+ return {
182
+ content: [{
183
+ type: "text",
184
+ text: `${indicator}🧙 **Configuration Wizard**\n\n` +
185
+ `The configuration wizard helps you set up DollhouseMCP.\n\n` +
186
+ `**Key Settings to Configure:**\n\n` +
187
+ `1. **User Identity**\n` +
188
+ ` \`dollhouse_config action: "set", setting: "user.username", value: "your-username"\`\n\n` +
189
+ `2. **GitHub Portfolio**\n` +
190
+ ` \`dollhouse_config action: "set", setting: "github.portfolio.repository_name", value: "dollhouse-portfolio"\`\n\n` +
191
+ `3. **Sync Settings**\n` +
192
+ ` \`dollhouse_config action: "set", setting: "sync.enabled", value: true\`\n` +
193
+ ` \`dollhouse_config action: "set", setting: "sync.bulk.upload_enabled", value: true\`\n\n` +
194
+ `4. **Privacy Settings**\n` +
195
+ ` \`dollhouse_config action: "set", setting: "sync.privacy.scan_for_secrets", value: true\`\n` +
196
+ ` \`dollhouse_config action: "set", setting: "sync.privacy.respect_local_only", value: true\`\n\n` +
197
+ `Run \`dollhouse_config action: "get"\` to see current settings.`
198
+ }]
199
+ };
200
+ }
201
+ }
202
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"ConfigHandler.js","sourceRoot":"","sources":["../../src/handlers/ConfigHandler.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Configuration handler for the dollhouse_config MCP tool\n * Provides unified interface for all configuration management operations\n */\n\nimport { ConfigManager } from '../config/ConfigManager.js';\nimport { SecureErrorHandler } from '../security/errorHandler.js';\nimport * as yaml from 'js-yaml';\n\nexport interface ConfigOperationOptions {\n  action: 'get' | 'set' | 'reset' | 'export' | 'import' | 'wizard';\n  setting?: string;\n  value?: any;\n  section?: string;\n  format?: 'yaml' | 'json';\n  data?: string;\n}\n\nexport class ConfigHandler {\n  private configManager: ConfigManager;\n  \n  constructor() {\n    this.configManager = ConfigManager.getInstance();\n  }\n  \n  /**\n   * Handle configuration operations via the dollhouse_config tool\n   */\n  async handleConfigOperation(options: ConfigOperationOptions, indicator: string = '') {\n    try {\n      await this.configManager.initialize();\n      \n      switch (options.action) {\n        case 'get':\n          return this.handleGet(options, indicator);\n        \n        case 'set':\n          return this.handleSet(options, indicator);\n        \n        case 'reset':\n          return this.handleReset(options, indicator);\n        \n        case 'export':\n          return this.handleExport(options, indicator);\n        \n        case 'import':\n          return this.handleImport(options, indicator);\n        \n        case 'wizard':\n          return this.handleWizard(indicator);\n        \n        default:\n          return {\n            content: [{\n              type: \"text\",\n              text: `${indicator}❌ Invalid action '${options.action}'.\\n\\n` +\n                    `Valid actions: get, set, reset, export, import, wizard`\n            }]\n          };\n      }\n      \n    } catch (error) {\n      const sanitizedError = SecureErrorHandler.sanitizeError(error);\n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}❌ Configuration operation failed: ${sanitizedError.message}`\n        }]\n      };\n    }\n  }\n  \n  private async handleGet(options: ConfigOperationOptions, indicator: string) {\n    // Get a specific setting or all settings\n    if (options.setting) {\n      const value = this.configManager.getSetting(options.setting);\n      if (value === undefined) {\n        return {\n          content: [{\n            type: \"text\",\n            text: `${indicator}❌ Setting '${options.setting}' not found.\\n\\n` +\n                  `Use \\`dollhouse_config action: \"get\"\\` to see all available settings.`\n          }]\n        };\n      }\n      \n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}⚙️ **Configuration Setting**\\n\\n` +\n                `**${options.setting}**: ${JSON.stringify(value, null, 2)}`\n        }]\n      };\n    }\n    \n    // Get all settings\n    const config = this.configManager.getConfig();\n    return {\n      content: [{\n        type: \"text\",\n        text: `${indicator}⚙️ **DollhouseMCP Configuration**\\n\\n` +\n              `\\`\\`\\`yaml\\n${yaml.dump(config, { lineWidth: -1 })}\\`\\`\\``\n      }]\n    };\n  }\n  \n  private async handleSet(options: ConfigOperationOptions, indicator: string) {\n    // Set a configuration value\n    if (!options.setting || options.value === undefined) {\n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}❌ Both 'setting' and 'value' are required for set operation.\\n\\n` +\n                `Example: \\`dollhouse_config action: \"set\", setting: \"sync.enabled\", value: true\\``\n        }]\n      };\n    }\n    \n    // Type coercion for common string-to-type conversions\n    let coercedValue = options.value;\n    \n    // Convert string booleans to actual booleans\n    if (typeof coercedValue === 'string') {\n      const lowerValue = coercedValue.toLowerCase();\n      if (lowerValue === 'true') {\n        coercedValue = true;\n      } else if (lowerValue === 'false') {\n        coercedValue = false;\n      } else if (/^\\d+$/.test(coercedValue)) {\n        // Convert numeric strings to numbers\n        const numValue = parseInt(coercedValue, 10);\n        if (!isNaN(numValue)) {\n          coercedValue = numValue;\n        }\n      }\n    }\n    \n    await this.configManager.updateSetting(options.setting, coercedValue);\n    \n    return {\n      content: [{\n        type: \"text\",\n        text: `${indicator}✅ **Configuration Updated**\\n\\n` +\n              `**${options.setting}** set to: ${JSON.stringify(coercedValue, null, 2)}\\n\\n` +\n              `Changes have been saved to the configuration file.`\n      }]\n    };\n  }\n  \n  private async handleReset(options: ConfigOperationOptions, indicator: string) {\n    // Reset configuration to defaults\n    if (options.section) {\n      await this.configManager.resetConfig(options.section);\n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}🔄 **Configuration Reset**\\n\\n` +\n                `Section '${options.section}' has been reset to default values.`\n        }]\n      };\n    }\n    \n    // Reset all configuration\n    await this.configManager.resetConfig();\n    return {\n      content: [{\n        type: \"text\",\n        text: `${indicator}🔄 **Configuration Reset**\\n\\n` +\n              `All settings have been reset to default values.\\n\\n` +\n              `Note: User identity and GitHub authentication are preserved.`\n      }]\n    };\n  }\n  \n  private async handleExport(options: ConfigOperationOptions, indicator: string) {\n    // Export configuration\n    const format = options.format || 'yaml';\n    const exported = await this.configManager.exportConfig(format);\n    \n    return {\n      content: [{\n        type: \"text\",\n        text: `${indicator}📤 **Configuration Export**\\n\\n` +\n              `\\`\\`\\`${format}\\n${exported}\\`\\`\\`\\n\\n` +\n              `You can save this configuration and import it later.`\n      }]\n    };\n  }\n  \n  private async handleImport(options: ConfigOperationOptions, indicator: string) {\n    // Import configuration\n    if (!options.data) {\n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}❌ Configuration data is required for import.\\n\\n` +\n                `Provide YAML or JSON configuration in the 'data' parameter.`\n        }]\n      };\n    }\n    \n    await this.configManager.importConfig(options.data);\n    \n    return {\n      content: [{\n        type: \"text\",\n        text: `${indicator}✅ **Configuration Imported**\\n\\n` +\n              `Configuration has been successfully imported and saved.`\n      }]\n    };\n  }\n  \n  private handleWizard(indicator: string) {\n    // Interactive configuration wizard\n    return {\n      content: [{\n        type: \"text\",\n        text: `${indicator}🧙 **Configuration Wizard**\\n\\n` +\n              `The configuration wizard helps you set up DollhouseMCP.\\n\\n` +\n              `**Key Settings to Configure:**\\n\\n` +\n              `1. **User Identity**\\n` +\n              `   \\`dollhouse_config action: \"set\", setting: \"user.username\", value: \"your-username\"\\`\\n\\n` +\n              `2. **GitHub Portfolio**\\n` +\n              `   \\`dollhouse_config action: \"set\", setting: \"github.portfolio.repository_name\", value: \"dollhouse-portfolio\"\\`\\n\\n` +\n              `3. **Sync Settings**\\n` +\n              `   \\`dollhouse_config action: \"set\", setting: \"sync.enabled\", value: true\\`\\n` +\n              `   \\`dollhouse_config action: \"set\", setting: \"sync.bulk.upload_enabled\", value: true\\`\\n\\n` +\n              `4. **Privacy Settings**\\n` +\n              `   \\`dollhouse_config action: \"set\", setting: \"sync.privacy.scan_for_secrets\", value: true\\`\\n` +\n              `   \\`dollhouse_config action: \"set\", setting: \"sync.privacy.respect_local_only\", value: true\\`\\n\\n` +\n              `Run \\`dollhouse_config action: \"get\"\\` to see current settings.`\n      }]\n    };\n  }\n}"]}
@@ -0,0 +1,42 @@
1
+ /**
2
+ * Portfolio sync handler for the sync_portfolio MCP tool
3
+ * Manages bi-directional synchronization between local portfolio and GitHub
4
+ * This V2 version works with the actual PortfolioSyncManager implementation
5
+ */
6
+ import { ElementType } from '../portfolio/PortfolioManager.js';
7
+ export interface SyncOperationOptions {
8
+ operation: 'list-remote' | 'download' | 'upload' | 'compare' | 'bulk-download' | 'bulk-upload';
9
+ element_name?: string;
10
+ element_type?: ElementType;
11
+ filter?: {
12
+ type?: ElementType;
13
+ author?: string;
14
+ updated_after?: string;
15
+ };
16
+ options?: {
17
+ force?: boolean;
18
+ dry_run?: boolean;
19
+ include_private?: boolean;
20
+ };
21
+ }
22
+ export declare class SyncHandler {
23
+ private syncManager;
24
+ private configManager;
25
+ constructor();
26
+ /**
27
+ * Handle portfolio sync operations
28
+ */
29
+ handleSyncOperation(options: SyncOperationOptions, indicator?: string): Promise<{
30
+ content: {
31
+ type: string;
32
+ text: string;
33
+ }[];
34
+ }>;
35
+ private mapOperation;
36
+ private formatResult;
37
+ private formatListResult;
38
+ private formatDownloadResult;
39
+ private formatUploadResult;
40
+ private formatCompareResult;
41
+ }
42
+ //# sourceMappingURL=SyncHandlerV2.d.ts.map
@@ -0,0 +1 @@
1
+ {"version":3,"file":"SyncHandlerV2.d.ts","sourceRoot":"","sources":["../../src/handlers/SyncHandlerV2.ts"],"names":[],"mappings":"AAAA;;;;GAIG;AAKH,OAAO,EAAE,WAAW,EAAE,MAAM,kCAAkC,CAAC;AAG/D,MAAM,WAAW,oBAAoB;IACnC,SAAS,EAAE,aAAa,GAAG,UAAU,GAAG,QAAQ,GAAG,SAAS,GAAG,eAAe,GAAG,aAAa,CAAC;IAC/F,YAAY,CAAC,EAAE,MAAM,CAAC;IACtB,YAAY,CAAC,EAAE,WAAW,CAAC;IAC3B,MAAM,CAAC,EAAE;QACP,IAAI,CAAC,EAAE,WAAW,CAAC;QACnB,MAAM,CAAC,EAAE,MAAM,CAAC;QAChB,aAAa,CAAC,EAAE,MAAM,CAAC;KACxB,CAAC;IACF,OAAO,CAAC,EAAE;QACR,KAAK,CAAC,EAAE,OAAO,CAAC;QAChB,OAAO,CAAC,EAAE,OAAO,CAAC;QAClB,eAAe,CAAC,EAAE,OAAO,CAAC;KAC3B,CAAC;CACH;AAED,qBAAa,WAAW;IACtB,OAAO,CAAC,WAAW,CAAuB;IAC1C,OAAO,CAAC,aAAa,CAAgB;;IAOrC;;OAEG;IACG,mBAAmB,CAAC,OAAO,EAAE,oBAAoB,EAAE,SAAS,GAAE,MAAW;;;;;;IAgD/E,OAAO,CAAC,YAAY;IAiBpB,OAAO,CAAC,YAAY;IAmCpB,OAAO,CAAC,gBAAgB;IAgDxB,OAAO,CAAC,oBAAoB;IA2B5B,OAAO,CAAC,kBAAkB;IA2B1B,OAAO,CAAC,mBAAmB;CAmC5B"}
@@ -0,0 +1,231 @@
1
+ /**
2
+ * Portfolio sync handler for the sync_portfolio MCP tool
3
+ * Manages bi-directional synchronization between local portfolio and GitHub
4
+ * This V2 version works with the actual PortfolioSyncManager implementation
5
+ */
6
+ import { PortfolioSyncManager } from '../portfolio/PortfolioSyncManager.js';
7
+ import { ConfigManager } from '../config/ConfigManager.js';
8
+ import { SecureErrorHandler } from '../security/errorHandler.js';
9
+ export class SyncHandler {
10
+ syncManager;
11
+ configManager;
12
+ constructor() {
13
+ this.syncManager = new PortfolioSyncManager();
14
+ this.configManager = ConfigManager.getInstance();
15
+ }
16
+ /**
17
+ * Handle portfolio sync operations
18
+ */
19
+ async handleSyncOperation(options, indicator = '') {
20
+ try {
21
+ await this.configManager.initialize();
22
+ // Check if sync is enabled (allow list-remote and compare even when disabled)
23
+ const syncEnabled = this.configManager.getSetting('sync.enabled');
24
+ const readOnlyOperations = ['list-remote', 'compare'];
25
+ if (!syncEnabled && !readOnlyOperations.includes(options.operation)) {
26
+ return {
27
+ content: [{
28
+ type: "text",
29
+ text: `${indicator}⚠️ **Sync is Disabled**\n\n` +
30
+ `Portfolio sync is currently disabled for privacy.\n\n` +
31
+ `To enable sync:\n` +
32
+ `\`dollhouse_config action: "set", setting: "sync.enabled", value: true\`\n\n` +
33
+ `You can still use \`list-remote\` and \`compare\` to view differences.`
34
+ }]
35
+ };
36
+ }
37
+ // Map our operation to PortfolioSyncManager's SyncOperation format
38
+ const syncOp = {
39
+ operation: this.mapOperation(options.operation),
40
+ element_name: options.element_name,
41
+ element_type: options.element_type || options.filter?.type, // Use filter.type if element_type not provided
42
+ bulk: options.operation.includes('bulk'),
43
+ show_diff: options.operation === 'compare',
44
+ force: options.options?.force,
45
+ confirm: options.options?.force || options.options?.dry_run === false // force implies confirm, dry_run=false means confirm
46
+ };
47
+ // Call the unified handleSyncOperation method
48
+ const result = await this.syncManager.handleSyncOperation(syncOp);
49
+ // Format the result based on the operation type
50
+ return this.formatResult(result, options, indicator);
51
+ }
52
+ catch (error) {
53
+ const sanitizedError = SecureErrorHandler.sanitizeError(error);
54
+ return {
55
+ content: [{
56
+ type: "text",
57
+ text: `${indicator}❌ Sync operation failed: ${sanitizedError.message}`
58
+ }]
59
+ };
60
+ }
61
+ }
62
+ mapOperation(operation) {
63
+ switch (operation) {
64
+ case 'list-remote':
65
+ return 'list-remote';
66
+ case 'download':
67
+ case 'bulk-download':
68
+ return 'download';
69
+ case 'upload':
70
+ case 'bulk-upload':
71
+ return 'upload';
72
+ case 'compare':
73
+ return 'compare';
74
+ default:
75
+ return 'list-remote';
76
+ }
77
+ }
78
+ formatResult(result, options, indicator) {
79
+ if (!result.success) {
80
+ return {
81
+ content: [{
82
+ type: "text",
83
+ text: `${indicator}❌ ${result.message}`
84
+ }]
85
+ };
86
+ }
87
+ switch (options.operation) {
88
+ case 'list-remote':
89
+ return this.formatListResult(result, indicator);
90
+ case 'download':
91
+ case 'bulk-download':
92
+ return this.formatDownloadResult(result, options, indicator);
93
+ case 'upload':
94
+ case 'bulk-upload':
95
+ return this.formatUploadResult(result, options, indicator);
96
+ case 'compare':
97
+ return this.formatCompareResult(result, options, indicator);
98
+ default:
99
+ return {
100
+ content: [{
101
+ type: "text",
102
+ text: `${indicator}✅ ${result.message}`
103
+ }]
104
+ };
105
+ }
106
+ }
107
+ formatListResult(result, indicator) {
108
+ if (!result.elements || result.elements.length === 0) {
109
+ return {
110
+ content: [{
111
+ type: "text",
112
+ text: `${indicator}📋 **GitHub Portfolio is Empty**\n\n` +
113
+ `No elements found in your GitHub portfolio.\n\n` +
114
+ `Upload elements using:\n` +
115
+ `\`sync_portfolio operation: "upload", element_name: "name", element_type: "type"\``
116
+ }]
117
+ };
118
+ }
119
+ let text = `${indicator}📋 **GitHub Portfolio Contents**\n\n`;
120
+ text += `Found ${result.elements.length} elements:\n\n`;
121
+ // Group by type
122
+ const byType = {};
123
+ for (const element of result.elements) {
124
+ if (!byType[element.type]) {
125
+ byType[element.type] = [];
126
+ }
127
+ byType[element.type].push(element);
128
+ }
129
+ for (const [type, elements] of Object.entries(byType)) {
130
+ text += `**${type}** (${elements.length}):\n`;
131
+ for (const element of elements) {
132
+ text += ` • ${element.name}`;
133
+ if (element.remoteVersion) {
134
+ text += ` v${element.remoteVersion}`;
135
+ }
136
+ if (element.status) {
137
+ text += ` (${element.status})`;
138
+ }
139
+ text += '\n';
140
+ }
141
+ text += '\n';
142
+ }
143
+ return {
144
+ content: [{
145
+ type: "text",
146
+ text
147
+ }]
148
+ };
149
+ }
150
+ formatDownloadResult(result, options, indicator) {
151
+ if (options.operation === 'bulk-download') {
152
+ const elements = result.elements || [];
153
+ const downloaded = elements.filter(e => e.action === 'download').length;
154
+ const skipped = elements.filter(e => e.action === 'skip').length;
155
+ return {
156
+ content: [{
157
+ type: "text",
158
+ text: `${indicator}✅ **Bulk Download Complete**\n\n` +
159
+ `Downloaded: ${downloaded} elements\n` +
160
+ `Skipped: ${skipped} elements\n\n` +
161
+ result.message
162
+ }]
163
+ };
164
+ }
165
+ return {
166
+ content: [{
167
+ type: "text",
168
+ text: `${indicator}✅ **Element Downloaded**\n\n` +
169
+ `Element: ${options.element_name} (${options.element_type})\n\n` +
170
+ result.message
171
+ }]
172
+ };
173
+ }
174
+ formatUploadResult(result, options, indicator) {
175
+ if (options.operation === 'bulk-upload') {
176
+ const elements = result.elements || [];
177
+ const uploaded = elements.filter(e => e.action === 'upload').length;
178
+ const skipped = elements.filter(e => e.action === 'skip').length;
179
+ return {
180
+ content: [{
181
+ type: "text",
182
+ text: `${indicator}✅ **Bulk Upload Complete**\n\n` +
183
+ `Uploaded: ${uploaded} elements\n` +
184
+ `Skipped: ${skipped} elements\n\n` +
185
+ result.message
186
+ }]
187
+ };
188
+ }
189
+ return {
190
+ content: [{
191
+ type: "text",
192
+ text: `${indicator}✅ **Element Uploaded**\n\n` +
193
+ `Element: ${options.element_name} (${options.element_type})\n\n` +
194
+ result.message
195
+ }]
196
+ };
197
+ }
198
+ formatCompareResult(result, options, indicator) {
199
+ let text = `${indicator}🔍 **Version Comparison**\n\n`;
200
+ text += `Element: ${options.element_name} (${options.element_type})\n\n`;
201
+ if (result.data) {
202
+ // If we have detailed comparison data
203
+ const data = result.data;
204
+ if (data.local) {
205
+ text += `**Local Version**: ${data.local.version}\n`;
206
+ text += ` Modified: ${new Date(data.local.timestamp).toLocaleString()}\n`;
207
+ }
208
+ else {
209
+ text += `**Local Version**: Not found\n`;
210
+ }
211
+ if (data.remote) {
212
+ text += `\n**Remote Version**: ${data.remote.version}\n`;
213
+ text += ` Modified: ${new Date(data.remote.timestamp).toLocaleString()}\n`;
214
+ }
215
+ else {
216
+ text += `\n**Remote Version**: Not found\n`;
217
+ }
218
+ if (data.diff) {
219
+ text += `\n**Differences**:\n${data.diff}`;
220
+ }
221
+ }
222
+ text += `\n\n${result.message}`;
223
+ return {
224
+ content: [{
225
+ type: "text",
226
+ text
227
+ }]
228
+ };
229
+ }
230
+ }
231
+ //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"SyncHandlerV2.js","sourceRoot":"","sources":["../../src/handlers/SyncHandlerV2.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Portfolio sync handler for the sync_portfolio MCP tool\n * Manages bi-directional synchronization between local portfolio and GitHub\n * This V2 version works with the actual PortfolioSyncManager implementation\n */\n\nimport { PortfolioSyncManager, SyncOperation, SyncResult } from '../portfolio/PortfolioSyncManager.js';\nimport { ConfigManager } from '../config/ConfigManager.js';\nimport { SecureErrorHandler } from '../security/errorHandler.js';\nimport { ElementType } from '../portfolio/PortfolioManager.js';\nimport { logger } from '../utils/logger.js';\n\nexport interface SyncOperationOptions {\n  operation: 'list-remote' | 'download' | 'upload' | 'compare' | 'bulk-download' | 'bulk-upload';\n  element_name?: string;\n  element_type?: ElementType;\n  filter?: {\n    type?: ElementType;\n    author?: string;\n    updated_after?: string;\n  };\n  options?: {\n    force?: boolean;\n    dry_run?: boolean;\n    include_private?: boolean;\n  };\n}\n\nexport class SyncHandler {\n  private syncManager: PortfolioSyncManager;\n  private configManager: ConfigManager;\n  \n  constructor() {\n    this.syncManager = new PortfolioSyncManager();\n    this.configManager = ConfigManager.getInstance();\n  }\n  \n  /**\n   * Handle portfolio sync operations\n   */\n  async handleSyncOperation(options: SyncOperationOptions, indicator: string = '') {\n    try {\n      await this.configManager.initialize();\n      \n      // Check if sync is enabled (allow list-remote and compare even when disabled)\n      const syncEnabled = this.configManager.getSetting('sync.enabled');\n      const readOnlyOperations = ['list-remote', 'compare'];\n      if (!syncEnabled && !readOnlyOperations.includes(options.operation)) {\n        return {\n          content: [{\n            type: \"text\",\n            text: `${indicator}⚠️ **Sync is Disabled**\\n\\n` +\n                  `Portfolio sync is currently disabled for privacy.\\n\\n` +\n                  `To enable sync:\\n` +\n                  `\\`dollhouse_config action: \"set\", setting: \"sync.enabled\", value: true\\`\\n\\n` +\n                  `You can still use \\`list-remote\\` and \\`compare\\` to view differences.`\n          }]\n        };\n      }\n      \n      // Map our operation to PortfolioSyncManager's SyncOperation format\n      const syncOp: SyncOperation = {\n        operation: this.mapOperation(options.operation),\n        element_name: options.element_name,\n        element_type: options.element_type || options.filter?.type, // Use filter.type if element_type not provided\n        bulk: options.operation.includes('bulk'),\n        show_diff: options.operation === 'compare',\n        force: options.options?.force,\n        confirm: options.options?.force || options.options?.dry_run === false // force implies confirm, dry_run=false means confirm\n      };\n      \n      // Call the unified handleSyncOperation method\n      const result = await this.syncManager.handleSyncOperation(syncOp);\n      \n      // Format the result based on the operation type\n      return this.formatResult(result, options, indicator);\n      \n    } catch (error) {\n      const sanitizedError = SecureErrorHandler.sanitizeError(error);\n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}❌ Sync operation failed: ${sanitizedError.message}`\n        }]\n      };\n    }\n  }\n  \n  private mapOperation(operation: string): 'download' | 'upload' | 'compare' | 'list-remote' {\n    switch (operation) {\n      case 'list-remote':\n        return 'list-remote';\n      case 'download':\n      case 'bulk-download':\n        return 'download';\n      case 'upload':\n      case 'bulk-upload':\n        return 'upload';\n      case 'compare':\n        return 'compare';\n      default:\n        return 'list-remote';\n    }\n  }\n  \n  private formatResult(result: SyncResult, options: SyncOperationOptions, indicator: string) {\n    if (!result.success) {\n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}❌ ${result.message}`\n        }]\n      };\n    }\n    \n    switch (options.operation) {\n      case 'list-remote':\n        return this.formatListResult(result, indicator);\n      \n      case 'download':\n      case 'bulk-download':\n        return this.formatDownloadResult(result, options, indicator);\n      \n      case 'upload':\n      case 'bulk-upload':\n        return this.formatUploadResult(result, options, indicator);\n      \n      case 'compare':\n        return this.formatCompareResult(result, options, indicator);\n      \n      default:\n        return {\n          content: [{\n            type: \"text\",\n            text: `${indicator}✅ ${result.message}`\n          }]\n        };\n    }\n  }\n  \n  private formatListResult(result: SyncResult, indicator: string) {\n    if (!result.elements || result.elements.length === 0) {\n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}📋 **GitHub Portfolio is Empty**\\n\\n` +\n                `No elements found in your GitHub portfolio.\\n\\n` +\n                `Upload elements using:\\n` +\n                `\\`sync_portfolio operation: \"upload\", element_name: \"name\", element_type: \"type\"\\``\n        }]\n      };\n    }\n    \n    let text = `${indicator}📋 **GitHub Portfolio Contents**\\n\\n`;\n    text += `Found ${result.elements.length} elements:\\n\\n`;\n    \n    // Group by type\n    const byType: Record<string, any[]> = {};\n    for (const element of result.elements) {\n      if (!byType[element.type]) {\n        byType[element.type] = [];\n      }\n      byType[element.type].push(element);\n    }\n    \n    for (const [type, elements] of Object.entries(byType)) {\n      text += `**${type}** (${elements.length}):\\n`;\n      for (const element of elements) {\n        text += `  • ${element.name}`;\n        if (element.remoteVersion) {\n          text += ` v${element.remoteVersion}`;\n        }\n        if (element.status) {\n          text += ` (${element.status})`;\n        }\n        text += '\\n';\n      }\n      text += '\\n';\n    }\n    \n    return {\n      content: [{\n        type: \"text\",\n        text\n      }]\n    };\n  }\n  \n  private formatDownloadResult(result: SyncResult, options: SyncOperationOptions, indicator: string) {\n    if (options.operation === 'bulk-download') {\n      const elements = result.elements || [];\n      const downloaded = elements.filter(e => e.action === 'download').length;\n      const skipped = elements.filter(e => e.action === 'skip').length;\n      \n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}✅ **Bulk Download Complete**\\n\\n` +\n                `Downloaded: ${downloaded} elements\\n` +\n                `Skipped: ${skipped} elements\\n\\n` +\n                result.message\n        }]\n      };\n    }\n    \n    return {\n      content: [{\n        type: \"text\",\n        text: `${indicator}✅ **Element Downloaded**\\n\\n` +\n              `Element: ${options.element_name} (${options.element_type})\\n\\n` +\n              result.message\n      }]\n    };\n  }\n  \n  private formatUploadResult(result: SyncResult, options: SyncOperationOptions, indicator: string) {\n    if (options.operation === 'bulk-upload') {\n      const elements = result.elements || [];\n      const uploaded = elements.filter(e => e.action === 'upload').length;\n      const skipped = elements.filter(e => e.action === 'skip').length;\n      \n      return {\n        content: [{\n          type: \"text\",\n          text: `${indicator}✅ **Bulk Upload Complete**\\n\\n` +\n                `Uploaded: ${uploaded} elements\\n` +\n                `Skipped: ${skipped} elements\\n\\n` +\n                result.message\n        }]\n      };\n    }\n    \n    return {\n      content: [{\n        type: \"text\",\n        text: `${indicator}✅ **Element Uploaded**\\n\\n` +\n              `Element: ${options.element_name} (${options.element_type})\\n\\n` +\n              result.message\n      }]\n    };\n  }\n  \n  private formatCompareResult(result: SyncResult, options: SyncOperationOptions, indicator: string) {\n    let text = `${indicator}🔍 **Version Comparison**\\n\\n`;\n    text += `Element: ${options.element_name} (${options.element_type})\\n\\n`;\n    \n    if (result.data) {\n      // If we have detailed comparison data\n      const data = result.data;\n      if (data.local) {\n        text += `**Local Version**: ${data.local.version}\\n`;\n        text += `  Modified: ${new Date(data.local.timestamp).toLocaleString()}\\n`;\n      } else {\n        text += `**Local Version**: Not found\\n`;\n      }\n      \n      if (data.remote) {\n        text += `\\n**Remote Version**: ${data.remote.version}\\n`;\n        text += `  Modified: ${new Date(data.remote.timestamp).toLocaleString()}\\n`;\n      } else {\n        text += `\\n**Remote Version**: Not found\\n`;\n      }\n      \n      if (data.diff) {\n        text += `\\n**Differences**:\\n${data.diff}`;\n      }\n    }\n    \n    text += `\\n\\n${result.message}`;\n    \n    return {\n      content: [{\n        type: \"text\",\n        text\n      }]\n    };\n  }\n}"]}
package/dist/index.d.ts CHANGED
@@ -416,6 +416,24 @@ export declare class DollhouseMCPServer implements IToolHandler {
416
416
  text: string;
417
417
  }[];
418
418
  }>;
419
+ /**
420
+ * Handle configuration operations - delegates to ConfigHandler
421
+ */
422
+ handleConfigOperation(options: any): Promise<{
423
+ content: {
424
+ type: string;
425
+ text: string;
426
+ }[];
427
+ }>;
428
+ /**
429
+ * Handle sync operations - delegates to SyncHandler
430
+ */
431
+ handleSyncOperation(options: any): Promise<{
432
+ content: {
433
+ type: string;
434
+ text: string;
435
+ }[];
436
+ }>;
419
437
  /**
420
438
  * Search local portfolio using the metadata index system
421
439
  * This provides fast, comprehensive search across all element types