@dollhousemcp/mcp-server 1.7.2 → 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/README.md.backup +0 -8
- 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/package.json +1 -1
|
@@ -0,0 +1,818 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* PortfolioSyncManager - Handles synchronization between local and GitHub portfolios
|
|
3
|
+
*
|
|
4
|
+
* Features:
|
|
5
|
+
* - Download elements from GitHub portfolio
|
|
6
|
+
* - Upload elements with consent
|
|
7
|
+
* - Version comparison and diff viewing
|
|
8
|
+
* - Privacy-first with explicit permissions
|
|
9
|
+
* - Conflict resolution strategies
|
|
10
|
+
* - Bulk operations with configuration checks
|
|
11
|
+
*/
|
|
12
|
+
import * as fs from 'fs/promises';
|
|
13
|
+
import * as path from 'path';
|
|
14
|
+
import { createHash } from 'crypto';
|
|
15
|
+
import { logger } from '../utils/logger.js';
|
|
16
|
+
import { ConfigManager } from '../config/ConfigManager.js';
|
|
17
|
+
import { PortfolioManager } from './PortfolioManager.js';
|
|
18
|
+
import { PortfolioRepoManager } from './PortfolioRepoManager.js';
|
|
19
|
+
import { GitHubPortfolioIndexer } from './GitHubPortfolioIndexer.js';
|
|
20
|
+
import { TokenManager } from '../security/tokenManager.js';
|
|
21
|
+
import { ContentValidator } from '../security/contentValidator.js';
|
|
22
|
+
import { UnicodeValidator } from '../security/validators/unicodeValidator.js';
|
|
23
|
+
import { SecureYamlParser } from '../security/secureYamlParser.js';
|
|
24
|
+
import { ElementType } from './types.js';
|
|
25
|
+
import { ElementStatus } from '../types/elements/IElement.js';
|
|
26
|
+
export class PortfolioSyncManager {
|
|
27
|
+
configManager;
|
|
28
|
+
portfolioManager;
|
|
29
|
+
repoManager;
|
|
30
|
+
indexer;
|
|
31
|
+
constructor() {
|
|
32
|
+
this.configManager = ConfigManager.getInstance();
|
|
33
|
+
this.portfolioManager = PortfolioManager.getInstance();
|
|
34
|
+
this.repoManager = new PortfolioRepoManager();
|
|
35
|
+
this.indexer = GitHubPortfolioIndexer.getInstance();
|
|
36
|
+
}
|
|
37
|
+
/**
|
|
38
|
+
* Main handler for sync operations
|
|
39
|
+
*/
|
|
40
|
+
async handleSyncOperation(params) {
|
|
41
|
+
try {
|
|
42
|
+
// Check if sync is enabled in config
|
|
43
|
+
const config = this.configManager.getConfig();
|
|
44
|
+
if (!config.sync.enabled && params.operation !== 'list-remote') {
|
|
45
|
+
return {
|
|
46
|
+
success: false,
|
|
47
|
+
message: 'Sync is disabled. Enable it with: dollhouse_config --action update --setting sync.enabled --value true'
|
|
48
|
+
};
|
|
49
|
+
}
|
|
50
|
+
// Check bulk permissions
|
|
51
|
+
if (params.bulk) {
|
|
52
|
+
const bulkAllowed = this.isBulkOperationAllowed(params.operation, config);
|
|
53
|
+
if (!bulkAllowed.allowed) {
|
|
54
|
+
return {
|
|
55
|
+
success: false,
|
|
56
|
+
message: bulkAllowed.message
|
|
57
|
+
};
|
|
58
|
+
}
|
|
59
|
+
}
|
|
60
|
+
// Handle operations
|
|
61
|
+
switch (params.operation) {
|
|
62
|
+
case 'list-remote':
|
|
63
|
+
return await this.listRemoteElements(params.element_type);
|
|
64
|
+
case 'download':
|
|
65
|
+
if (params.bulk) {
|
|
66
|
+
return await this.bulkDownload(params.element_type, params.confirm);
|
|
67
|
+
}
|
|
68
|
+
else if (params.element_name) {
|
|
69
|
+
return await this.downloadElement(params.element_name, params.element_type, params.version, params.force);
|
|
70
|
+
}
|
|
71
|
+
else {
|
|
72
|
+
return {
|
|
73
|
+
success: false,
|
|
74
|
+
message: 'Element name required for individual download'
|
|
75
|
+
};
|
|
76
|
+
}
|
|
77
|
+
case 'upload':
|
|
78
|
+
if (params.bulk) {
|
|
79
|
+
return await this.bulkUpload(params.element_type, params.confirm);
|
|
80
|
+
}
|
|
81
|
+
else if (params.element_name) {
|
|
82
|
+
return await this.uploadElement(params.element_name, params.element_type, params.confirm);
|
|
83
|
+
}
|
|
84
|
+
else {
|
|
85
|
+
return {
|
|
86
|
+
success: false,
|
|
87
|
+
message: 'Element name required for individual upload'
|
|
88
|
+
};
|
|
89
|
+
}
|
|
90
|
+
case 'compare':
|
|
91
|
+
if (params.element_name && params.element_type) {
|
|
92
|
+
return await this.compareVersions(params.element_name, params.element_type, params.show_diff);
|
|
93
|
+
}
|
|
94
|
+
else {
|
|
95
|
+
return {
|
|
96
|
+
success: false,
|
|
97
|
+
message: 'Element name and type required for comparison'
|
|
98
|
+
};
|
|
99
|
+
}
|
|
100
|
+
default:
|
|
101
|
+
return {
|
|
102
|
+
success: false,
|
|
103
|
+
message: `Unknown operation: ${params.operation}`
|
|
104
|
+
};
|
|
105
|
+
}
|
|
106
|
+
}
|
|
107
|
+
catch (error) {
|
|
108
|
+
logger.error('Sync operation failed', {
|
|
109
|
+
operation: params.operation,
|
|
110
|
+
error: error instanceof Error ? error.message : String(error)
|
|
111
|
+
});
|
|
112
|
+
return {
|
|
113
|
+
success: false,
|
|
114
|
+
message: `Sync operation failed: ${error instanceof Error ? error.message : String(error)}`
|
|
115
|
+
};
|
|
116
|
+
}
|
|
117
|
+
}
|
|
118
|
+
/**
|
|
119
|
+
* Check if bulk operation is allowed
|
|
120
|
+
*/
|
|
121
|
+
isBulkOperationAllowed(operation, config) {
|
|
122
|
+
if (operation === 'download' && !config.sync.bulk.download_enabled) {
|
|
123
|
+
return {
|
|
124
|
+
allowed: false,
|
|
125
|
+
message: 'Bulk download is disabled. Enable with: dollhouse_config --action update --setting sync.bulk.download_enabled --value true'
|
|
126
|
+
};
|
|
127
|
+
}
|
|
128
|
+
if (operation === 'upload' && !config.sync.bulk.upload_enabled) {
|
|
129
|
+
return {
|
|
130
|
+
allowed: false,
|
|
131
|
+
message: 'Bulk upload is disabled. Enable with: dollhouse_config --action update --setting sync.bulk.upload_enabled --value true'
|
|
132
|
+
};
|
|
133
|
+
}
|
|
134
|
+
return { allowed: true, message: '' };
|
|
135
|
+
}
|
|
136
|
+
/**
|
|
137
|
+
* List elements available in GitHub portfolio
|
|
138
|
+
*/
|
|
139
|
+
async listRemoteElements(filterType) {
|
|
140
|
+
try {
|
|
141
|
+
// Get GitHub token
|
|
142
|
+
const token = await TokenManager.getGitHubTokenAsync();
|
|
143
|
+
if (!token) {
|
|
144
|
+
return {
|
|
145
|
+
success: false,
|
|
146
|
+
message: 'GitHub authentication required. Use setup_github_auth first.'
|
|
147
|
+
};
|
|
148
|
+
}
|
|
149
|
+
this.repoManager.setToken(token);
|
|
150
|
+
// Get index of GitHub portfolio
|
|
151
|
+
const index = await this.indexer.getIndex();
|
|
152
|
+
if (!index || index.totalElements === 0) {
|
|
153
|
+
return {
|
|
154
|
+
success: true,
|
|
155
|
+
message: 'No elements found in GitHub portfolio',
|
|
156
|
+
elements: []
|
|
157
|
+
};
|
|
158
|
+
}
|
|
159
|
+
// Format elements for display
|
|
160
|
+
const elements = [];
|
|
161
|
+
for (const [type, entries] of index.elements) {
|
|
162
|
+
// Skip if filtering by type and this isn't the requested type
|
|
163
|
+
if (filterType && type !== filterType) {
|
|
164
|
+
continue;
|
|
165
|
+
}
|
|
166
|
+
for (const entry of entries) {
|
|
167
|
+
elements.push({
|
|
168
|
+
name: entry.name,
|
|
169
|
+
type: type,
|
|
170
|
+
remoteVersion: entry.version,
|
|
171
|
+
status: 'unchanged',
|
|
172
|
+
action: 'download'
|
|
173
|
+
});
|
|
174
|
+
}
|
|
175
|
+
}
|
|
176
|
+
return {
|
|
177
|
+
success: true,
|
|
178
|
+
message: `Found ${elements.length} elements in GitHub portfolio`,
|
|
179
|
+
elements
|
|
180
|
+
};
|
|
181
|
+
}
|
|
182
|
+
catch (error) {
|
|
183
|
+
return {
|
|
184
|
+
success: false,
|
|
185
|
+
message: `Failed to list remote elements: ${error instanceof Error ? error.message : String(error)}`
|
|
186
|
+
};
|
|
187
|
+
}
|
|
188
|
+
}
|
|
189
|
+
/**
|
|
190
|
+
* Download a specific element from GitHub
|
|
191
|
+
*/
|
|
192
|
+
async downloadElement(elementName, elementType, version, force) {
|
|
193
|
+
try {
|
|
194
|
+
const config = this.configManager.getConfig();
|
|
195
|
+
// Validate element name
|
|
196
|
+
const validation = UnicodeValidator.normalize(elementName);
|
|
197
|
+
if (!validation.isValid) {
|
|
198
|
+
return {
|
|
199
|
+
success: false,
|
|
200
|
+
message: `Invalid element name: ${validation.detectedIssues?.[0] || 'unknown error'}`
|
|
201
|
+
};
|
|
202
|
+
}
|
|
203
|
+
// Get token and set it
|
|
204
|
+
const token = await TokenManager.getGitHubTokenAsync();
|
|
205
|
+
if (!token) {
|
|
206
|
+
return {
|
|
207
|
+
success: false,
|
|
208
|
+
message: 'GitHub authentication required'
|
|
209
|
+
};
|
|
210
|
+
}
|
|
211
|
+
this.repoManager.setToken(token);
|
|
212
|
+
// Get GitHub index
|
|
213
|
+
const index = await this.indexer.getIndex();
|
|
214
|
+
// Find the element - first try exact match, then fuzzy match
|
|
215
|
+
const entries = index.elements.get(elementType) || [];
|
|
216
|
+
let entry = entries.find(e => e.name === elementName);
|
|
217
|
+
// If exact match not found, try fuzzy matching
|
|
218
|
+
if (!entry) {
|
|
219
|
+
// Try case-insensitive exact match first
|
|
220
|
+
entry = entries.find(e => e.name.toLowerCase() === elementName.toLowerCase());
|
|
221
|
+
// If still not found, try fuzzy matching
|
|
222
|
+
if (!entry) {
|
|
223
|
+
const fuzzyMatch = this.findFuzzyMatch(elementName, entries);
|
|
224
|
+
if (fuzzyMatch) {
|
|
225
|
+
logger.info(`Fuzzy match found: '${elementName}' matched to '${fuzzyMatch.name}'`);
|
|
226
|
+
entry = fuzzyMatch;
|
|
227
|
+
}
|
|
228
|
+
}
|
|
229
|
+
}
|
|
230
|
+
if (!entry) {
|
|
231
|
+
// Generate helpful suggestions
|
|
232
|
+
const suggestions = this.getSuggestions(elementName, entries);
|
|
233
|
+
const suggestionText = suggestions.length > 0
|
|
234
|
+
? `\n\nDid you mean one of these?\n${suggestions.map(s => ` • ${s.name}`).join('\n')}`
|
|
235
|
+
: '';
|
|
236
|
+
return {
|
|
237
|
+
success: false,
|
|
238
|
+
message: `Element '${elementName}' (${elementType}) not found in GitHub portfolio${suggestionText}`
|
|
239
|
+
};
|
|
240
|
+
}
|
|
241
|
+
// Check for local conflicts
|
|
242
|
+
const localPath = this.portfolioManager.getElementPath(elementType, `${elementName}.md`);
|
|
243
|
+
let hasLocalVersion = false;
|
|
244
|
+
let localContent = null;
|
|
245
|
+
try {
|
|
246
|
+
localContent = await fs.readFile(localPath, 'utf-8');
|
|
247
|
+
hasLocalVersion = true;
|
|
248
|
+
}
|
|
249
|
+
catch {
|
|
250
|
+
// No local version exists
|
|
251
|
+
}
|
|
252
|
+
// Download the element
|
|
253
|
+
const response = await fetch(entry.downloadUrl, {
|
|
254
|
+
headers: {
|
|
255
|
+
'Authorization': `Bearer ${token}`,
|
|
256
|
+
'Accept': 'application/vnd.github.v3.raw'
|
|
257
|
+
}
|
|
258
|
+
});
|
|
259
|
+
if (!response.ok) {
|
|
260
|
+
throw new Error(`Failed to download: ${response.statusText}`);
|
|
261
|
+
}
|
|
262
|
+
const remoteContent = await response.text();
|
|
263
|
+
// Validate content security
|
|
264
|
+
const validationResult = ContentValidator.validateAndSanitize(remoteContent);
|
|
265
|
+
if (!validationResult.isValid && validationResult.severity === 'critical') {
|
|
266
|
+
return {
|
|
267
|
+
success: false,
|
|
268
|
+
message: `Security issue detected in remote content: ${validationResult.detectedPatterns?.join(', ')}`
|
|
269
|
+
};
|
|
270
|
+
}
|
|
271
|
+
// Check if content is different
|
|
272
|
+
if (hasLocalVersion && localContent) {
|
|
273
|
+
const localHash = createHash('sha256').update(localContent).digest('hex');
|
|
274
|
+
const remoteHash = createHash('sha256').update(remoteContent).digest('hex');
|
|
275
|
+
if (localHash === remoteHash) {
|
|
276
|
+
return {
|
|
277
|
+
success: true,
|
|
278
|
+
message: `Element '${elementName}' is already up to date`
|
|
279
|
+
};
|
|
280
|
+
}
|
|
281
|
+
// Show confirmation for overwrite unless force flag is set
|
|
282
|
+
if (config.sync.individual.require_confirmation && !force) {
|
|
283
|
+
const diff = await this.generateDiff(localContent, remoteContent);
|
|
284
|
+
return {
|
|
285
|
+
success: false,
|
|
286
|
+
message: `Local version exists. Please confirm download will overwrite:\n\n${diff}\n\nTo proceed, use --force flag`,
|
|
287
|
+
data: { requiresConfirmation: true }
|
|
288
|
+
};
|
|
289
|
+
}
|
|
290
|
+
}
|
|
291
|
+
// Save the element
|
|
292
|
+
await fs.mkdir(path.dirname(localPath), { recursive: true });
|
|
293
|
+
await fs.writeFile(localPath, remoteContent, 'utf-8');
|
|
294
|
+
logger.info('Element downloaded from GitHub', {
|
|
295
|
+
element: elementName,
|
|
296
|
+
type: elementType,
|
|
297
|
+
version: entry.version
|
|
298
|
+
});
|
|
299
|
+
return {
|
|
300
|
+
success: true,
|
|
301
|
+
message: `Successfully downloaded '${elementName}' (${elementType}) from GitHub portfolio`
|
|
302
|
+
};
|
|
303
|
+
}
|
|
304
|
+
catch (error) {
|
|
305
|
+
return {
|
|
306
|
+
success: false,
|
|
307
|
+
message: `Failed to download element: ${error instanceof Error ? error.message : String(error)}`
|
|
308
|
+
};
|
|
309
|
+
}
|
|
310
|
+
}
|
|
311
|
+
/**
|
|
312
|
+
* Upload a specific element to GitHub
|
|
313
|
+
*/
|
|
314
|
+
async uploadElement(elementName, elementType, confirm) {
|
|
315
|
+
try {
|
|
316
|
+
const config = this.configManager.getConfig();
|
|
317
|
+
// Check for local element
|
|
318
|
+
const localPath = this.portfolioManager.getElementPath(elementType, `${elementName}.md`);
|
|
319
|
+
let content;
|
|
320
|
+
try {
|
|
321
|
+
content = await fs.readFile(localPath, 'utf-8');
|
|
322
|
+
}
|
|
323
|
+
catch {
|
|
324
|
+
return {
|
|
325
|
+
success: false,
|
|
326
|
+
message: `Element '${elementName}' (${elementType}) not found locally`
|
|
327
|
+
};
|
|
328
|
+
}
|
|
329
|
+
// Check privacy metadata
|
|
330
|
+
const parsed = SecureYamlParser.parse(content, {
|
|
331
|
+
maxYamlSize: 64 * 1024,
|
|
332
|
+
validateContent: false,
|
|
333
|
+
validateFields: false
|
|
334
|
+
});
|
|
335
|
+
if (parsed.data?.privacy?.local_only === true) {
|
|
336
|
+
return {
|
|
337
|
+
success: false,
|
|
338
|
+
message: `Element '${elementName}' is marked as local-only and cannot be uploaded`
|
|
339
|
+
};
|
|
340
|
+
}
|
|
341
|
+
// Validate content security
|
|
342
|
+
const validationResult = ContentValidator.validateAndSanitize(content);
|
|
343
|
+
if (!validationResult.isValid && validationResult.severity === 'critical') {
|
|
344
|
+
return {
|
|
345
|
+
success: false,
|
|
346
|
+
message: `Security issue detected: ${validationResult.detectedPatterns?.join(', ')}`
|
|
347
|
+
};
|
|
348
|
+
}
|
|
349
|
+
// Scan for sensitive content if configured
|
|
350
|
+
if (config.sync.privacy.scan_for_secrets) {
|
|
351
|
+
logger.debug('Scanning for secrets before upload');
|
|
352
|
+
// Implement actual secret scanning
|
|
353
|
+
const secretPatterns = [
|
|
354
|
+
/api[_-]?key\s*[:=]\s*['"][^'"]+['"]/gi,
|
|
355
|
+
/secret\s*[:=]\s*['"][^'"]+['"]/gi,
|
|
356
|
+
/password\s*[:=]\s*['"][^'"]+['"]/gi,
|
|
357
|
+
/token\s*[:=]\s*['"][^'"]+['"]/gi,
|
|
358
|
+
/private[_-]?key\s*[:=]\s*['"][^'"]+['"]/gi
|
|
359
|
+
];
|
|
360
|
+
for (const pattern of secretPatterns) {
|
|
361
|
+
if (pattern.test(content)) {
|
|
362
|
+
return {
|
|
363
|
+
success: false,
|
|
364
|
+
message: `Potential secret detected in content. Please review and remove sensitive information before uploading.`
|
|
365
|
+
};
|
|
366
|
+
}
|
|
367
|
+
}
|
|
368
|
+
}
|
|
369
|
+
// Get confirmation if required (unless already confirmed)
|
|
370
|
+
if (config.sync.individual.require_confirmation && !confirm) {
|
|
371
|
+
return {
|
|
372
|
+
success: false,
|
|
373
|
+
message: `Please confirm upload of '${elementName}' (${elementType}) to GitHub.\n\nContent preview:\n${content.substring(0, 500)}...\n\nTo proceed, use --confirm flag`,
|
|
374
|
+
data: { requiresConfirmation: true }
|
|
375
|
+
};
|
|
376
|
+
}
|
|
377
|
+
// Get token and validate
|
|
378
|
+
const token = await TokenManager.getGitHubTokenAsync();
|
|
379
|
+
if (!token) {
|
|
380
|
+
return {
|
|
381
|
+
success: false,
|
|
382
|
+
message: 'GitHub authentication required'
|
|
383
|
+
};
|
|
384
|
+
}
|
|
385
|
+
// Create an IElement object for the PortfolioRepoManager
|
|
386
|
+
const element = {
|
|
387
|
+
id: `${elementType}_${elementName}_${Date.now()}`,
|
|
388
|
+
type: elementType,
|
|
389
|
+
version: parsed.data?.version || '1.0.0',
|
|
390
|
+
metadata: {
|
|
391
|
+
name: elementName,
|
|
392
|
+
description: parsed.data?.description || '',
|
|
393
|
+
author: parsed.data?.author || 'unknown',
|
|
394
|
+
created: parsed.data?.created || new Date().toISOString(),
|
|
395
|
+
modified: new Date().toISOString(),
|
|
396
|
+
tags: parsed.data?.tags || [],
|
|
397
|
+
custom: parsed.data
|
|
398
|
+
},
|
|
399
|
+
validate: () => ({ valid: true, errors: [], warnings: [] }),
|
|
400
|
+
serialize: () => content,
|
|
401
|
+
deserialize: () => { },
|
|
402
|
+
getStatus: () => ElementStatus.ACTIVE
|
|
403
|
+
};
|
|
404
|
+
// Use PortfolioRepoManager to upload
|
|
405
|
+
this.repoManager.setToken(token);
|
|
406
|
+
try {
|
|
407
|
+
const url = await this.repoManager.saveElement(element, true); // consent is true since we've already checked
|
|
408
|
+
logger.info('Element uploaded to GitHub', {
|
|
409
|
+
element: elementName,
|
|
410
|
+
type: elementType,
|
|
411
|
+
url
|
|
412
|
+
});
|
|
413
|
+
return {
|
|
414
|
+
success: true,
|
|
415
|
+
message: `Successfully uploaded '${elementName}' (${elementType}) to GitHub portfolio`,
|
|
416
|
+
data: { url }
|
|
417
|
+
};
|
|
418
|
+
}
|
|
419
|
+
catch (uploadError) {
|
|
420
|
+
// Handle specific errors
|
|
421
|
+
if (uploadError instanceof Error && uploadError.message.includes('repository does not exist')) {
|
|
422
|
+
return {
|
|
423
|
+
success: false,
|
|
424
|
+
message: `GitHub portfolio repository not found. Please initialize it first using init_portfolio tool.`
|
|
425
|
+
};
|
|
426
|
+
}
|
|
427
|
+
throw uploadError;
|
|
428
|
+
}
|
|
429
|
+
}
|
|
430
|
+
catch (error) {
|
|
431
|
+
return {
|
|
432
|
+
success: false,
|
|
433
|
+
message: `Failed to upload element: ${error instanceof Error ? error.message : String(error)}`
|
|
434
|
+
};
|
|
435
|
+
}
|
|
436
|
+
}
|
|
437
|
+
/**
|
|
438
|
+
* Compare local and remote versions
|
|
439
|
+
*/
|
|
440
|
+
async compareVersions(elementName, elementType, showDiff) {
|
|
441
|
+
try {
|
|
442
|
+
// Get local version
|
|
443
|
+
const localPath = this.portfolioManager.getElementPath(elementType, `${elementName}.md`);
|
|
444
|
+
let localContent = null;
|
|
445
|
+
let localVersion = null;
|
|
446
|
+
try {
|
|
447
|
+
localContent = await fs.readFile(localPath, 'utf-8');
|
|
448
|
+
const parsed = SecureYamlParser.parse(localContent, {
|
|
449
|
+
maxYamlSize: 64 * 1024,
|
|
450
|
+
validateContent: false,
|
|
451
|
+
validateFields: false
|
|
452
|
+
});
|
|
453
|
+
localVersion = {
|
|
454
|
+
version: parsed.data?.version || '1.0.0',
|
|
455
|
+
timestamp: new Date(parsed.data?.updated || parsed.data?.created || Date.now()),
|
|
456
|
+
author: parsed.data?.author || 'unknown',
|
|
457
|
+
hash: createHash('sha256').update(localContent).digest('hex'),
|
|
458
|
+
size: Buffer.byteLength(localContent),
|
|
459
|
+
source: 'local'
|
|
460
|
+
};
|
|
461
|
+
}
|
|
462
|
+
catch {
|
|
463
|
+
// No local version
|
|
464
|
+
}
|
|
465
|
+
// Get remote version
|
|
466
|
+
const token = await TokenManager.getGitHubTokenAsync();
|
|
467
|
+
if (!token) {
|
|
468
|
+
return {
|
|
469
|
+
success: false,
|
|
470
|
+
message: 'GitHub authentication required'
|
|
471
|
+
};
|
|
472
|
+
}
|
|
473
|
+
const index = await this.indexer.getIndex();
|
|
474
|
+
const entries = index.elements.get(elementType) || [];
|
|
475
|
+
const entry = entries.find(e => e.name === elementName);
|
|
476
|
+
let remoteVersion = null;
|
|
477
|
+
let remoteContent = null;
|
|
478
|
+
if (entry) {
|
|
479
|
+
const response = await fetch(entry.downloadUrl, {
|
|
480
|
+
headers: {
|
|
481
|
+
'Authorization': `Bearer ${token}`,
|
|
482
|
+
'Accept': 'application/vnd.github.v3.raw'
|
|
483
|
+
}
|
|
484
|
+
});
|
|
485
|
+
if (response.ok) {
|
|
486
|
+
remoteContent = await response.text();
|
|
487
|
+
remoteVersion = {
|
|
488
|
+
version: entry.version || '1.0.0',
|
|
489
|
+
timestamp: entry.lastModified,
|
|
490
|
+
author: entry.author || 'unknown',
|
|
491
|
+
hash: createHash('sha256').update(remoteContent).digest('hex'),
|
|
492
|
+
size: entry.size,
|
|
493
|
+
source: 'remote'
|
|
494
|
+
};
|
|
495
|
+
}
|
|
496
|
+
}
|
|
497
|
+
// Build comparison result
|
|
498
|
+
const result = {
|
|
499
|
+
element: elementName,
|
|
500
|
+
type: elementType,
|
|
501
|
+
local: localVersion,
|
|
502
|
+
remote: remoteVersion
|
|
503
|
+
};
|
|
504
|
+
if (localVersion && remoteVersion) {
|
|
505
|
+
result.status = localVersion.hash === remoteVersion.hash ? 'identical' : 'different';
|
|
506
|
+
if (showDiff && localContent && remoteContent && result.status === 'different') {
|
|
507
|
+
result.diff = await this.generateDiff(localContent, remoteContent);
|
|
508
|
+
}
|
|
509
|
+
}
|
|
510
|
+
else if (localVersion && !remoteVersion) {
|
|
511
|
+
result.status = 'local-only';
|
|
512
|
+
}
|
|
513
|
+
else if (!localVersion && remoteVersion) {
|
|
514
|
+
result.status = 'remote-only';
|
|
515
|
+
}
|
|
516
|
+
else {
|
|
517
|
+
result.status = 'not-found';
|
|
518
|
+
}
|
|
519
|
+
return {
|
|
520
|
+
success: true,
|
|
521
|
+
message: `Version comparison for '${elementName}' (${elementType})`,
|
|
522
|
+
data: result
|
|
523
|
+
};
|
|
524
|
+
}
|
|
525
|
+
catch (error) {
|
|
526
|
+
return {
|
|
527
|
+
success: false,
|
|
528
|
+
message: `Failed to compare versions: ${error instanceof Error ? error.message : String(error)}`
|
|
529
|
+
};
|
|
530
|
+
}
|
|
531
|
+
}
|
|
532
|
+
/**
|
|
533
|
+
* Bulk download elements
|
|
534
|
+
*/
|
|
535
|
+
async bulkDownload(elementType, confirm) {
|
|
536
|
+
const config = this.configManager.getConfig();
|
|
537
|
+
if (!config.sync.bulk.download_enabled) {
|
|
538
|
+
return {
|
|
539
|
+
success: false,
|
|
540
|
+
message: 'Bulk download is not enabled in configuration'
|
|
541
|
+
};
|
|
542
|
+
}
|
|
543
|
+
// Get list of remote elements
|
|
544
|
+
const remoteResult = await this.listRemoteElements();
|
|
545
|
+
if (!remoteResult.success || !remoteResult.elements) {
|
|
546
|
+
return remoteResult;
|
|
547
|
+
}
|
|
548
|
+
// Filter by type if specified
|
|
549
|
+
let elementsToDownload = remoteResult.elements;
|
|
550
|
+
if (elementType) {
|
|
551
|
+
elementsToDownload = elementsToDownload.filter(e => e.type === elementType);
|
|
552
|
+
}
|
|
553
|
+
if (elementsToDownload.length === 0) {
|
|
554
|
+
return {
|
|
555
|
+
success: true,
|
|
556
|
+
message: 'No elements to download',
|
|
557
|
+
elements: []
|
|
558
|
+
};
|
|
559
|
+
}
|
|
560
|
+
// Show preview if required (unless already confirmed)
|
|
561
|
+
if (config.sync.bulk.require_preview && !confirm) {
|
|
562
|
+
return {
|
|
563
|
+
success: false,
|
|
564
|
+
message: `Bulk download preview:\n\n${elementsToDownload.length} elements will be downloaded:\n${elementsToDownload.map(e => `- ${e.name} (${e.type})`).join('\n')}\n\nTo proceed, use --confirm flag`,
|
|
565
|
+
data: { requiresConfirmation: true },
|
|
566
|
+
elements: elementsToDownload
|
|
567
|
+
};
|
|
568
|
+
}
|
|
569
|
+
// Perform actual bulk download
|
|
570
|
+
const results = {
|
|
571
|
+
downloaded: [],
|
|
572
|
+
skipped: [],
|
|
573
|
+
failed: []
|
|
574
|
+
};
|
|
575
|
+
for (const element of elementsToDownload) {
|
|
576
|
+
try {
|
|
577
|
+
const result = await this.downloadElement(element.name, element.type, undefined, true); // force=true to skip individual confirmations
|
|
578
|
+
if (result.success) {
|
|
579
|
+
results.downloaded.push(element.name);
|
|
580
|
+
}
|
|
581
|
+
else if (result.message?.includes('already up to date')) {
|
|
582
|
+
results.skipped.push(element.name);
|
|
583
|
+
}
|
|
584
|
+
else {
|
|
585
|
+
results.failed.push({ name: element.name, error: result.message || 'Unknown error' });
|
|
586
|
+
}
|
|
587
|
+
}
|
|
588
|
+
catch (error) {
|
|
589
|
+
results.failed.push({
|
|
590
|
+
name: element.name,
|
|
591
|
+
error: error instanceof Error ? error.message : String(error)
|
|
592
|
+
});
|
|
593
|
+
}
|
|
594
|
+
}
|
|
595
|
+
// Build summary message
|
|
596
|
+
let message = `Bulk download complete:\n`;
|
|
597
|
+
message += `- Downloaded: ${results.downloaded.length} elements\n`;
|
|
598
|
+
message += `- Skipped (up to date): ${results.skipped.length} elements\n`;
|
|
599
|
+
message += `- Failed: ${results.failed.length} elements`;
|
|
600
|
+
if (results.failed.length > 0) {
|
|
601
|
+
message += `\n\nFailed downloads:\n${results.failed.map(f => `- ${f.name}: ${f.error}`).join('\n')}`;
|
|
602
|
+
}
|
|
603
|
+
return {
|
|
604
|
+
success: results.failed.length === 0,
|
|
605
|
+
message,
|
|
606
|
+
data: results
|
|
607
|
+
};
|
|
608
|
+
}
|
|
609
|
+
/**
|
|
610
|
+
* Bulk upload elements
|
|
611
|
+
*/
|
|
612
|
+
async bulkUpload(elementType, confirm) {
|
|
613
|
+
const config = this.configManager.getConfig();
|
|
614
|
+
if (!config.sync.bulk.upload_enabled) {
|
|
615
|
+
return {
|
|
616
|
+
success: false,
|
|
617
|
+
message: 'Bulk upload is not enabled in configuration'
|
|
618
|
+
};
|
|
619
|
+
}
|
|
620
|
+
// Get list of local elements
|
|
621
|
+
const types = elementType ? [elementType] : [
|
|
622
|
+
ElementType.PERSONA,
|
|
623
|
+
ElementType.SKILL,
|
|
624
|
+
ElementType.TEMPLATE,
|
|
625
|
+
ElementType.AGENT,
|
|
626
|
+
ElementType.MEMORY,
|
|
627
|
+
ElementType.ENSEMBLE
|
|
628
|
+
];
|
|
629
|
+
const localElements = [];
|
|
630
|
+
for (const type of types) {
|
|
631
|
+
const dir = this.portfolioManager.getElementDir(type);
|
|
632
|
+
try {
|
|
633
|
+
const files = await fs.readdir(dir);
|
|
634
|
+
for (const file of files) {
|
|
635
|
+
if (file.endsWith('.md')) {
|
|
636
|
+
localElements.push({
|
|
637
|
+
name: file.replace('.md', ''),
|
|
638
|
+
type,
|
|
639
|
+
path: path.join(dir, file)
|
|
640
|
+
});
|
|
641
|
+
}
|
|
642
|
+
}
|
|
643
|
+
}
|
|
644
|
+
catch (error) {
|
|
645
|
+
// Directory may not exist yet
|
|
646
|
+
logger.debug(`Directory for ${type} does not exist yet`);
|
|
647
|
+
}
|
|
648
|
+
}
|
|
649
|
+
if (localElements.length === 0) {
|
|
650
|
+
return {
|
|
651
|
+
success: true,
|
|
652
|
+
message: 'No local elements to upload',
|
|
653
|
+
elements: []
|
|
654
|
+
};
|
|
655
|
+
}
|
|
656
|
+
// Show preview if required (unless already confirmed)
|
|
657
|
+
if (config.sync.bulk.require_preview && !confirm) {
|
|
658
|
+
// Convert to SyncElementInfo format for preview
|
|
659
|
+
const previewElements = localElements.map(e => ({
|
|
660
|
+
name: e.name,
|
|
661
|
+
type: e.type,
|
|
662
|
+
status: 'local-only',
|
|
663
|
+
action: 'upload'
|
|
664
|
+
}));
|
|
665
|
+
return {
|
|
666
|
+
success: false,
|
|
667
|
+
message: `Bulk upload preview:\n\n${localElements.length} elements will be uploaded:\n${localElements.map(e => `- ${e.name} (${e.type})`).join('\n')}\n\nTo proceed, use --confirm flag`,
|
|
668
|
+
data: { requiresConfirmation: true },
|
|
669
|
+
elements: previewElements
|
|
670
|
+
};
|
|
671
|
+
}
|
|
672
|
+
// Perform actual bulk upload
|
|
673
|
+
const results = {
|
|
674
|
+
uploaded: [],
|
|
675
|
+
skipped: [],
|
|
676
|
+
failed: []
|
|
677
|
+
};
|
|
678
|
+
for (const element of localElements) {
|
|
679
|
+
try {
|
|
680
|
+
const result = await this.uploadElement(element.name, element.type, true); // confirm=true to skip individual confirmations
|
|
681
|
+
if (result.success) {
|
|
682
|
+
results.uploaded.push(element.name);
|
|
683
|
+
}
|
|
684
|
+
else if (result.message?.includes('local-only')) {
|
|
685
|
+
results.skipped.push(element.name);
|
|
686
|
+
}
|
|
687
|
+
else {
|
|
688
|
+
results.failed.push({ name: element.name, error: result.message || 'Unknown error' });
|
|
689
|
+
}
|
|
690
|
+
}
|
|
691
|
+
catch (error) {
|
|
692
|
+
results.failed.push({
|
|
693
|
+
name: element.name,
|
|
694
|
+
error: error instanceof Error ? error.message : String(error)
|
|
695
|
+
});
|
|
696
|
+
}
|
|
697
|
+
}
|
|
698
|
+
// Build summary message
|
|
699
|
+
let message = `Bulk upload complete:\n`;
|
|
700
|
+
message += `- Uploaded: ${results.uploaded.length} elements\n`;
|
|
701
|
+
message += `- Skipped (local-only): ${results.skipped.length} elements\n`;
|
|
702
|
+
message += `- Failed: ${results.failed.length} elements`;
|
|
703
|
+
if (results.failed.length > 0) {
|
|
704
|
+
message += `\n\nFailed uploads:\n${results.failed.map(f => `- ${f.name}: ${f.error}`).join('\n')}`;
|
|
705
|
+
}
|
|
706
|
+
return {
|
|
707
|
+
success: results.failed.length === 0,
|
|
708
|
+
message,
|
|
709
|
+
data: results
|
|
710
|
+
};
|
|
711
|
+
}
|
|
712
|
+
/**
|
|
713
|
+
* Generate diff between two content versions
|
|
714
|
+
*/
|
|
715
|
+
async generateDiff(local, remote) {
|
|
716
|
+
// Simple line-based diff for now
|
|
717
|
+
const localLines = local.split('\n');
|
|
718
|
+
const remoteLines = remote.split('\n');
|
|
719
|
+
let diff = '';
|
|
720
|
+
const maxLines = Math.max(localLines.length, remoteLines.length);
|
|
721
|
+
for (let i = 0; i < maxLines && i < 10; i++) { // Show first 10 lines of diff
|
|
722
|
+
const localLine = localLines[i] || '';
|
|
723
|
+
const remoteLine = remoteLines[i] || '';
|
|
724
|
+
if (localLine !== remoteLine) {
|
|
725
|
+
if (localLine && !remoteLine) {
|
|
726
|
+
diff += `- ${localLine}\n`;
|
|
727
|
+
}
|
|
728
|
+
else if (!localLine && remoteLine) {
|
|
729
|
+
diff += `+ ${remoteLine}\n`;
|
|
730
|
+
}
|
|
731
|
+
else {
|
|
732
|
+
diff += `- ${localLine}\n`;
|
|
733
|
+
diff += `+ ${remoteLine}\n`;
|
|
734
|
+
}
|
|
735
|
+
}
|
|
736
|
+
}
|
|
737
|
+
if (maxLines > 10) {
|
|
738
|
+
diff += `\n... ${maxLines - 10} more lines ...`;
|
|
739
|
+
}
|
|
740
|
+
return diff || 'No differences found';
|
|
741
|
+
}
|
|
742
|
+
/**
|
|
743
|
+
* Find a fuzzy match for an element name
|
|
744
|
+
*/
|
|
745
|
+
findFuzzyMatch(searchName, entries) {
|
|
746
|
+
const search = searchName.toLowerCase().replace(/[-_]/g, '');
|
|
747
|
+
let bestMatch = null;
|
|
748
|
+
let bestScore = 0;
|
|
749
|
+
for (const entry of entries) {
|
|
750
|
+
// Normalize the entry name for comparison
|
|
751
|
+
const normalized = entry.name.toLowerCase().replace(/[-_]/g, '');
|
|
752
|
+
// Calculate similarity score
|
|
753
|
+
const score = this.calculateSimilarity(search, normalized);
|
|
754
|
+
if (score > bestScore && score > 0.5) { // Minimum threshold of 0.5
|
|
755
|
+
bestScore = score;
|
|
756
|
+
bestMatch = entry;
|
|
757
|
+
}
|
|
758
|
+
}
|
|
759
|
+
return bestMatch;
|
|
760
|
+
}
|
|
761
|
+
/**
|
|
762
|
+
* Get suggestions for similar element names
|
|
763
|
+
*/
|
|
764
|
+
getSuggestions(searchName, entries) {
|
|
765
|
+
const search = searchName.toLowerCase().replace(/[-_]/g, '');
|
|
766
|
+
const scored = [];
|
|
767
|
+
for (const entry of entries) {
|
|
768
|
+
const normalized = entry.name.toLowerCase().replace(/[-_]/g, '');
|
|
769
|
+
const score = this.calculateSimilarity(search, normalized);
|
|
770
|
+
if (score > 0.3) { // Lower threshold for suggestions
|
|
771
|
+
scored.push({ entry, score });
|
|
772
|
+
}
|
|
773
|
+
}
|
|
774
|
+
// Sort by score and return top 5
|
|
775
|
+
return scored
|
|
776
|
+
.sort((a, b) => b.score - a.score)
|
|
777
|
+
.slice(0, 5)
|
|
778
|
+
.map(s => ({ name: s.entry.name }));
|
|
779
|
+
}
|
|
780
|
+
/**
|
|
781
|
+
* Calculate similarity between two strings
|
|
782
|
+
* Returns a score between 0 and 1
|
|
783
|
+
*/
|
|
784
|
+
calculateSimilarity(a, b) {
|
|
785
|
+
// Exact match
|
|
786
|
+
if (a === b)
|
|
787
|
+
return 1.0;
|
|
788
|
+
// One contains the other
|
|
789
|
+
if (a.includes(b) || b.includes(a))
|
|
790
|
+
return 0.8;
|
|
791
|
+
// Calculate word overlap
|
|
792
|
+
const wordsA = a.split(/[^a-z0-9]+/);
|
|
793
|
+
const wordsB = b.split(/[^a-z0-9]+/);
|
|
794
|
+
let matches = 0;
|
|
795
|
+
for (const wordA of wordsA) {
|
|
796
|
+
if (wordA && wordsB.some(wordB => wordB === wordA)) {
|
|
797
|
+
matches++;
|
|
798
|
+
}
|
|
799
|
+
}
|
|
800
|
+
if (matches > 0) {
|
|
801
|
+
const overlap = (matches * 2) / (wordsA.length + wordsB.length);
|
|
802
|
+
return Math.max(0.6, overlap); // At least 0.6 for any word match
|
|
803
|
+
}
|
|
804
|
+
// Check for partial matches
|
|
805
|
+
for (const wordA of wordsA) {
|
|
806
|
+
for (const wordB of wordsB) {
|
|
807
|
+
if (wordA.length > 3 && wordB.length > 3) {
|
|
808
|
+
if (wordA.includes(wordB) || wordB.includes(wordA)) {
|
|
809
|
+
return 0.5;
|
|
810
|
+
}
|
|
811
|
+
}
|
|
812
|
+
}
|
|
813
|
+
}
|
|
814
|
+
// No significant similarity
|
|
815
|
+
return 0;
|
|
816
|
+
}
|
|
817
|
+
}
|
|
818
|
+
//# sourceMappingURL=data:application/json;base64,{"version":3,"file":"PortfolioSyncManager.js","sourceRoot":"","sources":["../../src/portfolio/PortfolioSyncManager.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * PortfolioSyncManager - Handles synchronization between local and GitHub portfolios\n * \n * Features:\n * - Download elements from GitHub portfolio\n * - Upload elements with consent\n * - Version comparison and diff viewing\n * - Privacy-first with explicit permissions\n * - Conflict resolution strategies\n * - Bulk operations with configuration checks\n */\n\nimport * as fs from 'fs/promises';\nimport * as path from 'path';\nimport { createHash } from 'crypto';\nimport { logger } from '../utils/logger.js';\nimport { ConfigManager, DollhouseConfig } from '../config/ConfigManager.js';\nimport { PortfolioManager } from './PortfolioManager.js';\nimport { PortfolioRepoManager } from './PortfolioRepoManager.js';\nimport { GitHubPortfolioIndexer, GitHubIndexEntry } from './GitHubPortfolioIndexer.js';\nimport { TokenManager } from '../security/tokenManager.js';\nimport { ContentValidator } from '../security/contentValidator.js';\nimport { UnicodeValidator } from '../security/validators/unicodeValidator.js';\nimport { SecureYamlParser } from '../security/secureYamlParser.js';\nimport { ElementType } from './types.js';\nimport { IElement, ElementStatus } from '../types/elements/IElement.js';\n\nexport interface SyncOperation {\n  operation: 'download' | 'upload' | 'compare' | 'list-remote';\n  element_name?: string;\n  element_type?: ElementType;\n  bulk?: boolean;\n  version?: string;\n  show_diff?: boolean;\n  force?: boolean;\n  confirm?: boolean;\n}\n\nexport interface SyncResult {\n  success: boolean;\n  message: string;\n  data?: any;\n  elements?: SyncElementInfo[];\n  conflicts?: ConflictInfo[];\n}\n\nexport interface SyncElementInfo {\n  name: string;\n  type: ElementType;\n  localVersion?: string;\n  remoteVersion?: string;\n  status: 'new' | 'updated' | 'conflict' | 'unchanged' | 'local-only';\n  action?: 'download' | 'upload' | 'skip';\n}\n\nexport interface ConflictInfo {\n  element: string;\n  type: ElementType;\n  localVersion: string;\n  remoteVersion: string;\n  localModified: Date;\n  remoteModified: Date;\n  resolution?: 'local' | 'remote' | 'manual';\n}\n\nexport interface VersionInfo {\n  version: string;\n  timestamp: Date;\n  author: string;\n  hash: string;\n  size: number;\n  source: 'local' | 'remote';\n}\n\nexport interface ElementDiff {\n  element: string;\n  type: ElementType;\n  changes: {\n    metadata?: {\n      field: string;\n      oldValue: any;\n      newValue: any;\n    }[];\n    content?: {\n      additions: number;\n      deletions: number;\n      diff: string;\n    };\n  };\n}\n\nexport class PortfolioSyncManager {\n  private configManager: ConfigManager;\n  private portfolioManager: PortfolioManager;\n  private repoManager: PortfolioRepoManager;\n  private indexer: GitHubPortfolioIndexer;\n  \n  constructor() {\n    this.configManager = ConfigManager.getInstance();\n    this.portfolioManager = PortfolioManager.getInstance();\n    this.repoManager = new PortfolioRepoManager();\n    this.indexer = GitHubPortfolioIndexer.getInstance();\n  }\n  \n  /**\n   * Main handler for sync operations\n   */\n  public async handleSyncOperation(params: SyncOperation): Promise<SyncResult> {\n    try {\n      // Check if sync is enabled in config\n      const config = this.configManager.getConfig();\n      if (!config.sync.enabled && params.operation !== 'list-remote') {\n        return {\n          success: false,\n          message: 'Sync is disabled. Enable it with: dollhouse_config --action update --setting sync.enabled --value true'\n        };\n      }\n      \n      // Check bulk permissions\n      if (params.bulk) {\n        const bulkAllowed = this.isBulkOperationAllowed(params.operation, config);\n        if (!bulkAllowed.allowed) {\n          return {\n            success: false,\n            message: bulkAllowed.message\n          };\n        }\n      }\n      \n      // Handle operations\n      switch (params.operation) {\n        case 'list-remote':\n          return await this.listRemoteElements(params.element_type);\n          \n        case 'download':\n          if (params.bulk) {\n            return await this.bulkDownload(params.element_type, params.confirm);\n          } else if (params.element_name) {\n            return await this.downloadElement(\n              params.element_name,\n              params.element_type!,\n              params.version,\n              params.force\n            );\n          } else {\n            return {\n              success: false,\n              message: 'Element name required for individual download'\n            };\n          }\n          \n        case 'upload':\n          if (params.bulk) {\n            return await this.bulkUpload(params.element_type, params.confirm);\n          } else if (params.element_name) {\n            return await this.uploadElement(\n              params.element_name,\n              params.element_type!,\n              params.confirm\n            );\n          } else {\n            return {\n              success: false,\n              message: 'Element name required for individual upload'\n            };\n          }\n          \n        case 'compare':\n          if (params.element_name && params.element_type) {\n            return await this.compareVersions(\n              params.element_name,\n              params.element_type,\n              params.show_diff\n            );\n          } else {\n            return {\n              success: false,\n              message: 'Element name and type required for comparison'\n            };\n          }\n          \n        default:\n          return {\n            success: false,\n            message: `Unknown operation: ${params.operation}`\n          };\n      }\n    } catch (error) {\n      logger.error('Sync operation failed', {\n        operation: params.operation,\n        error: error instanceof Error ? error.message : String(error)\n      });\n      \n      return {\n        success: false,\n        message: `Sync operation failed: ${error instanceof Error ? error.message : String(error)}`\n      };\n    }\n  }\n  \n  /**\n   * Check if bulk operation is allowed\n   */\n  private isBulkOperationAllowed(operation: string, config: DollhouseConfig): { allowed: boolean; message: string } {\n    if (operation === 'download' && !config.sync.bulk.download_enabled) {\n      return {\n        allowed: false,\n        message: 'Bulk download is disabled. Enable with: dollhouse_config --action update --setting sync.bulk.download_enabled --value true'\n      };\n    }\n    \n    if (operation === 'upload' && !config.sync.bulk.upload_enabled) {\n      return {\n        allowed: false,\n        message: 'Bulk upload is disabled. Enable with: dollhouse_config --action update --setting sync.bulk.upload_enabled --value true'\n      };\n    }\n    \n    return { allowed: true, message: '' };\n  }\n  \n  /**\n   * List elements available in GitHub portfolio\n   */\n  private async listRemoteElements(filterType?: ElementType): Promise<SyncResult> {\n    try {\n      // Get GitHub token\n      const token = await TokenManager.getGitHubTokenAsync();\n      if (!token) {\n        return {\n          success: false,\n          message: 'GitHub authentication required. Use setup_github_auth first.'\n        };\n      }\n      \n      this.repoManager.setToken(token);\n      \n      // Get index of GitHub portfolio\n      const index = await this.indexer.getIndex();\n      \n      if (!index || index.totalElements === 0) {\n        return {\n          success: true,\n          message: 'No elements found in GitHub portfolio',\n          elements: []\n        };\n      }\n      \n      // Format elements for display\n      const elements: SyncElementInfo[] = [];\n      \n      for (const [type, entries] of index.elements) {\n        // Skip if filtering by type and this isn't the requested type\n        if (filterType && type !== filterType) {\n          continue;\n        }\n        \n        for (const entry of entries) {\n          elements.push({\n            name: entry.name,\n            type: type,\n            remoteVersion: entry.version,\n            status: 'unchanged',\n            action: 'download'\n          });\n        }\n      }\n      \n      return {\n        success: true,\n        message: `Found ${elements.length} elements in GitHub portfolio`,\n        elements\n      };\n      \n    } catch (error) {\n      return {\n        success: false,\n        message: `Failed to list remote elements: ${error instanceof Error ? error.message : String(error)}`\n      };\n    }\n  }\n  \n  /**\n   * Download a specific element from GitHub\n   */\n  private async downloadElement(\n    elementName: string,\n    elementType: ElementType,\n    version?: string,\n    force?: boolean\n  ): Promise<SyncResult> {\n    try {\n      const config = this.configManager.getConfig();\n      \n      // Validate element name\n      const validation = UnicodeValidator.normalize(elementName);\n      if (!validation.isValid) {\n        return {\n          success: false,\n          message: `Invalid element name: ${validation.detectedIssues?.[0] || 'unknown error'}`\n        };\n      }\n      \n      // Get token and set it\n      const token = await TokenManager.getGitHubTokenAsync();\n      if (!token) {\n        return {\n          success: false,\n          message: 'GitHub authentication required'\n        };\n      }\n      \n      this.repoManager.setToken(token);\n      \n      // Get GitHub index\n      const index = await this.indexer.getIndex();\n      \n      // Find the element - first try exact match, then fuzzy match\n      const entries = index.elements.get(elementType) || [];\n      let entry = entries.find(e => e.name === elementName);\n      \n      // If exact match not found, try fuzzy matching\n      if (!entry) {\n        // Try case-insensitive exact match first\n        entry = entries.find(e => e.name.toLowerCase() === elementName.toLowerCase());\n        \n        // If still not found, try fuzzy matching\n        if (!entry) {\n          const fuzzyMatch = this.findFuzzyMatch(elementName, entries);\n          if (fuzzyMatch) {\n            logger.info(`Fuzzy match found: '${elementName}' matched to '${fuzzyMatch.name}'`);\n            entry = fuzzyMatch;\n          }\n        }\n      }\n      \n      if (!entry) {\n        // Generate helpful suggestions\n        const suggestions = this.getSuggestions(elementName, entries);\n        const suggestionText = suggestions.length > 0 \n          ? `\\n\\nDid you mean one of these?\\n${suggestions.map(s => `  • ${s.name}`).join('\\n')}`\n          : '';\n        \n        return {\n          success: false,\n          message: `Element '${elementName}' (${elementType}) not found in GitHub portfolio${suggestionText}`\n        };\n      }\n      \n      // Check for local conflicts\n      const localPath = this.portfolioManager.getElementPath(elementType, `${elementName}.md`);\n      let hasLocalVersion = false;\n      let localContent: string | null = null;\n      \n      try {\n        localContent = await fs.readFile(localPath, 'utf-8');\n        hasLocalVersion = true;\n      } catch {\n        // No local version exists\n      }\n      \n      // Download the element\n      const response = await fetch(entry.downloadUrl, {\n        headers: {\n          'Authorization': `Bearer ${token}`,\n          'Accept': 'application/vnd.github.v3.raw'\n        }\n      });\n      \n      if (!response.ok) {\n        throw new Error(`Failed to download: ${response.statusText}`);\n      }\n      \n      const remoteContent = await response.text();\n      \n      // Validate content security\n      const validationResult = ContentValidator.validateAndSanitize(remoteContent);\n      if (!validationResult.isValid && validationResult.severity === 'critical') {\n        return {\n          success: false,\n          message: `Security issue detected in remote content: ${validationResult.detectedPatterns?.join(', ')}`\n        };\n      }\n      \n      // Check if content is different\n      if (hasLocalVersion && localContent) {\n        const localHash = createHash('sha256').update(localContent).digest('hex');\n        const remoteHash = createHash('sha256').update(remoteContent).digest('hex');\n        \n        if (localHash === remoteHash) {\n          return {\n            success: true,\n            message: `Element '${elementName}' is already up to date`\n          };\n        }\n        \n        // Show confirmation for overwrite unless force flag is set\n        if (config.sync.individual.require_confirmation && !force) {\n          const diff = await this.generateDiff(localContent, remoteContent);\n          \n          return {\n            success: false,\n            message: `Local version exists. Please confirm download will overwrite:\\n\\n${diff}\\n\\nTo proceed, use --force flag`,\n            data: { requiresConfirmation: true }\n          };\n        }\n      }\n      \n      // Save the element\n      await fs.mkdir(path.dirname(localPath), { recursive: true });\n      await fs.writeFile(localPath, remoteContent, 'utf-8');\n      \n      logger.info('Element downloaded from GitHub', {\n        element: elementName,\n        type: elementType,\n        version: entry.version\n      });\n      \n      return {\n        success: true,\n        message: `Successfully downloaded '${elementName}' (${elementType}) from GitHub portfolio`\n      };\n      \n    } catch (error) {\n      return {\n        success: false,\n        message: `Failed to download element: ${error instanceof Error ? error.message : String(error)}`\n      };\n    }\n  }\n  \n  /**\n   * Upload a specific element to GitHub\n   */\n  private async uploadElement(\n    elementName: string,\n    elementType: ElementType,\n    confirm?: boolean\n  ): Promise<SyncResult> {\n    try {\n      const config = this.configManager.getConfig();\n      \n      // Check for local element\n      const localPath = this.portfolioManager.getElementPath(elementType, `${elementName}.md`);\n      \n      let content: string;\n      try {\n        content = await fs.readFile(localPath, 'utf-8');\n      } catch {\n        return {\n          success: false,\n          message: `Element '${elementName}' (${elementType}) not found locally`\n        };\n      }\n      \n      // Check privacy metadata\n      const parsed = SecureYamlParser.parse(content, {\n        maxYamlSize: 64 * 1024,\n        validateContent: false,\n        validateFields: false\n      });\n      \n      if (parsed.data?.privacy?.local_only === true) {\n        return {\n          success: false,\n          message: `Element '${elementName}' is marked as local-only and cannot be uploaded`\n        };\n      }\n      \n      // Validate content security\n      const validationResult = ContentValidator.validateAndSanitize(content);\n      if (!validationResult.isValid && validationResult.severity === 'critical') {\n        return {\n          success: false,\n          message: `Security issue detected: ${validationResult.detectedPatterns?.join(', ')}`\n        };\n      }\n      \n      // Scan for sensitive content if configured\n      if (config.sync.privacy.scan_for_secrets) {\n        logger.debug('Scanning for secrets before upload');\n        // Implement actual secret scanning\n        const secretPatterns = [\n          /api[_-]?key\\s*[:=]\\s*['\"][^'\"]+['\"]/gi,\n          /secret\\s*[:=]\\s*['\"][^'\"]+['\"]/gi,\n          /password\\s*[:=]\\s*['\"][^'\"]+['\"]/gi,\n          /token\\s*[:=]\\s*['\"][^'\"]+['\"]/gi,\n          /private[_-]?key\\s*[:=]\\s*['\"][^'\"]+['\"]/gi\n        ];\n        \n        for (const pattern of secretPatterns) {\n          if (pattern.test(content)) {\n            return {\n              success: false,\n              message: `Potential secret detected in content. Please review and remove sensitive information before uploading.`\n            };\n          }\n        }\n      }\n      \n      // Get confirmation if required (unless already confirmed)\n      if (config.sync.individual.require_confirmation && !confirm) {\n        return {\n          success: false,\n          message: `Please confirm upload of '${elementName}' (${elementType}) to GitHub.\\n\\nContent preview:\\n${content.substring(0, 500)}...\\n\\nTo proceed, use --confirm flag`,\n          data: { requiresConfirmation: true }\n        };\n      }\n      \n      // Get token and validate\n      const token = await TokenManager.getGitHubTokenAsync();\n      if (!token) {\n        return {\n          success: false,\n          message: 'GitHub authentication required'\n        };\n      }\n      \n      // Create an IElement object for the PortfolioRepoManager\n      const element: IElement = {\n        id: `${elementType}_${elementName}_${Date.now()}`,\n        type: elementType,\n        version: parsed.data?.version || '1.0.0',\n        metadata: {\n          name: elementName,\n          description: parsed.data?.description || '',\n          author: parsed.data?.author || 'unknown',\n          created: parsed.data?.created || new Date().toISOString(),\n          modified: new Date().toISOString(),\n          tags: parsed.data?.tags || [],\n          custom: parsed.data\n        },\n        validate: () => ({ valid: true, errors: [], warnings: [] }),\n        serialize: () => content,\n        deserialize: () => {},\n        getStatus: () => ElementStatus.ACTIVE\n      };\n      \n      // Use PortfolioRepoManager to upload\n      this.repoManager.setToken(token);\n      \n      try {\n        const url = await this.repoManager.saveElement(element, true); // consent is true since we've already checked\n        \n        logger.info('Element uploaded to GitHub', {\n          element: elementName,\n          type: elementType,\n          url\n        });\n        \n        return {\n          success: true,\n          message: `Successfully uploaded '${elementName}' (${elementType}) to GitHub portfolio`,\n          data: { url }\n        };\n      } catch (uploadError) {\n        // Handle specific errors\n        if (uploadError instanceof Error && uploadError.message.includes('repository does not exist')) {\n          return {\n            success: false,\n            message: `GitHub portfolio repository not found. Please initialize it first using init_portfolio tool.`\n          };\n        }\n        throw uploadError;\n      }\n      \n    } catch (error) {\n      return {\n        success: false,\n        message: `Failed to upload element: ${error instanceof Error ? error.message : String(error)}`\n      };\n    }\n  }\n  \n  /**\n   * Compare local and remote versions\n   */\n  private async compareVersions(\n    elementName: string,\n    elementType: ElementType,\n    showDiff?: boolean\n  ): Promise<SyncResult> {\n    try {\n      // Get local version\n      const localPath = this.portfolioManager.getElementPath(elementType, `${elementName}.md`);\n      let localContent: string | null = null;\n      let localVersion: VersionInfo | null = null;\n      \n      try {\n        localContent = await fs.readFile(localPath, 'utf-8');\n        const parsed = SecureYamlParser.parse(localContent, {\n          maxYamlSize: 64 * 1024,\n          validateContent: false,\n          validateFields: false\n        });\n        \n        localVersion = {\n          version: parsed.data?.version || '1.0.0',\n          timestamp: new Date(parsed.data?.updated || parsed.data?.created || Date.now()),\n          author: parsed.data?.author || 'unknown',\n          hash: createHash('sha256').update(localContent).digest('hex'),\n          size: Buffer.byteLength(localContent),\n          source: 'local'\n        };\n      } catch {\n        // No local version\n      }\n      \n      // Get remote version\n      const token = await TokenManager.getGitHubTokenAsync();\n      if (!token) {\n        return {\n          success: false,\n          message: 'GitHub authentication required'\n        };\n      }\n      \n      const index = await this.indexer.getIndex();\n      const entries = index.elements.get(elementType) || [];\n      const entry = entries.find(e => e.name === elementName);\n      \n      let remoteVersion: VersionInfo | null = null;\n      let remoteContent: string | null = null;\n      \n      if (entry) {\n        const response = await fetch(entry.downloadUrl, {\n          headers: {\n            'Authorization': `Bearer ${token}`,\n            'Accept': 'application/vnd.github.v3.raw'\n          }\n        });\n        \n        if (response.ok) {\n          remoteContent = await response.text();\n          remoteVersion = {\n            version: entry.version || '1.0.0',\n            timestamp: entry.lastModified,\n            author: entry.author || 'unknown',\n            hash: createHash('sha256').update(remoteContent).digest('hex'),\n            size: entry.size,\n            source: 'remote'\n          };\n        }\n      }\n      \n      // Build comparison result\n      const result: any = {\n        element: elementName,\n        type: elementType,\n        local: localVersion,\n        remote: remoteVersion\n      };\n      \n      if (localVersion && remoteVersion) {\n        result.status = localVersion.hash === remoteVersion.hash ? 'identical' : 'different';\n        \n        if (showDiff && localContent && remoteContent && result.status === 'different') {\n          result.diff = await this.generateDiff(localContent, remoteContent);\n        }\n      } else if (localVersion && !remoteVersion) {\n        result.status = 'local-only';\n      } else if (!localVersion && remoteVersion) {\n        result.status = 'remote-only';\n      } else {\n        result.status = 'not-found';\n      }\n      \n      return {\n        success: true,\n        message: `Version comparison for '${elementName}' (${elementType})`,\n        data: result\n      };\n      \n    } catch (error) {\n      return {\n        success: false,\n        message: `Failed to compare versions: ${error instanceof Error ? error.message : String(error)}`\n      };\n    }\n  }\n  \n  /**\n   * Bulk download elements\n   */\n  private async bulkDownload(elementType?: ElementType, confirm?: boolean): Promise<SyncResult> {\n    const config = this.configManager.getConfig();\n    \n    if (!config.sync.bulk.download_enabled) {\n      return {\n        success: false,\n        message: 'Bulk download is not enabled in configuration'\n      };\n    }\n    \n    // Get list of remote elements\n    const remoteResult = await this.listRemoteElements();\n    if (!remoteResult.success || !remoteResult.elements) {\n      return remoteResult;\n    }\n    \n    // Filter by type if specified\n    let elementsToDownload = remoteResult.elements;\n    if (elementType) {\n      elementsToDownload = elementsToDownload.filter(e => e.type === elementType);\n    }\n    \n    if (elementsToDownload.length === 0) {\n      return {\n        success: true,\n        message: 'No elements to download',\n        elements: []\n      };\n    }\n    \n    // Show preview if required (unless already confirmed)\n    if (config.sync.bulk.require_preview && !confirm) {\n      return {\n        success: false,\n        message: `Bulk download preview:\\n\\n${elementsToDownload.length} elements will be downloaded:\\n${elementsToDownload.map(e => `- ${e.name} (${e.type})`).join('\\n')}\\n\\nTo proceed, use --confirm flag`,\n        data: { requiresConfirmation: true },\n        elements: elementsToDownload\n      };\n    }\n    \n    // Perform actual bulk download\n    const results = {\n      downloaded: [] as string[],\n      skipped: [] as string[],\n      failed: [] as { name: string; error: string }[]\n    };\n    \n    for (const element of elementsToDownload) {\n      try {\n        const result = await this.downloadElement(element.name, element.type, undefined, true); // force=true to skip individual confirmations\n        if (result.success) {\n          results.downloaded.push(element.name);\n        } else if (result.message?.includes('already up to date')) {\n          results.skipped.push(element.name);\n        } else {\n          results.failed.push({ name: element.name, error: result.message || 'Unknown error' });\n        }\n      } catch (error) {\n        results.failed.push({ \n          name: element.name, \n          error: error instanceof Error ? error.message : String(error) \n        });\n      }\n    }\n    \n    // Build summary message\n    let message = `Bulk download complete:\\n`;\n    message += `- Downloaded: ${results.downloaded.length} elements\\n`;\n    message += `- Skipped (up to date): ${results.skipped.length} elements\\n`;\n    message += `- Failed: ${results.failed.length} elements`;\n    \n    if (results.failed.length > 0) {\n      message += `\\n\\nFailed downloads:\\n${results.failed.map(f => `- ${f.name}: ${f.error}`).join('\\n')}`;\n    }\n    \n    return {\n      success: results.failed.length === 0,\n      message,\n      data: results\n    };\n  }\n  \n  /**\n   * Bulk upload elements\n   */\n  private async bulkUpload(elementType?: ElementType, confirm?: boolean): Promise<SyncResult> {\n    const config = this.configManager.getConfig();\n    \n    if (!config.sync.bulk.upload_enabled) {\n      return {\n        success: false,\n        message: 'Bulk upload is not enabled in configuration'\n      };\n    }\n    \n    // Get list of local elements\n    const types = elementType ? [elementType] : [\n      ElementType.PERSONA,\n      ElementType.SKILL,\n      ElementType.TEMPLATE,\n      ElementType.AGENT,\n      ElementType.MEMORY,\n      ElementType.ENSEMBLE\n    ];\n    \n    const localElements: { name: string; type: ElementType; path: string }[] = [];\n    \n    for (const type of types) {\n      const dir = this.portfolioManager.getElementDir(type);\n      try {\n        const files = await fs.readdir(dir);\n        for (const file of files) {\n          if (file.endsWith('.md')) {\n            localElements.push({\n              name: file.replace('.md', ''),\n              type,\n              path: path.join(dir, file)\n            });\n          }\n        }\n      } catch (error) {\n        // Directory may not exist yet\n        logger.debug(`Directory for ${type} does not exist yet`);\n      }\n    }\n    \n    if (localElements.length === 0) {\n      return {\n        success: true,\n        message: 'No local elements to upload',\n        elements: []\n      };\n    }\n    \n    // Show preview if required (unless already confirmed)\n    if (config.sync.bulk.require_preview && !confirm) {\n      // Convert to SyncElementInfo format for preview\n      const previewElements: SyncElementInfo[] = localElements.map(e => ({\n        name: e.name,\n        type: e.type,\n        status: 'local-only' as const,\n        action: 'upload' as const\n      }));\n      \n      return {\n        success: false,\n        message: `Bulk upload preview:\\n\\n${localElements.length} elements will be uploaded:\\n${localElements.map(e => `- ${e.name} (${e.type})`).join('\\n')}\\n\\nTo proceed, use --confirm flag`,\n        data: { requiresConfirmation: true },\n        elements: previewElements\n      };\n    }\n    \n    // Perform actual bulk upload\n    const results = {\n      uploaded: [] as string[],\n      skipped: [] as string[],\n      failed: [] as { name: string; error: string }[]\n    };\n    \n    for (const element of localElements) {\n      try {\n        const result = await this.uploadElement(element.name, element.type, true); // confirm=true to skip individual confirmations\n        if (result.success) {\n          results.uploaded.push(element.name);\n        } else if (result.message?.includes('local-only')) {\n          results.skipped.push(element.name);\n        } else {\n          results.failed.push({ name: element.name, error: result.message || 'Unknown error' });\n        }\n      } catch (error) {\n        results.failed.push({ \n          name: element.name, \n          error: error instanceof Error ? error.message : String(error) \n        });\n      }\n    }\n    \n    // Build summary message\n    let message = `Bulk upload complete:\\n`;\n    message += `- Uploaded: ${results.uploaded.length} elements\\n`;\n    message += `- Skipped (local-only): ${results.skipped.length} elements\\n`;\n    message += `- Failed: ${results.failed.length} elements`;\n    \n    if (results.failed.length > 0) {\n      message += `\\n\\nFailed uploads:\\n${results.failed.map(f => `- ${f.name}: ${f.error}`).join('\\n')}`;\n    }\n    \n    return {\n      success: results.failed.length === 0,\n      message,\n      data: results\n    };\n  }\n  \n  /**\n   * Generate diff between two content versions\n   */\n  private async generateDiff(local: string, remote: string): Promise<string> {\n    // Simple line-based diff for now\n    const localLines = local.split('\\n');\n    const remoteLines = remote.split('\\n');\n    \n    let diff = '';\n    const maxLines = Math.max(localLines.length, remoteLines.length);\n    \n    for (let i = 0; i < maxLines && i < 10; i++) { // Show first 10 lines of diff\n      const localLine = localLines[i] || '';\n      const remoteLine = remoteLines[i] || '';\n      \n      if (localLine !== remoteLine) {\n        if (localLine && !remoteLine) {\n          diff += `- ${localLine}\\n`;\n        } else if (!localLine && remoteLine) {\n          diff += `+ ${remoteLine}\\n`;\n        } else {\n          diff += `- ${localLine}\\n`;\n          diff += `+ ${remoteLine}\\n`;\n        }\n      }\n    }\n    \n    if (maxLines > 10) {\n      diff += `\\n... ${maxLines - 10} more lines ...`;\n    }\n    \n    return diff || 'No differences found';\n  }\n  \n  /**\n   * Find a fuzzy match for an element name\n   */\n  private findFuzzyMatch(searchName: string, entries: GitHubIndexEntry[]): GitHubIndexEntry | null {\n    const search = searchName.toLowerCase().replace(/[-_]/g, '');\n    let bestMatch: typeof entries[0] | null = null;\n    let bestScore = 0;\n    \n    for (const entry of entries) {\n      // Normalize the entry name for comparison\n      const normalized = entry.name.toLowerCase().replace(/[-_]/g, '');\n      \n      // Calculate similarity score\n      const score = this.calculateSimilarity(search, normalized);\n      if (score > bestScore && score > 0.5) { // Minimum threshold of 0.5\n        bestScore = score;\n        bestMatch = entry;\n      }\n    }\n    \n    return bestMatch;\n  }\n  \n  /**\n   * Get suggestions for similar element names\n   */\n  private getSuggestions(searchName: string, entries: GitHubIndexEntry[]): Array<{name: string}> {\n    const search = searchName.toLowerCase().replace(/[-_]/g, '');\n    const scored: Array<{entry: typeof entries[0]; score: number}> = [];\n    \n    for (const entry of entries) {\n      const normalized = entry.name.toLowerCase().replace(/[-_]/g, '');\n      const score = this.calculateSimilarity(search, normalized);\n      if (score > 0.3) { // Lower threshold for suggestions\n        scored.push({ entry, score });\n      }\n    }\n    \n    // Sort by score and return top 5\n    return scored\n      .sort((a, b) => b.score - a.score)\n      .slice(0, 5)\n      .map(s => ({ name: s.entry.name }));\n  }\n  \n  /**\n   * Calculate similarity between two strings\n   * Returns a score between 0 and 1\n   */\n  private calculateSimilarity(a: string, b: string): number {\n    // Exact match\n    if (a === b) return 1.0;\n    \n    // One contains the other\n    if (a.includes(b) || b.includes(a)) return 0.8;\n    \n    // Calculate word overlap\n    const wordsA = a.split(/[^a-z0-9]+/);\n    const wordsB = b.split(/[^a-z0-9]+/);\n    \n    let matches = 0;\n    for (const wordA of wordsA) {\n      if (wordA && wordsB.some(wordB => wordB === wordA)) {\n        matches++;\n      }\n    }\n    \n    if (matches > 0) {\n      const overlap = (matches * 2) / (wordsA.length + wordsB.length);\n      return Math.max(0.6, overlap); // At least 0.6 for any word match\n    }\n    \n    // Check for partial matches\n    for (const wordA of wordsA) {\n      for (const wordB of wordsB) {\n        if (wordA.length > 3 && wordB.length > 3) {\n          if (wordA.includes(wordB) || wordB.includes(wordA)) {\n            return 0.5;\n          }\n        }\n      }\n    }\n    \n    // No significant similarity\n    return 0;\n  }\n}"]}
|