@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.
- package/CHANGELOG.md +22 -0
- package/dist/auth/GitHubAuthManager.js +2 -2
- package/dist/config/ConfigManager.d.ts +158 -25
- package/dist/config/ConfigManager.d.ts.map +1 -1
- package/dist/config/ConfigManager.js +627 -88
- package/dist/generated/version.d.ts +2 -2
- package/dist/generated/version.js +3 -3
- package/dist/handlers/ConfigHandler.d.ts +32 -0
- package/dist/handlers/ConfigHandler.d.ts.map +1 -0
- package/dist/handlers/ConfigHandler.js +202 -0
- package/dist/handlers/SyncHandlerV2.d.ts +42 -0
- package/dist/handlers/SyncHandlerV2.d.ts.map +1 -0
- package/dist/handlers/SyncHandlerV2.js +231 -0
- package/dist/index.d.ts +18 -0
- package/dist/index.d.ts.map +1 -1
- package/dist/index.js +19 -3
- package/dist/portfolio/GitHubPortfolioIndexer.d.ts +0 -1
- package/dist/portfolio/GitHubPortfolioIndexer.d.ts.map +1 -1
- package/dist/portfolio/GitHubPortfolioIndexer.js +36 -16
- package/dist/portfolio/PortfolioRepoManager.d.ts +2 -1
- package/dist/portfolio/PortfolioRepoManager.d.ts.map +1 -1
- package/dist/portfolio/PortfolioRepoManager.js +2 -1
- package/dist/portfolio/PortfolioSyncManager.d.ts +127 -0
- package/dist/portfolio/PortfolioSyncManager.d.ts.map +1 -0
- package/dist/portfolio/PortfolioSyncManager.js +818 -0
- package/dist/security/audit/config/suppressions.d.ts.map +1 -1
- package/dist/security/audit/config/suppressions.js +54 -2
- package/dist/security/secureYamlParser.d.ts +46 -2
- package/dist/security/secureYamlParser.d.ts.map +1 -1
- package/dist/security/secureYamlParser.js +47 -3
- package/dist/server/ServerSetup.d.ts.map +1 -1
- package/dist/server/ServerSetup.js +16 -10
- package/dist/server/tools/ConfigToolsV2.d.ts +10 -0
- package/dist/server/tools/ConfigToolsV2.d.ts.map +1 -0
- package/dist/server/tools/ConfigToolsV2.js +110 -0
- package/dist/server/types.d.ts +2 -0
- package/dist/server/types.d.ts.map +1 -1
- package/dist/server/types.js +1 -1
- package/dist/utils/logger.d.ts +45 -0
- package/dist/utils/logger.d.ts.map +1 -1
- package/dist/utils/logger.js +202 -9
- 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.
|
|
6
|
-
export declare const BUILD_TIMESTAMP = "2025-
|
|
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.
|
|
6
|
-
export const BUILD_TIMESTAMP = '2025-
|
|
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,
|
|
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
|