@dollhousemcp/mcp-server 1.3.2 → 1.3.3

This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
Files changed (64) hide show
  1. package/README.md +87 -5
  2. package/dist/collection/ElementInstaller.d.ts.map +1 -1
  3. package/dist/collection/ElementInstaller.js +3 -7
  4. package/dist/elements/index.d.ts +0 -2
  5. package/dist/elements/index.d.ts.map +1 -1
  6. package/dist/elements/index.js +1 -3
  7. package/dist/elements/skills/Skill.d.ts.map +1 -1
  8. package/dist/elements/skills/Skill.js +3 -2
  9. package/dist/elements/skills/SkillManager.d.ts +83 -0
  10. package/dist/elements/skills/SkillManager.d.ts.map +1 -0
  11. package/dist/elements/skills/SkillManager.js +383 -0
  12. package/dist/elements/skills/index.d.ts +1 -0
  13. package/dist/elements/skills/index.d.ts.map +1 -1
  14. package/dist/elements/skills/index.js +2 -1
  15. package/dist/index.d.ts +51 -0
  16. package/dist/index.d.ts.map +1 -1
  17. package/dist/index.js +577 -3
  18. package/dist/portfolio/PortfolioManager.d.ts.map +1 -1
  19. package/dist/portfolio/PortfolioManager.js +1 -3
  20. package/dist/portfolio/types.d.ts +1 -3
  21. package/dist/portfolio/types.d.ts.map +1 -1
  22. package/dist/portfolio/types.js +1 -3
  23. package/dist/security/audit/config/suppressions.d.ts.map +1 -1
  24. package/dist/security/audit/config/suppressions.js +6 -1
  25. package/dist/server/ServerSetup.d.ts.map +1 -1
  26. package/dist/server/ServerSetup.js +5 -2
  27. package/dist/server/tools/CollectionTools.js +6 -6
  28. package/dist/server/tools/ElementTools.d.ts +11 -0
  29. package/dist/server/tools/ElementTools.d.ts.map +1 -0
  30. package/dist/server/tools/ElementTools.js +174 -0
  31. package/dist/server/tools/index.d.ts +1 -0
  32. package/dist/server/tools/index.d.ts.map +1 -1
  33. package/dist/server/tools/index.js +2 -1
  34. package/dist/server/types.d.ts +8 -0
  35. package/dist/server/types.d.ts.map +1 -1
  36. package/dist/server/types.js +1 -1
  37. package/package.json +1 -1
  38. package/dist/elements/ensembles/Ensemble.d.ts +0 -144
  39. package/dist/elements/ensembles/Ensemble.d.ts.map +0 -1
  40. package/dist/elements/ensembles/Ensemble.js +0 -860
  41. package/dist/elements/ensembles/EnsembleManager.d.ts +0 -85
  42. package/dist/elements/ensembles/EnsembleManager.d.ts.map +0 -1
  43. package/dist/elements/ensembles/EnsembleManager.js +0 -378
  44. package/dist/elements/ensembles/constants.d.ts +0 -73
  45. package/dist/elements/ensembles/constants.d.ts.map +0 -1
  46. package/dist/elements/ensembles/constants.js +0 -92
  47. package/dist/elements/ensembles/index.d.ts +0 -8
  48. package/dist/elements/ensembles/index.d.ts.map +0 -1
  49. package/dist/elements/ensembles/index.js +0 -8
  50. package/dist/elements/ensembles/types.d.ts +0 -92
  51. package/dist/elements/ensembles/types.d.ts.map +0 -1
  52. package/dist/elements/ensembles/types.js +0 -8
  53. package/dist/elements/memories/Memory.d.ts +0 -110
  54. package/dist/elements/memories/Memory.d.ts.map +0 -1
  55. package/dist/elements/memories/Memory.js +0 -470
  56. package/dist/elements/memories/MemoryManager.d.ts +0 -86
  57. package/dist/elements/memories/MemoryManager.d.ts.map +0 -1
  58. package/dist/elements/memories/MemoryManager.js +0 -435
  59. package/dist/elements/memories/constants.d.ts +0 -42
  60. package/dist/elements/memories/constants.d.ts.map +0 -1
  61. package/dist/elements/memories/constants.js +0 -49
  62. package/dist/elements/memories/index.d.ts +0 -6
  63. package/dist/elements/memories/index.d.ts.map +0 -1
  64. package/dist/elements/memories/index.js +0 -6
@@ -1,470 +0,0 @@
1
- /**
2
- * Memory Element - Persistent context storage for continuity and learning
3
- *
4
- * Provides multiple storage backends, retention policies, and search capabilities
5
- * for maintaining context across sessions and interactions.
6
- *
7
- * SECURITY MEASURES IMPLEMENTED:
8
- * 1. Input sanitization for all memory content
9
- * 2. Memory size limits to prevent unbounded growth
10
- * 3. Path validation for file-based storage
11
- * 4. Retention policy enforcement
12
- * 5. Privacy level access control
13
- * 6. Audit logging for all operations
14
- */
15
- import { BaseElement } from '../BaseElement.js';
16
- import { ElementType } from '../../portfolio/types.js';
17
- import { UnicodeValidator } from '../../security/validators/unicodeValidator.js';
18
- import { SecurityMonitor } from '../../security/securityMonitor.js';
19
- import { sanitizeInput } from '../../security/InputValidator.js';
20
- import { MEMORY_CONSTANTS, MEMORY_SECURITY_EVENTS } from './constants.js';
21
- import DOMPurify from 'dompurify';
22
- import { JSDOM } from 'jsdom';
23
- // Initialize DOMPurify with JSDOM
24
- const window = new JSDOM('').window;
25
- const purify = DOMPurify(window);
26
- // Configure DOMPurify for memory content - strip all HTML but keep text
27
- purify.setConfig({
28
- ALLOWED_TAGS: [], // No HTML tags allowed
29
- ALLOWED_ATTR: [], // No attributes allowed
30
- KEEP_CONTENT: true // Keep text content
31
- });
32
- /**
33
- * Sanitize content for memory storage
34
- * More permissive than sanitizeInput - allows punctuation, quotes, etc.
35
- * but still prevents XSS and control characters
36
- */
37
- function sanitizeMemoryContent(content, maxLength) {
38
- if (!content || typeof content !== 'string') {
39
- return '';
40
- }
41
- // First normalize Unicode
42
- const normalized = UnicodeValidator.normalize(content).normalizedContent;
43
- // Use DOMPurify to strip any HTML/XSS attempts but keep text
44
- const cleaned = purify.sanitize(normalized);
45
- // Remove only control characters and null bytes
46
- return cleaned
47
- .replace(/[\x00-\x08\x0B\x0C\x0E-\x1F\x7F]/g, '') // Remove control chars except \t \n \r
48
- .substring(0, maxLength)
49
- .trim();
50
- }
51
- export class Memory extends BaseElement {
52
- // Memory-specific properties
53
- entries = new Map();
54
- storageBackend;
55
- retentionDays;
56
- privacyLevel;
57
- searchable;
58
- maxEntries;
59
- constructor(metadata = {}) {
60
- // SECURITY FIX: Sanitize all inputs during construction
61
- const sanitizedMetadata = {
62
- ...metadata,
63
- name: metadata.name ?
64
- sanitizeInput(UnicodeValidator.normalize(metadata.name).normalizedContent, 100) :
65
- 'Unnamed Memory',
66
- description: metadata.description ?
67
- sanitizeInput(UnicodeValidator.normalize(metadata.description).normalizedContent, 500) :
68
- undefined
69
- };
70
- super(ElementType.MEMORY, sanitizedMetadata);
71
- // Initialize memory-specific properties with defaults
72
- this.storageBackend = metadata.storageBackend || MEMORY_CONSTANTS.DEFAULT_STORAGE_BACKEND;
73
- this.retentionDays = metadata.retentionDays || MEMORY_CONSTANTS.DEFAULT_RETENTION_DAYS;
74
- // Validate privacy level - default to private if invalid
75
- this.privacyLevel = (metadata.privacyLevel && MEMORY_CONSTANTS.PRIVACY_LEVELS.includes(metadata.privacyLevel))
76
- ? metadata.privacyLevel
77
- : MEMORY_CONSTANTS.DEFAULT_PRIVACY_LEVEL;
78
- this.searchable = metadata.searchable !== false;
79
- this.maxEntries = Math.min(metadata.maxEntries || MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT, MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT);
80
- // Set up extensions
81
- this.extensions = {
82
- storageBackend: this.storageBackend,
83
- retentionDays: this.retentionDays,
84
- privacyLevel: this.privacyLevel,
85
- searchable: this.searchable,
86
- maxEntries: this.maxEntries,
87
- encryptionEnabled: metadata.encryptionEnabled || false
88
- };
89
- // Log memory creation
90
- SecurityMonitor.logSecurityEvent({
91
- type: MEMORY_SECURITY_EVENTS.MEMORY_CREATED,
92
- severity: 'LOW',
93
- source: 'Memory.constructor',
94
- details: `Memory created: ${this.metadata.name} with ${this.storageBackend} backend`
95
- });
96
- }
97
- /**
98
- * Add a new memory entry
99
- * SECURITY: Validates and sanitizes all input, enforces size limits
100
- */
101
- async addEntry(content, tags, metadata) {
102
- // Validate memory size limits
103
- if (this.entries.size >= this.maxEntries) {
104
- // SECURITY FIX: Enforce retention policy when at capacity
105
- await this.enforceRetentionPolicy();
106
- // If still at capacity after retention, remove oldest to make room
107
- if (this.entries.size >= this.maxEntries) {
108
- const oldestEntry = Array.from(this.entries.values())
109
- .sort((a, b) => a.timestamp.getTime() - b.timestamp.getTime())[0];
110
- if (oldestEntry) {
111
- this.entries.delete(oldestEntry.id);
112
- }
113
- }
114
- }
115
- // SECURITY FIX: Validate and sanitize content
116
- const sanitizedContent = sanitizeMemoryContent(content, MEMORY_CONSTANTS.MAX_ENTRY_SIZE);
117
- if (!sanitizedContent || sanitizedContent.trim().length === 0) {
118
- throw new Error('Memory content cannot be empty');
119
- }
120
- // SECURITY FIX: Validate and sanitize tags
121
- const sanitizedTags = tags ? this.sanitizeTags(tags) : [];
122
- // Create memory entry
123
- const entry = {
124
- id: `mem_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`,
125
- timestamp: new Date(),
126
- content: sanitizedContent,
127
- tags: sanitizedTags,
128
- metadata: this.sanitizeMetadata(metadata),
129
- privacyLevel: this.privacyLevel,
130
- expiresAt: this.calculateExpiryDate()
131
- };
132
- // Store entry
133
- this.entries.set(entry.id, entry);
134
- this._isDirty = true;
135
- // Log memory addition
136
- SecurityMonitor.logSecurityEvent({
137
- type: MEMORY_SECURITY_EVENTS.MEMORY_ADDED,
138
- severity: 'LOW',
139
- source: 'Memory.addEntry',
140
- details: `Added memory entry ${entry.id} with ${sanitizedTags.length} tags`
141
- });
142
- return entry;
143
- }
144
- /**
145
- * Search memory entries
146
- * SECURITY: Respects privacy levels and sanitizes search queries
147
- */
148
- async search(options = {}) {
149
- // SECURITY FIX: Sanitize search query (use regular sanitizeInput for queries)
150
- const sanitizedQuery = options.query ?
151
- sanitizeInput(UnicodeValidator.normalize(options.query).normalizedContent, 200) :
152
- undefined;
153
- // PERFORMANCE OPTIMIZATION: Single-pass filtering to reduce allocations
154
- let results = [];
155
- const queryLower = sanitizedQuery?.toLowerCase();
156
- const searchTags = options.tags && options.tags.length > 0 ? this.sanitizeTags(options.tags) : null;
157
- // Single iteration through entries with all filters applied
158
- for (const entry of this.entries.values()) {
159
- // Privacy level check
160
- if (options.privacyLevel &&
161
- !this.canAccessPrivacyLevel(entry.privacyLevel || MEMORY_CONSTANTS.DEFAULT_PRIVACY_LEVEL, options.privacyLevel)) {
162
- continue;
163
- }
164
- // Query text check
165
- if (queryLower) {
166
- const contentMatch = entry.content.toLowerCase().includes(queryLower);
167
- const tagMatch = entry.tags?.some(tag => tag.toLowerCase().includes(queryLower));
168
- if (!contentMatch && !tagMatch) {
169
- continue;
170
- }
171
- }
172
- // Tag filter check
173
- if (searchTags && !searchTags.some(searchTag => entry.tags?.includes(searchTag))) {
174
- continue;
175
- }
176
- // Date range checks
177
- if (options.startDate && entry.timestamp < options.startDate) {
178
- continue;
179
- }
180
- if (options.endDate && entry.timestamp > options.endDate) {
181
- continue;
182
- }
183
- // Entry passes all filters
184
- results.push(entry);
185
- }
186
- // Sort by timestamp (newest first) - using string comparison for IDs as secondary sort
187
- results.sort((a, b) => {
188
- const timeDiff = b.timestamp.getTime() - a.timestamp.getTime();
189
- if (timeDiff !== 0)
190
- return timeDiff;
191
- // If timestamps are exactly the same, sort by ID (which contains timestamp)
192
- return b.id.localeCompare(a.id);
193
- });
194
- // Apply limit
195
- if (options.limit && options.limit > 0) {
196
- results = results.slice(0, options.limit);
197
- }
198
- // Log search operation
199
- SecurityMonitor.logSecurityEvent({
200
- type: MEMORY_SECURITY_EVENTS.MEMORY_SEARCHED,
201
- severity: 'LOW',
202
- source: 'Memory.search',
203
- details: `Searched memories with query: ${sanitizedQuery || 'none'}, found ${results.length} results`
204
- });
205
- return results;
206
- }
207
- /**
208
- * Get a specific memory entry by ID
209
- */
210
- async getEntry(id) {
211
- return this.entries.get(id);
212
- }
213
- /**
214
- * Delete a memory entry
215
- * SECURITY: Validates permissions and logs deletion
216
- */
217
- async deleteEntry(id) {
218
- const entry = this.entries.get(id);
219
- if (!entry) {
220
- return false;
221
- }
222
- // SECURITY: Check if sensitive memories can be deleted
223
- if (entry.privacyLevel === 'sensitive') {
224
- SecurityMonitor.logSecurityEvent({
225
- type: MEMORY_SECURITY_EVENTS.SENSITIVE_MEMORY_DELETED,
226
- severity: 'MEDIUM',
227
- source: 'Memory.deleteEntry',
228
- details: `Sensitive memory ${id} deleted`
229
- });
230
- }
231
- this.entries.delete(id);
232
- this._isDirty = true;
233
- return true;
234
- }
235
- /**
236
- * Enforce retention policy by removing expired entries
237
- * SECURITY: Ensures memory doesn't grow unbounded
238
- */
239
- async enforceRetentionPolicy() {
240
- const now = new Date();
241
- let deletedCount = 0;
242
- // Remove expired entries
243
- for (const [id, entry] of this.entries) {
244
- if (entry.expiresAt && entry.expiresAt < now) {
245
- this.entries.delete(id);
246
- deletedCount++;
247
- }
248
- }
249
- // If still at or over capacity, remove oldest entries to make room for one more
250
- if (this.entries.size >= this.maxEntries) {
251
- const sortedEntries = Array.from(this.entries.entries())
252
- .sort((a, b) => a[1].timestamp.getTime() - b[1].timestamp.getTime());
253
- // Remove one extra to make room for new entry
254
- const toDelete = Math.max(1, this.entries.size - this.maxEntries + 1);
255
- for (let i = 0; i < toDelete && i < sortedEntries.length; i++) {
256
- this.entries.delete(sortedEntries[i][0]);
257
- deletedCount++;
258
- }
259
- }
260
- if (deletedCount > 0) {
261
- this._isDirty = true;
262
- SecurityMonitor.logSecurityEvent({
263
- type: MEMORY_SECURITY_EVENTS.RETENTION_POLICY_ENFORCED,
264
- severity: 'LOW',
265
- source: 'Memory.enforceRetentionPolicy',
266
- details: `Removed ${deletedCount} expired memories`
267
- });
268
- }
269
- return deletedCount;
270
- }
271
- /**
272
- * Clear all memory entries
273
- * SECURITY: Requires confirmation and logs the action
274
- */
275
- async clearAll(confirm = false) {
276
- if (!confirm) {
277
- throw new Error('Memory clear requires confirmation');
278
- }
279
- const count = this.entries.size;
280
- this.entries.clear();
281
- this._isDirty = true;
282
- SecurityMonitor.logSecurityEvent({
283
- type: MEMORY_SECURITY_EVENTS.MEMORY_CLEARED,
284
- severity: 'HIGH',
285
- source: 'Memory.clearAll',
286
- details: `Cleared all ${count} memory entries`
287
- });
288
- }
289
- /**
290
- * Get memory statistics
291
- */
292
- getStats() {
293
- let totalSize = 0;
294
- let oldestEntry;
295
- let newestEntry;
296
- const tagFrequency = new Map();
297
- for (const entry of this.entries.values()) {
298
- totalSize += entry.content.length;
299
- if (!oldestEntry || entry.timestamp < oldestEntry) {
300
- oldestEntry = entry.timestamp;
301
- }
302
- if (!newestEntry || entry.timestamp > newestEntry) {
303
- newestEntry = entry.timestamp;
304
- }
305
- entry.tags?.forEach(tag => {
306
- tagFrequency.set(tag, (tagFrequency.get(tag) || 0) + 1);
307
- });
308
- }
309
- return {
310
- totalEntries: this.entries.size,
311
- totalSize,
312
- oldestEntry,
313
- newestEntry,
314
- tagFrequency
315
- };
316
- }
317
- /**
318
- * Validate the memory element
319
- */
320
- validate() {
321
- const result = super.validate();
322
- // Initialize errors array if not present
323
- if (!result.errors) {
324
- result.errors = [];
325
- }
326
- // Additional memory-specific validation
327
- if (this.retentionDays < MEMORY_CONSTANTS.MIN_RETENTION_DAYS || this.retentionDays > MEMORY_CONSTANTS.MAX_RETENTION_DAYS) {
328
- result.errors.push({
329
- field: 'retentionDays',
330
- message: `Retention days must be between ${MEMORY_CONSTANTS.MIN_RETENTION_DAYS} and ${MEMORY_CONSTANTS.MAX_RETENTION_DAYS}`,
331
- severity: 'error'
332
- });
333
- }
334
- if (this.maxEntries < 1 || this.maxEntries > MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT) {
335
- result.errors.push({
336
- field: 'maxEntries',
337
- message: `Max entries must be between 1 and ${MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT}`,
338
- severity: 'error'
339
- });
340
- }
341
- // Check memory size
342
- const stats = this.getStats();
343
- if (stats.totalSize > MEMORY_CONSTANTS.MAX_MEMORY_SIZE) {
344
- result.errors.push({
345
- field: 'memory',
346
- message: `Total memory size (${stats.totalSize}) exceeds limit (${MEMORY_CONSTANTS.MAX_MEMORY_SIZE})`,
347
- severity: 'error'
348
- });
349
- }
350
- // Update validity based on our checks
351
- return {
352
- ...result,
353
- valid: result.errors.length === 0
354
- };
355
- }
356
- /**
357
- * Serialize memory to string
358
- */
359
- serialize() {
360
- const data = {
361
- id: this.id,
362
- type: this.type,
363
- version: this.version,
364
- metadata: this.metadata,
365
- extensions: this.extensions,
366
- entries: Array.from(this.entries.values())
367
- };
368
- return JSON.stringify(data, null, 2);
369
- }
370
- /**
371
- * Deserialize memory from string
372
- * SECURITY: Validates all loaded data
373
- */
374
- deserialize(data) {
375
- try {
376
- const parsed = JSON.parse(data);
377
- // Validate basic structure
378
- if (!parsed.id || !parsed.type || parsed.type !== ElementType.MEMORY) {
379
- throw new Error('Invalid memory data format');
380
- }
381
- // Update properties
382
- this.id = parsed.id;
383
- this.version = parsed.version || '1.0.0';
384
- this.metadata = parsed.metadata || {};
385
- this.extensions = parsed.extensions || {};
386
- // Clear and reload entries
387
- this.entries.clear();
388
- if (Array.isArray(parsed.entries)) {
389
- for (const entry of parsed.entries) {
390
- if (this.isValidEntry(entry)) {
391
- // Re-sanitize on load
392
- entry.content = sanitizeMemoryContent(entry.content, MEMORY_CONSTANTS.MAX_ENTRY_SIZE);
393
- entry.tags = this.sanitizeTags(entry.tags || []);
394
- entry.timestamp = new Date(entry.timestamp);
395
- if (entry.expiresAt) {
396
- entry.expiresAt = new Date(entry.expiresAt);
397
- }
398
- this.entries.set(entry.id, entry);
399
- }
400
- }
401
- }
402
- // Enforce retention policy after loading
403
- this.enforceRetentionPolicy();
404
- }
405
- catch (error) {
406
- SecurityMonitor.logSecurityEvent({
407
- type: MEMORY_SECURITY_EVENTS.MEMORY_DESERIALIZE_FAILED,
408
- severity: 'HIGH',
409
- source: 'Memory.deserialize',
410
- details: `Failed to deserialize memory: ${error}`
411
- });
412
- throw new Error(`Failed to deserialize memory: ${error}`);
413
- }
414
- }
415
- // Private helper methods
416
- calculateExpiryDate() {
417
- const expiry = new Date();
418
- expiry.setDate(expiry.getDate() + this.retentionDays);
419
- return expiry;
420
- }
421
- sanitizeTags(tags) {
422
- // SECURITY FIX: Limit number of tags and sanitize each
423
- const limitedTags = tags.slice(0, MEMORY_CONSTANTS.MAX_TAGS_PER_ENTRY);
424
- return limitedTags
425
- .map(tag => {
426
- const normalized = UnicodeValidator.normalize(tag).normalizedContent;
427
- return sanitizeInput(normalized, MEMORY_CONSTANTS.MAX_TAG_LENGTH);
428
- })
429
- .filter(tag => tag && tag.length > 0);
430
- }
431
- sanitizeMetadata(metadata) {
432
- if (!metadata)
433
- return undefined;
434
- // SECURITY FIX: Sanitize metadata values
435
- const sanitized = {};
436
- const maxKeys = MEMORY_CONSTANTS.MAX_METADATA_KEYS;
437
- let keyCount = 0;
438
- for (const [key, value] of Object.entries(metadata)) {
439
- if (keyCount >= maxKeys)
440
- break;
441
- const sanitizedKey = sanitizeInput(key, MEMORY_CONSTANTS.MAX_METADATA_KEY_LENGTH);
442
- if (sanitizedKey && typeof value === 'string') {
443
- sanitized[sanitizedKey] = sanitizeInput(value, MEMORY_CONSTANTS.MAX_METADATA_VALUE_LENGTH);
444
- keyCount++;
445
- }
446
- else if (sanitizedKey && typeof value === 'number') {
447
- sanitized[sanitizedKey] = value;
448
- keyCount++;
449
- }
450
- // Skip other types for security
451
- }
452
- return sanitized;
453
- }
454
- canAccessPrivacyLevel(entryLevel, requestedLevel) {
455
- const levels = MEMORY_CONSTANTS.PRIVACY_LEVELS;
456
- const entryIndex = levels.indexOf(entryLevel);
457
- const requestedIndex = levels.indexOf(requestedLevel);
458
- // Can only access entries at or below the requested privacy level
459
- // e.g., if requesting 'private', can see 'public' and 'private' but not 'sensitive'
460
- return entryIndex <= requestedIndex;
461
- }
462
- isValidEntry(entry) {
463
- return entry &&
464
- typeof entry.id === 'string' &&
465
- typeof entry.content === 'string' &&
466
- entry.timestamp &&
467
- (!entry.tags || Array.isArray(entry.tags));
468
- }
469
- }
470
- //# sourceMappingURL=data:application/json;base64,{"version":3,"file":"Memory.js","sourceRoot":"","sources":["../../../src/elements/memories/Memory.ts"],"names":[],"mappings":"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","sourcesContent":["/**\n * Memory Element - Persistent context storage for continuity and learning\n * \n * Provides multiple storage backends, retention policies, and search capabilities\n * for maintaining context across sessions and interactions.\n * \n * SECURITY MEASURES IMPLEMENTED:\n * 1. Input sanitization for all memory content\n * 2. Memory size limits to prevent unbounded growth\n * 3. Path validation for file-based storage\n * 4. Retention policy enforcement\n * 5. Privacy level access control\n * 6. Audit logging for all operations\n */\n\nimport { BaseElement } from '../BaseElement.js';\nimport { IElement, ElementValidationResult, ValidationError } from '../../types/elements/index.js';\nimport { ElementType } from '../../portfolio/types.js';\nimport { IElementMetadata } from '../../types/elements/IElement.js';\nimport { UnicodeValidator } from '../../security/validators/unicodeValidator.js';\nimport { SecurityMonitor } from '../../security/securityMonitor.js';\nimport { sanitizeInput } from '../../security/InputValidator.js';\nimport { MEMORY_CONSTANTS, MEMORY_SECURITY_EVENTS, PrivacyLevel, StorageBackend } from './constants.js';\nimport DOMPurify from 'dompurify';\nimport { JSDOM } from 'jsdom';\nimport * as path from 'path';\n\n// Initialize DOMPurify with JSDOM\nconst window = new JSDOM('').window;\nconst purify = DOMPurify(window as any);\n\n// Configure DOMPurify for memory content - strip all HTML but keep text\npurify.setConfig({\n  ALLOWED_TAGS: [],  // No HTML tags allowed\n  ALLOWED_ATTR: [],  // No attributes allowed\n  KEEP_CONTENT: true // Keep text content\n});\n\n/**\n * Sanitize content for memory storage\n * More permissive than sanitizeInput - allows punctuation, quotes, etc.\n * but still prevents XSS and control characters\n */\nfunction sanitizeMemoryContent(content: string, maxLength: number): string {\n  if (!content || typeof content !== 'string') {\n    return '';\n  }\n  \n  // First normalize Unicode\n  const normalized = UnicodeValidator.normalize(content).normalizedContent;\n  \n  // Use DOMPurify to strip any HTML/XSS attempts but keep text\n  const cleaned = purify.sanitize(normalized);\n  \n  // Remove only control characters and null bytes\n  return cleaned\n    .replace(/[\\x00-\\x08\\x0B\\x0C\\x0E-\\x1F\\x7F]/g, '') // Remove control chars except \\t \\n \\r\n    .substring(0, maxLength)\n    .trim();\n}\n\nexport interface MemoryMetadata extends IElementMetadata {\n  storageBackend?: StorageBackend;\n  retentionDays?: number;\n  privacyLevel?: PrivacyLevel;\n  searchable?: boolean;\n  maxEntries?: number;\n  encryptionEnabled?: boolean;\n}\n\nexport interface MemoryEntry {\n  id: string;\n  timestamp: Date;\n  content: string;\n  tags?: string[];\n  metadata?: Record<string, any>;\n  expiresAt?: Date;\n  privacyLevel?: PrivacyLevel;\n}\n\nexport interface MemorySearchOptions {\n  query?: string;\n  tags?: string[];\n  startDate?: Date;\n  endDate?: Date;\n  limit?: number;\n  privacyLevel?: PrivacyLevel;\n}\n\nexport class Memory extends BaseElement implements IElement {\n  // Memory-specific properties\n  private entries: Map<string, MemoryEntry> = new Map();\n  private storageBackend: StorageBackend;\n  private retentionDays: number;\n  private privacyLevel: PrivacyLevel;\n  private searchable: boolean;\n  private maxEntries: number;\n  \n  constructor(metadata: Partial<MemoryMetadata> = {}) {\n    // SECURITY FIX: Sanitize all inputs during construction\n    const sanitizedMetadata = {\n      ...metadata,\n      name: metadata.name ? \n        sanitizeInput(UnicodeValidator.normalize(metadata.name).normalizedContent, 100) : \n        'Unnamed Memory',\n      description: metadata.description ? \n        sanitizeInput(UnicodeValidator.normalize(metadata.description).normalizedContent, 500) : \n        undefined\n    };\n    \n    super(ElementType.MEMORY, sanitizedMetadata);\n    \n    // Initialize memory-specific properties with defaults\n    this.storageBackend = metadata.storageBackend || MEMORY_CONSTANTS.DEFAULT_STORAGE_BACKEND;\n    this.retentionDays = metadata.retentionDays || MEMORY_CONSTANTS.DEFAULT_RETENTION_DAYS;\n    // Validate privacy level - default to private if invalid\n    this.privacyLevel = (metadata.privacyLevel && MEMORY_CONSTANTS.PRIVACY_LEVELS.includes(metadata.privacyLevel)) \n      ? metadata.privacyLevel \n      : MEMORY_CONSTANTS.DEFAULT_PRIVACY_LEVEL;\n    this.searchable = metadata.searchable !== false;\n    this.maxEntries = Math.min(\n      metadata.maxEntries || MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT,\n      MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT\n    );\n    \n    // Set up extensions\n    this.extensions = {\n      storageBackend: this.storageBackend,\n      retentionDays: this.retentionDays,\n      privacyLevel: this.privacyLevel,\n      searchable: this.searchable,\n      maxEntries: this.maxEntries,\n      encryptionEnabled: metadata.encryptionEnabled || false\n    };\n    \n    // Log memory creation\n    SecurityMonitor.logSecurityEvent({\n      type: MEMORY_SECURITY_EVENTS.MEMORY_CREATED,\n      severity: 'LOW',\n      source: 'Memory.constructor',\n      details: `Memory created: ${this.metadata.name} with ${this.storageBackend} backend`\n    });\n  }\n  \n  /**\n   * Add a new memory entry\n   * SECURITY: Validates and sanitizes all input, enforces size limits\n   */\n  public async addEntry(content: string, tags?: string[], metadata?: Record<string, any>): Promise<MemoryEntry> {\n    // Validate memory size limits\n    if (this.entries.size >= this.maxEntries) {\n      // SECURITY FIX: Enforce retention policy when at capacity\n      await this.enforceRetentionPolicy();\n      \n      // If still at capacity after retention, remove oldest to make room\n      if (this.entries.size >= this.maxEntries) {\n        const oldestEntry = Array.from(this.entries.values())\n          .sort((a, b) => a.timestamp.getTime() - b.timestamp.getTime())[0];\n        if (oldestEntry) {\n          this.entries.delete(oldestEntry.id);\n        }\n      }\n    }\n    \n    // SECURITY FIX: Validate and sanitize content\n    const sanitizedContent = sanitizeMemoryContent(content, MEMORY_CONSTANTS.MAX_ENTRY_SIZE);\n    \n    if (!sanitizedContent || sanitizedContent.trim().length === 0) {\n      throw new Error('Memory content cannot be empty');\n    }\n    \n    // SECURITY FIX: Validate and sanitize tags\n    const sanitizedTags = tags ? this.sanitizeTags(tags) : [];\n    \n    // Create memory entry\n    const entry: MemoryEntry = {\n      id: `mem_${Date.now()}_${Math.random().toString(36).substr(2, 9)}`,\n      timestamp: new Date(),\n      content: sanitizedContent,\n      tags: sanitizedTags,\n      metadata: this.sanitizeMetadata(metadata),\n      privacyLevel: this.privacyLevel,\n      expiresAt: this.calculateExpiryDate()\n    };\n    \n    // Store entry\n    this.entries.set(entry.id, entry);\n    this._isDirty = true;\n    \n    // Log memory addition\n    SecurityMonitor.logSecurityEvent({\n      type: MEMORY_SECURITY_EVENTS.MEMORY_ADDED,\n      severity: 'LOW',\n      source: 'Memory.addEntry',\n      details: `Added memory entry ${entry.id} with ${sanitizedTags.length} tags`\n    });\n    \n    return entry;\n  }\n  \n  /**\n   * Search memory entries\n   * SECURITY: Respects privacy levels and sanitizes search queries\n   */\n  public async search(options: MemorySearchOptions = {}): Promise<MemoryEntry[]> {\n    // SECURITY FIX: Sanitize search query (use regular sanitizeInput for queries)\n    const sanitizedQuery = options.query ? \n      sanitizeInput(UnicodeValidator.normalize(options.query).normalizedContent, 200) : \n      undefined;\n    \n    // PERFORMANCE OPTIMIZATION: Single-pass filtering to reduce allocations\n    let results: MemoryEntry[] = [];\n    const queryLower = sanitizedQuery?.toLowerCase();\n    const searchTags = options.tags && options.tags.length > 0 ? this.sanitizeTags(options.tags) : null;\n    \n    // Single iteration through entries with all filters applied\n    for (const entry of this.entries.values()) {\n      // Privacy level check\n      if (options.privacyLevel && \n          !this.canAccessPrivacyLevel(entry.privacyLevel || MEMORY_CONSTANTS.DEFAULT_PRIVACY_LEVEL, options.privacyLevel)) {\n        continue;\n      }\n      \n      // Query text check\n      if (queryLower) {\n        const contentMatch = entry.content.toLowerCase().includes(queryLower);\n        const tagMatch = entry.tags?.some(tag => tag.toLowerCase().includes(queryLower));\n        if (!contentMatch && !tagMatch) {\n          continue;\n        }\n      }\n      \n      // Tag filter check\n      if (searchTags && !searchTags.some(searchTag => entry.tags?.includes(searchTag))) {\n        continue;\n      }\n      \n      // Date range checks\n      if (options.startDate && entry.timestamp < options.startDate) {\n        continue;\n      }\n      if (options.endDate && entry.timestamp > options.endDate) {\n        continue;\n      }\n      \n      // Entry passes all filters\n      results.push(entry);\n    }\n    \n    // Sort by timestamp (newest first) - using string comparison for IDs as secondary sort\n    results.sort((a, b) => {\n      const timeDiff = b.timestamp.getTime() - a.timestamp.getTime();\n      if (timeDiff !== 0) return timeDiff;\n      // If timestamps are exactly the same, sort by ID (which contains timestamp)\n      return b.id.localeCompare(a.id);\n    });\n    \n    // Apply limit\n    if (options.limit && options.limit > 0) {\n      results = results.slice(0, options.limit);\n    }\n    \n    // Log search operation\n    SecurityMonitor.logSecurityEvent({\n      type: MEMORY_SECURITY_EVENTS.MEMORY_SEARCHED,\n      severity: 'LOW',\n      source: 'Memory.search',\n      details: `Searched memories with query: ${sanitizedQuery || 'none'}, found ${results.length} results`\n    });\n    \n    return results;\n  }\n  \n  /**\n   * Get a specific memory entry by ID\n   */\n  public async getEntry(id: string): Promise<MemoryEntry | undefined> {\n    return this.entries.get(id);\n  }\n  \n  /**\n   * Delete a memory entry\n   * SECURITY: Validates permissions and logs deletion\n   */\n  public async deleteEntry(id: string): Promise<boolean> {\n    const entry = this.entries.get(id);\n    if (!entry) {\n      return false;\n    }\n    \n    // SECURITY: Check if sensitive memories can be deleted\n    if (entry.privacyLevel === 'sensitive') {\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.SENSITIVE_MEMORY_DELETED,\n        severity: 'MEDIUM',\n        source: 'Memory.deleteEntry',\n        details: `Sensitive memory ${id} deleted`\n      });\n    }\n    \n    this.entries.delete(id);\n    this._isDirty = true;\n    \n    return true;\n  }\n  \n  /**\n   * Enforce retention policy by removing expired entries\n   * SECURITY: Ensures memory doesn't grow unbounded\n   */\n  public async enforceRetentionPolicy(): Promise<number> {\n    const now = new Date();\n    let deletedCount = 0;\n    \n    // Remove expired entries\n    for (const [id, entry] of this.entries) {\n      if (entry.expiresAt && entry.expiresAt < now) {\n        this.entries.delete(id);\n        deletedCount++;\n      }\n    }\n    \n    // If still at or over capacity, remove oldest entries to make room for one more\n    if (this.entries.size >= this.maxEntries) {\n      const sortedEntries = Array.from(this.entries.entries())\n        .sort((a, b) => a[1].timestamp.getTime() - b[1].timestamp.getTime());\n      \n      // Remove one extra to make room for new entry\n      const toDelete = Math.max(1, this.entries.size - this.maxEntries + 1);\n      for (let i = 0; i < toDelete && i < sortedEntries.length; i++) {\n        this.entries.delete(sortedEntries[i][0]);\n        deletedCount++;\n      }\n    }\n    \n    if (deletedCount > 0) {\n      this._isDirty = true;\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.RETENTION_POLICY_ENFORCED,\n        severity: 'LOW',\n        source: 'Memory.enforceRetentionPolicy',\n        details: `Removed ${deletedCount} expired memories`\n      });\n    }\n    \n    return deletedCount;\n  }\n  \n  /**\n   * Clear all memory entries\n   * SECURITY: Requires confirmation and logs the action\n   */\n  public async clearAll(confirm: boolean = false): Promise<void> {\n    if (!confirm) {\n      throw new Error('Memory clear requires confirmation');\n    }\n    \n    const count = this.entries.size;\n    this.entries.clear();\n    this._isDirty = true;\n    \n    SecurityMonitor.logSecurityEvent({\n      type: MEMORY_SECURITY_EVENTS.MEMORY_CLEARED,\n      severity: 'HIGH',\n      source: 'Memory.clearAll',\n      details: `Cleared all ${count} memory entries`\n    });\n  }\n  \n  /**\n   * Get memory statistics\n   */\n  public getStats(): {\n    totalEntries: number;\n    totalSize: number;\n    oldestEntry?: Date;\n    newestEntry?: Date;\n    tagFrequency: Map<string, number>;\n  } {\n    let totalSize = 0;\n    let oldestEntry: Date | undefined;\n    let newestEntry: Date | undefined;\n    const tagFrequency = new Map<string, number>();\n    \n    for (const entry of this.entries.values()) {\n      totalSize += entry.content.length;\n      \n      if (!oldestEntry || entry.timestamp < oldestEntry) {\n        oldestEntry = entry.timestamp;\n      }\n      if (!newestEntry || entry.timestamp > newestEntry) {\n        newestEntry = entry.timestamp;\n      }\n      \n      entry.tags?.forEach(tag => {\n        tagFrequency.set(tag, (tagFrequency.get(tag) || 0) + 1);\n      });\n    }\n    \n    return {\n      totalEntries: this.entries.size,\n      totalSize,\n      oldestEntry,\n      newestEntry,\n      tagFrequency\n    };\n  }\n  \n  /**\n   * Validate the memory element\n   */\n  public override validate(): ElementValidationResult {\n    const result = super.validate();\n    \n    // Initialize errors array if not present\n    if (!result.errors) {\n      result.errors = [];\n    }\n    \n    // Additional memory-specific validation\n    if (this.retentionDays < MEMORY_CONSTANTS.MIN_RETENTION_DAYS || this.retentionDays > MEMORY_CONSTANTS.MAX_RETENTION_DAYS) {\n      result.errors.push({\n        field: 'retentionDays',\n        message: `Retention days must be between ${MEMORY_CONSTANTS.MIN_RETENTION_DAYS} and ${MEMORY_CONSTANTS.MAX_RETENTION_DAYS}`,\n        severity: 'error'\n      } as ValidationError);\n    }\n    \n    if (this.maxEntries < 1 || this.maxEntries > MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT) {\n      result.errors.push({\n        field: 'maxEntries',\n        message: `Max entries must be between 1 and ${MEMORY_CONSTANTS.MAX_ENTRIES_DEFAULT}`,\n        severity: 'error'\n      } as ValidationError);\n    }\n    \n    // Check memory size\n    const stats = this.getStats();\n    if (stats.totalSize > MEMORY_CONSTANTS.MAX_MEMORY_SIZE) {\n      result.errors.push({\n        field: 'memory',\n        message: `Total memory size (${stats.totalSize}) exceeds limit (${MEMORY_CONSTANTS.MAX_MEMORY_SIZE})`,\n        severity: 'error'\n      } as ValidationError);\n    }\n    \n    // Update validity based on our checks\n    return {\n      ...result,\n      valid: result.errors.length === 0\n    };\n  }\n  \n  /**\n   * Serialize memory to string\n   */\n  public override serialize(): string {\n    const data = {\n      id: this.id,\n      type: this.type,\n      version: this.version,\n      metadata: this.metadata,\n      extensions: this.extensions,\n      entries: Array.from(this.entries.values())\n    };\n    \n    return JSON.stringify(data, null, 2);\n  }\n  \n  /**\n   * Deserialize memory from string\n   * SECURITY: Validates all loaded data\n   */\n  public override deserialize(data: string): void {\n    try {\n      const parsed = JSON.parse(data);\n      \n      // Validate basic structure\n      if (!parsed.id || !parsed.type || parsed.type !== ElementType.MEMORY) {\n        throw new Error('Invalid memory data format');\n      }\n      \n      // Update properties\n      this.id = parsed.id;\n      this.version = parsed.version || '1.0.0';\n      this.metadata = parsed.metadata || {};\n      this.extensions = parsed.extensions || {};\n      \n      // Clear and reload entries\n      this.entries.clear();\n      if (Array.isArray(parsed.entries)) {\n        for (const entry of parsed.entries) {\n          if (this.isValidEntry(entry)) {\n            // Re-sanitize on load\n            entry.content = sanitizeMemoryContent(entry.content, MEMORY_CONSTANTS.MAX_ENTRY_SIZE);\n            entry.tags = this.sanitizeTags(entry.tags || []);\n            entry.timestamp = new Date(entry.timestamp);\n            if (entry.expiresAt) {\n              entry.expiresAt = new Date(entry.expiresAt);\n            }\n            this.entries.set(entry.id, entry);\n          }\n        }\n      }\n      \n      // Enforce retention policy after loading\n      this.enforceRetentionPolicy();\n      \n    } catch (error) {\n      SecurityMonitor.logSecurityEvent({\n        type: MEMORY_SECURITY_EVENTS.MEMORY_DESERIALIZE_FAILED,\n        severity: 'HIGH',\n        source: 'Memory.deserialize',\n        details: `Failed to deserialize memory: ${error}`\n      });\n      throw new Error(`Failed to deserialize memory: ${error}`);\n    }\n  }\n  \n  // Private helper methods\n  \n  private calculateExpiryDate(): Date {\n    const expiry = new Date();\n    expiry.setDate(expiry.getDate() + this.retentionDays);\n    return expiry;\n  }\n  \n  private sanitizeTags(tags: string[]): string[] {\n    // SECURITY FIX: Limit number of tags and sanitize each\n    const limitedTags = tags.slice(0, MEMORY_CONSTANTS.MAX_TAGS_PER_ENTRY);\n    \n    return limitedTags\n      .map(tag => {\n        const normalized = UnicodeValidator.normalize(tag).normalizedContent;\n        return sanitizeInput(normalized, MEMORY_CONSTANTS.MAX_TAG_LENGTH);\n      })\n      .filter(tag => tag && tag.length > 0);\n  }\n  \n  private sanitizeMetadata(metadata?: Record<string, any>): Record<string, any> | undefined {\n    if (!metadata) return undefined;\n    \n    // SECURITY FIX: Sanitize metadata values\n    const sanitized: Record<string, any> = {};\n    const maxKeys = MEMORY_CONSTANTS.MAX_METADATA_KEYS;\n    let keyCount = 0;\n    \n    for (const [key, value] of Object.entries(metadata)) {\n      if (keyCount >= maxKeys) break;\n      \n      const sanitizedKey = sanitizeInput(key, MEMORY_CONSTANTS.MAX_METADATA_KEY_LENGTH);\n      if (sanitizedKey && typeof value === 'string') {\n        sanitized[sanitizedKey] = sanitizeInput(value, MEMORY_CONSTANTS.MAX_METADATA_VALUE_LENGTH);\n        keyCount++;\n      } else if (sanitizedKey && typeof value === 'number') {\n        sanitized[sanitizedKey] = value;\n        keyCount++;\n      }\n      // Skip other types for security\n    }\n    \n    return sanitized;\n  }\n  \n  private canAccessPrivacyLevel(entryLevel: string, requestedLevel: string): boolean {\n    const levels = MEMORY_CONSTANTS.PRIVACY_LEVELS;\n    const entryIndex = levels.indexOf(entryLevel as PrivacyLevel);\n    const requestedIndex = levels.indexOf(requestedLevel as PrivacyLevel);\n    \n    // Can only access entries at or below the requested privacy level\n    // e.g., if requesting 'private', can see 'public' and 'private' but not 'sensitive'\n    return entryIndex <= requestedIndex;\n  }\n  \n  private isValidEntry(entry: any): boolean {\n    return entry &&\n      typeof entry.id === 'string' &&\n      typeof entry.content === 'string' &&\n      entry.timestamp &&\n      (!entry.tags || Array.isArray(entry.tags));\n  }\n}"]}
@@ -1,86 +0,0 @@
1
- /**
2
- * MemoryManager - Implementation of IElementManager for Memory elements
3
- * Handles CRUD operations and lifecycle management for memories implementing IElement
4
- *
5
- * FIXES IMPLEMENTED:
6
- * 1. CRITICAL: Fixed race conditions in file operations by using FileLockManager for atomic reads/writes
7
- * 2. HIGH: Fixed unvalidated YAML parsing vulnerability by using SecureYamlParser
8
- * 3. MEDIUM: All user inputs are now validated and sanitized
9
- * 4. MEDIUM: Audit logging added for security operations
10
- * 5. MEDIUM: Path validation prevents directory traversal attacks
11
- */
12
- import { Memory, MemoryMetadata } from './Memory.js';
13
- import { IElementManager } from '../../types/elements/IElementManager.js';
14
- import { ElementValidationResult } from '../../types/elements/IElement.js';
15
- import { ElementType } from '../../portfolio/types.js';
16
- export declare class MemoryManager implements IElementManager<Memory> {
17
- private portfolioManager;
18
- private memoriesDir;
19
- private memoryCache;
20
- constructor();
21
- /**
22
- * Load a memory from file
23
- * SECURITY FIX #1: Uses FileLockManager.atomicReadFile() instead of fs.readFile()
24
- * to prevent race conditions and ensure atomic file operations
25
- */
26
- load(filePath: string): Promise<Memory>;
27
- /**
28
- * Save a memory to file
29
- * SECURITY FIX #1: Uses FileLockManager.atomicWriteFile() for atomic operations
30
- */
31
- save(element: Memory, filePath: string): Promise<void>;
32
- /**
33
- * List all available memories
34
- */
35
- list(): Promise<Memory[]>;
36
- /**
37
- * Find memories matching a predicate
38
- */
39
- find(predicate: (element: Memory) => boolean): Promise<Memory | undefined>;
40
- /**
41
- * Find multiple memories matching a predicate
42
- */
43
- findMany(predicate: (element: Memory) => boolean): Promise<Memory[]>;
44
- /**
45
- * Delete a memory file
46
- * SECURITY: Validates path and logs deletion
47
- */
48
- delete(filePath: string): Promise<void>;
49
- /**
50
- * Check if a memory file exists
51
- */
52
- exists(filePath: string): Promise<boolean>;
53
- /**
54
- * Create a new memory with metadata
55
- */
56
- create(metadata: Partial<MemoryMetadata>): Promise<Memory>;
57
- /**
58
- * Import a memory from JSON/YAML string
59
- * SECURITY: Full validation of imported content
60
- */
61
- importElement(data: string, format?: 'json' | 'yaml'): Promise<Memory>;
62
- /**
63
- * Export a memory to YAML string
64
- */
65
- exportElement(element: Memory): Promise<string>;
66
- /**
67
- * Validate a memory element
68
- */
69
- validate(element: Memory): ElementValidationResult;
70
- /**
71
- * Validate and resolve a file path
72
- * SECURITY: Prevents directory traversal attacks
73
- */
74
- validatePath(filePath: string): boolean;
75
- /**
76
- * Get the element type this manager handles
77
- */
78
- getElementType(): ElementType;
79
- /**
80
- * Get the file extension for memory files
81
- */
82
- getFileExtension(): string;
83
- private validateAndResolvePath;
84
- private parseMemoryFile;
85
- }
86
- //# sourceMappingURL=MemoryManager.d.ts.map
@@ -1 +0,0 @@
1
- {"version":3,"file":"MemoryManager.d.ts","sourceRoot":"","sources":["../../../src/elements/memories/MemoryManager.ts"],"names":[],"mappings":"AAAA;;;;;;;;;;GAUG;AAEH,OAAO,EAAE,MAAM,EAAE,cAAc,EAAe,MAAM,aAAa,CAAC;AAClE,OAAO,EAAE,eAAe,EAAE,MAAM,yCAAyC,CAAC;AAC1E,OAAO,EAAE,uBAAuB,EAAE,MAAM,kCAAkC,CAAC;AAC3E,OAAO,EAAE,WAAW,EAAE,MAAM,0BAA0B,CAAC;AAYvD,qBAAa,aAAc,YAAW,eAAe,CAAC,MAAM,CAAC;IAC3D,OAAO,CAAC,gBAAgB,CAAmB;IAC3C,OAAO,CAAC,WAAW,CAAS;IAC5B,OAAO,CAAC,WAAW,CAAkC;;IAOrD;;;;OAIG;IACG,IAAI,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAkE7C;;;OAGG;IACG,IAAI,CAAC,OAAO,EAAE,MAAM,EAAE,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IAoE5D;;OAEG;IACG,IAAI,IAAI,OAAO,CAAC,MAAM,EAAE,CAAC;IAiC/B;;OAEG;IACG,IAAI,CAAC,SAAS,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,OAAO,GAAG,OAAO,CAAC,MAAM,GAAG,SAAS,CAAC;IAKhF;;OAEG;IACG,QAAQ,CAAC,SAAS,EAAE,CAAC,OAAO,EAAE,MAAM,KAAK,OAAO,GAAG,OAAO,CAAC,MAAM,EAAE,CAAC;IAK1E;;;OAGG;IACG,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,IAAI,CAAC;IA6B7C;;OAEG;IACG,MAAM,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO,CAAC,OAAO,CAAC;IAUhD;;OAEG;IACG,MAAM,CAAC,QAAQ,EAAE,OAAO,CAAC,cAAc,CAAC,GAAG,OAAO,CAAC,MAAM,CAAC;IAIhE;;;OAGG;IACG,aAAa,CAAC,IAAI,EAAE,MAAM,EAAE,MAAM,GAAE,MAAM,GAAG,MAAe,GAAG,OAAO,CAAC,MAAM,CAAC;IA0EpF;;OAEG;IACG,aAAa,CAAC,OAAO,EAAE,MAAM,GAAG,OAAO,CAAC,MAAM,CAAC;IAuBrD;;OAEG;IACH,QAAQ,CAAC,OAAO,EAAE,MAAM,GAAG,uBAAuB;IAIlD;;;OAGG;IACH,YAAY,CAAC,QAAQ,EAAE,MAAM,GAAG,OAAO;IAqBvC;;OAEG;IACH,cAAc,IAAI,WAAW;IAI7B;;OAEG;IACH,gBAAgB,IAAI,MAAM;YAMZ,sBAAsB;IAgCpC,OAAO,CAAC,eAAe;CA0BxB"}