claude-flow-novice 1.1.8 → 1.2.0

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 (88) hide show
  1. package/.claude/SLASH-COMMANDS-READY.md +53 -0
  2. package/.claude/WORKING-SETUP.md +67 -0
  3. package/.claude/commands/README.md +157 -0
  4. package/.claude/commands/claude-md.js +237 -0
  5. package/.claude/commands/claude-md.md +64 -0
  6. package/.claude/commands/claude-soul.js +562 -0
  7. package/.claude/commands/claude-soul.md +22 -0
  8. package/.claude/commands/cli-integration.js +216 -0
  9. package/.claude/commands/dependency-recommendations.md +171 -0
  10. package/.claude/commands/github.js +638 -0
  11. package/.claude/commands/github.md +221 -0
  12. package/.claude/commands/hooks.js +648 -0
  13. package/.claude/commands/hooks.md +38 -0
  14. package/.claude/commands/index.js +115 -0
  15. package/.claude/commands/neural.js +572 -0
  16. package/.claude/commands/neural.md +39 -0
  17. package/.claude/commands/performance.js +582 -0
  18. package/.claude/commands/performance.md +41 -0
  19. package/.claude/commands/register-all-commands.js +314 -0
  20. package/.claude/commands/register-claude-md.js +82 -0
  21. package/.claude/commands/register-claude-soul.js +80 -0
  22. package/.claude/commands/sparc.js +110 -0
  23. package/.claude/commands/sparc.md +46 -0
  24. package/.claude/commands/suggest-improvements.md +95 -0
  25. package/.claude/commands/suggest-templates.md +147 -0
  26. package/.claude/commands/swarm.js +423 -0
  27. package/.claude/commands/swarm.md +24 -0
  28. package/.claude/commands/validate-commands.js +223 -0
  29. package/.claude/commands/workflow.js +606 -0
  30. package/.claude/commands/workflow.md +295 -0
  31. package/.claude/core/agent-manager.js +80 -0
  32. package/.claude/core/agent-manager.js.map +1 -0
  33. package/.claude/core/config.js +1221 -0
  34. package/.claude/core/config.js.map +1 -0
  35. package/.claude/core/event-bus.js +136 -0
  36. package/.claude/core/event-bus.js.map +1 -0
  37. package/.claude/core/index.js +6 -0
  38. package/.claude/core/index.js.map +1 -0
  39. package/.claude/core/json-persistence.js +112 -0
  40. package/.claude/core/json-persistence.js.map +1 -0
  41. package/.claude/core/logger.js +245 -0
  42. package/.claude/core/logger.js.map +1 -0
  43. package/.claude/core/orchestrator-fixed.js +236 -0
  44. package/.claude/core/orchestrator-fixed.js.map +1 -0
  45. package/.claude/core/orchestrator.js +1136 -0
  46. package/.claude/core/orchestrator.js.map +1 -0
  47. package/.claude/core/persistence.js +185 -0
  48. package/.claude/core/persistence.js.map +1 -0
  49. package/.claude/core/project-manager.js +80 -0
  50. package/.claude/core/project-manager.js.map +1 -0
  51. package/.claude/core/slash-command.js +24 -0
  52. package/.claude/core/version.js +35 -0
  53. package/.claude/core/version.js.map +1 -0
  54. package/.claude/slash-commands.json +92 -0
  55. package/dist/mcp/mcp-server-novice.js +24 -4
  56. package/dist/mcp/mcp-server-sdk.js +649 -0
  57. package/dist/mcp/mcp-server-with-slash-commands.js +776 -0
  58. package/dist/src/cli/simple-commands/hooks/session-start-soul.js +271 -0
  59. package/dist/src/slash-commands/README.md +157 -0
  60. package/dist/src/slash-commands/claude-md.js +237 -0
  61. package/dist/src/slash-commands/claude-soul.js +562 -0
  62. package/dist/src/slash-commands/cli-integration.js +216 -0
  63. package/dist/src/slash-commands/github.js +638 -0
  64. package/dist/src/slash-commands/hooks.js +648 -0
  65. package/dist/src/slash-commands/index.js +115 -0
  66. package/dist/src/slash-commands/mcp-slash-integration.js +146 -0
  67. package/dist/src/slash-commands/neural.js +572 -0
  68. package/dist/src/slash-commands/performance.js +582 -0
  69. package/dist/src/slash-commands/register-all-commands.js +314 -0
  70. package/dist/src/slash-commands/register-claude-md.js +82 -0
  71. package/dist/src/slash-commands/register-claude-soul.js +80 -0
  72. package/dist/src/slash-commands/sparc.js +110 -0
  73. package/dist/src/slash-commands/swarm.js +423 -0
  74. package/dist/src/slash-commands/validate-commands.js +223 -0
  75. package/dist/src/slash-commands/workflow.js +606 -0
  76. package/package.json +23 -10
  77. package/src/slash-commands/cli-integration.js +216 -0
  78. package/src/slash-commands/github.js +638 -0
  79. package/src/slash-commands/hooks.js +648 -0
  80. package/src/slash-commands/index.js +115 -0
  81. package/src/slash-commands/mcp-slash-integration.js +146 -0
  82. package/src/slash-commands/neural.js +572 -0
  83. package/src/slash-commands/performance.js +582 -0
  84. package/src/slash-commands/register-all-commands.js +314 -0
  85. package/src/slash-commands/sparc.js +110 -0
  86. package/src/slash-commands/swarm.js +423 -0
  87. package/src/slash-commands/validate-commands.js +223 -0
  88. package/src/slash-commands/workflow.js +606 -0
@@ -0,0 +1,1221 @@
1
+ /**
2
+ * Enterprise Configuration Management for Claude-Flow
3
+ * Features: Security masking, change tracking, multi-format support, credential management
4
+ */ import { promises as fs } from 'fs';
5
+ import { join } from 'path';
6
+ import { homedir } from 'os';
7
+ import { createHash, randomBytes, createCipheriv, createDecipheriv } from 'crypto';
8
+ import { safeParseJSON } from '../utils/helpers.js';
9
+ import { ConfigError, ValidationError } from '../utils/errors.js';
10
+ /**
11
+ * Security classifications for configuration paths
12
+ */ const SECURITY_CLASSIFICATIONS = {
13
+ credentials: {
14
+ level: 'secret',
15
+ encrypted: true
16
+ },
17
+ 'credentials.apiKey': {
18
+ level: 'secret',
19
+ maskPattern: '****...****',
20
+ encrypted: true
21
+ },
22
+ 'credentials.token': {
23
+ level: 'secret',
24
+ maskPattern: '****...****',
25
+ encrypted: true
26
+ },
27
+ 'credentials.password': {
28
+ level: 'secret',
29
+ maskPattern: '********',
30
+ encrypted: true
31
+ },
32
+ 'mcp.apiKey': {
33
+ level: 'confidential',
34
+ maskPattern: '****...****'
35
+ },
36
+ 'logging.destination': {
37
+ level: 'internal'
38
+ },
39
+ orchestrator: {
40
+ level: 'internal'
41
+ },
42
+ terminal: {
43
+ level: 'public'
44
+ }
45
+ };
46
+ /**
47
+ * Sensitive configuration paths that should be masked in output
48
+ */ const SENSITIVE_PATHS = [
49
+ 'credentials',
50
+ 'apiKey',
51
+ 'token',
52
+ 'password',
53
+ 'secret',
54
+ 'key',
55
+ 'auth'
56
+ ];
57
+ /**
58
+ * Format parsers for different configuration file types
59
+ */ const FORMAT_PARSERS = {
60
+ json: {
61
+ parse: JSON.parse,
62
+ stringify: (obj)=>JSON.stringify(obj, null, 2),
63
+ extension: '.json'
64
+ },
65
+ yaml: {
66
+ parse: (content)=>{
67
+ // Simple YAML parser for basic key-value pairs
68
+ const lines = content.split('\n');
69
+ const result = {};
70
+ let current = result;
71
+ const stack = [
72
+ result
73
+ ];
74
+ for (const line of lines){
75
+ const trimmed = line.trim();
76
+ if (!trimmed || trimmed.startsWith('#')) continue;
77
+ const indent = line.length - line.trimStart().length;
78
+ const colonIndex = trimmed.indexOf(':');
79
+ if (colonIndex === -1) continue;
80
+ const key = trimmed.substring(0, colonIndex).trim();
81
+ const value = trimmed.substring(colonIndex + 1).trim();
82
+ // Simple value parsing
83
+ let parsedValue = value;
84
+ if (value === 'true') parsedValue = true;
85
+ else if (value === 'false') parsedValue = false;
86
+ else if (!isNaN(Number(value)) && value !== '') parsedValue = Number(value);
87
+ else if (value.startsWith('"') && value.endsWith('"')) {
88
+ parsedValue = value.slice(1, -1);
89
+ }
90
+ current[key] = parsedValue;
91
+ }
92
+ return result;
93
+ },
94
+ stringify: (obj)=>{
95
+ const stringify = (obj, indent = 0)=>{
96
+ const spaces = ' '.repeat(indent);
97
+ let result = '';
98
+ for (const [key, value] of Object.entries(obj)){
99
+ if (typeof value === 'object' && value !== null && !Array.isArray(value)) {
100
+ result += `${spaces}${key}:\n${stringify(value, indent + 1)}`;
101
+ } else {
102
+ const formattedValue = typeof value === 'string' ? `"${value}"` : String(value);
103
+ result += `${spaces}${key}: ${formattedValue}\n`;
104
+ }
105
+ }
106
+ return result;
107
+ };
108
+ return stringify(obj);
109
+ },
110
+ extension: '.yaml'
111
+ },
112
+ toml: {
113
+ parse: (content)=>{
114
+ // Simple TOML parser for basic sections and key-value pairs
115
+ const lines = content.split('\n');
116
+ const result = {};
117
+ let currentSection = result;
118
+ for (const line of lines){
119
+ const trimmed = line.trim();
120
+ if (!trimmed || trimmed.startsWith('#')) continue;
121
+ // Section header
122
+ if (trimmed.startsWith('[') && trimmed.endsWith(']')) {
123
+ const sectionName = trimmed.slice(1, -1);
124
+ currentSection = result[sectionName] = {};
125
+ continue;
126
+ }
127
+ // Key-value pair
128
+ const equalsIndex = trimmed.indexOf('=');
129
+ if (equalsIndex === -1) continue;
130
+ const key = trimmed.substring(0, equalsIndex).trim();
131
+ const value = trimmed.substring(equalsIndex + 1).trim();
132
+ // Simple value parsing
133
+ let parsedValue = value;
134
+ if (value === 'true') parsedValue = true;
135
+ else if (value === 'false') parsedValue = false;
136
+ else if (!isNaN(Number(value)) && value !== '') parsedValue = Number(value);
137
+ else if (value.startsWith('"') && value.endsWith('"')) {
138
+ parsedValue = value.slice(1, -1);
139
+ }
140
+ currentSection[key] = parsedValue;
141
+ }
142
+ return result;
143
+ },
144
+ stringify: (obj)=>{
145
+ let result = '';
146
+ for (const [section, values] of Object.entries(obj)){
147
+ if (typeof values === 'object' && values !== null && !Array.isArray(values)) {
148
+ result += `[${section}]\n`;
149
+ for (const [key, value] of Object.entries(values)){
150
+ const formattedValue = typeof value === 'string' ? `"${value}"` : String(value);
151
+ result += `${key} = ${formattedValue}\n`;
152
+ }
153
+ result += '\n';
154
+ }
155
+ }
156
+ return result;
157
+ },
158
+ extension: '.toml'
159
+ }
160
+ };
161
+ /**
162
+ * Default configuration values
163
+ */ const DEFAULT_CONFIG = {
164
+ orchestrator: {
165
+ maxConcurrentAgents: 10,
166
+ taskQueueSize: 100,
167
+ healthCheckInterval: 30000,
168
+ shutdownTimeout: 30000
169
+ },
170
+ terminal: {
171
+ type: 'auto',
172
+ poolSize: 5,
173
+ recycleAfter: 10,
174
+ healthCheckInterval: 60000,
175
+ commandTimeout: 300000
176
+ },
177
+ memory: {
178
+ backend: 'hybrid',
179
+ cacheSizeMB: 100,
180
+ syncInterval: 5000,
181
+ conflictResolution: 'crdt',
182
+ retentionDays: 30
183
+ },
184
+ coordination: {
185
+ maxRetries: 3,
186
+ retryDelay: 1000,
187
+ deadlockDetection: true,
188
+ resourceTimeout: 60000,
189
+ messageTimeout: 30000
190
+ },
191
+ mcp: {
192
+ transport: 'stdio',
193
+ port: 3000,
194
+ tlsEnabled: false
195
+ },
196
+ logging: {
197
+ level: 'info',
198
+ format: 'json',
199
+ destination: 'console'
200
+ },
201
+ credentials: {
202
+ },
203
+ security: {
204
+ encryptionEnabled: true,
205
+ auditLogging: true,
206
+ maskSensitiveValues: true,
207
+ allowEnvironmentOverrides: true
208
+ }
209
+ };
210
+ /**
211
+ * Configuration manager
212
+ */ export class ConfigManager {
213
+ static instance;
214
+ config;
215
+ configPath;
216
+ profiles = new Map();
217
+ currentProfile;
218
+ userConfigDir;
219
+ changeHistory = [];
220
+ encryptionKey;
221
+ validationRules = new Map();
222
+ formatParsers = FORMAT_PARSERS;
223
+ constructor(){
224
+ this.config = deepClone(DEFAULT_CONFIG);
225
+ this.userConfigDir = this.getUserConfigDir();
226
+ this.setupValidationRules();
227
+ // Encryption will be initialized via init() method
228
+ }
229
+ /**
230
+ * Gets the singleton instance
231
+ */ static getInstance() {
232
+ if (!ConfigManager.instance) {
233
+ ConfigManager.instance = new ConfigManager();
234
+ }
235
+ return ConfigManager.instance;
236
+ }
237
+ /**
238
+ * Initialize async components
239
+ */ async init() {
240
+ await this.initializeEncryption();
241
+ }
242
+ /**
243
+ * Initializes encryption for sensitive configuration values
244
+ */ async initializeEncryption() {
245
+ try {
246
+ const keyFile = join(this.userConfigDir, '.encryption-key');
247
+ // Check if key file exists (simplified for demo)
248
+ try {
249
+ await fs.access(keyFile);
250
+ // In a real implementation, this would be more secure
251
+ this.encryptionKey = randomBytes(32);
252
+ } catch {
253
+ this.encryptionKey = randomBytes(32);
254
+ // Store key securely (in production, use proper key management)
255
+ }
256
+ } catch (error) {
257
+ console.warn('Failed to initialize encryption:', error.message);
258
+ }
259
+ }
260
+ /**
261
+ * Sets up validation rules for configuration paths
262
+ */ setupValidationRules() {
263
+ // Orchestrator validation rules
264
+ this.validationRules.set('orchestrator.maxConcurrentAgents', {
265
+ type: 'number',
266
+ required: true,
267
+ min: 1,
268
+ max: 100,
269
+ validator: (value, config)=>{
270
+ if (value > config.terminal?.poolSize * 2) {
271
+ return 'maxConcurrentAgents should not exceed 2x terminal pool size';
272
+ }
273
+ return null;
274
+ }
275
+ });
276
+ this.validationRules.set('orchestrator.taskQueueSize', {
277
+ type: 'number',
278
+ required: true,
279
+ min: 1,
280
+ max: 10000,
281
+ dependencies: [
282
+ 'orchestrator.maxConcurrentAgents'
283
+ ],
284
+ validator: (value, config)=>{
285
+ const maxAgents = config.orchestrator?.maxConcurrentAgents || 1;
286
+ if (value < maxAgents * 10) {
287
+ return 'taskQueueSize should be at least 10x maxConcurrentAgents';
288
+ }
289
+ return null;
290
+ }
291
+ });
292
+ // Terminal validation rules
293
+ this.validationRules.set('terminal.type', {
294
+ type: 'string',
295
+ required: true,
296
+ values: [
297
+ 'auto',
298
+ 'vscode',
299
+ 'native'
300
+ ]
301
+ });
302
+ this.validationRules.set('terminal.poolSize', {
303
+ type: 'number',
304
+ required: true,
305
+ min: 1,
306
+ max: 50
307
+ });
308
+ // Memory validation rules
309
+ this.validationRules.set('memory.backend', {
310
+ type: 'string',
311
+ required: true,
312
+ values: [
313
+ 'sqlite',
314
+ 'markdown',
315
+ 'hybrid'
316
+ ]
317
+ });
318
+ this.validationRules.set('memory.cacheSizeMB', {
319
+ type: 'number',
320
+ required: true,
321
+ min: 1,
322
+ max: 10000,
323
+ validator: (value)=>{
324
+ if (value > 1000) {
325
+ return 'Large cache sizes may impact system performance';
326
+ }
327
+ return null;
328
+ }
329
+ });
330
+ // Security validation rules
331
+ this.validationRules.set('security.encryptionEnabled', {
332
+ type: 'boolean',
333
+ required: true
334
+ });
335
+ // Credentials validation
336
+ this.validationRules.set('credentials.apiKey', {
337
+ type: 'string',
338
+ pattern: /^[a-zA-Z0-9_-]+$/,
339
+ validator: (value)=>{
340
+ if (value && value.length < 16) {
341
+ return 'API key should be at least 16 characters long';
342
+ }
343
+ return null;
344
+ }
345
+ });
346
+ }
347
+ /**
348
+ * Loads configuration from various sources
349
+ */ async load(configPath) {
350
+ if (configPath !== undefined) {
351
+ this.configPath = configPath;
352
+ }
353
+ // Start with defaults
354
+ let config = deepClone(DEFAULT_CONFIG);
355
+ // Load from file if specified
356
+ if (configPath) {
357
+ const fileConfig = await this.loadFromFile(configPath);
358
+ config = deepMergeConfig(config, fileConfig);
359
+ }
360
+ // Load from environment variables
361
+ const envConfig = this.loadFromEnv();
362
+ config = deepMergeConfig(config, envConfig);
363
+ // Validate the final configuration
364
+ this.validate(config);
365
+ this.config = config;
366
+ return config;
367
+ }
368
+ /**
369
+ * Gets the current configuration with optional security masking
370
+ */ get(maskSensitive = false) {
371
+ const config = deepClone(this.config);
372
+ if (maskSensitive && this.config.security?.maskSensitiveValues) {
373
+ return this.maskSensitiveValues(config);
374
+ }
375
+ return config;
376
+ }
377
+ /**
378
+ * Gets configuration with security masking applied
379
+ */ getSecure() {
380
+ return this.get(true);
381
+ }
382
+ /**
383
+ * Gets all configuration values (alias for get method for backward compatibility)
384
+ */ async getAll() {
385
+ return this.get();
386
+ }
387
+ /**
388
+ * Updates configuration values with change tracking
389
+ */ update(updates, options = {}) {
390
+ const oldConfig = deepClone(this.config);
391
+ // Track changes before applying
392
+ this.trackChanges(oldConfig, updates, options);
393
+ // Apply updates
394
+ this.config = deepMergeConfig(this.config, updates);
395
+ // Validate the updated configuration
396
+ this.validateWithDependencies(this.config);
397
+ return this.get();
398
+ }
399
+ /**
400
+ * Loads default configuration
401
+ */ loadDefault() {
402
+ this.config = deepClone(DEFAULT_CONFIG);
403
+ }
404
+ /**
405
+ * Saves configuration to file with format support
406
+ */ async save(path, format) {
407
+ const savePath = path || this.configPath;
408
+ if (!savePath) {
409
+ throw new ConfigError('No configuration file path specified');
410
+ }
411
+ const detectedFormat = format || this.detectFormat(savePath);
412
+ const parser = this.formatParsers[detectedFormat];
413
+ if (!parser) {
414
+ throw new ConfigError(`Unsupported format for saving: ${detectedFormat}`);
415
+ }
416
+ // Get configuration without sensitive values for saving
417
+ const configToSave = this.getConfigForSaving();
418
+ const content = parser.stringify(configToSave);
419
+ await fs.writeFile(savePath, content, 'utf8');
420
+ // Record the save operation
421
+ this.recordChange({
422
+ timestamp: new Date().toISOString(),
423
+ path: 'CONFIG_SAVED',
424
+ oldValue: null,
425
+ newValue: savePath,
426
+ source: 'file'
427
+ });
428
+ }
429
+ /**
430
+ * Gets configuration suitable for saving (excludes runtime-only values)
431
+ */ getConfigForSaving() {
432
+ const config = deepClone(this.config);
433
+ // Remove encrypted credentials from the saved config
434
+ // They should be stored separately in a secure location
435
+ if (config.credentials) {
436
+ delete config.credentials;
437
+ }
438
+ return config;
439
+ }
440
+ /**
441
+ * Gets user configuration directory
442
+ */ getUserConfigDir() {
443
+ const home = homedir();
444
+ return join(home, '.claude-flow');
445
+ }
446
+ /**
447
+ * Creates user config directory if it doesn't exist
448
+ */ async ensureUserConfigDir() {
449
+ try {
450
+ await fs.mkdir(this.userConfigDir, {
451
+ recursive: true
452
+ });
453
+ } catch (error) {
454
+ if (error.code !== 'EEXIST') {
455
+ throw new ConfigError(`Failed to create config directory: ${error.message}`);
456
+ }
457
+ }
458
+ }
459
+ /**
460
+ * Loads all profiles from the profiles directory
461
+ */ async loadProfiles() {
462
+ const profilesDir = join(this.userConfigDir, 'profiles');
463
+ try {
464
+ const entries = await fs.readdir(profilesDir, {
465
+ withFileTypes: true
466
+ });
467
+ for (const entry of entries){
468
+ if (entry.isFile() && entry.name.endsWith('.json')) {
469
+ const profileName = entry.name.replace('.json', '');
470
+ const profilePath = join(profilesDir, entry.name);
471
+ try {
472
+ const content = await fs.readFile(profilePath, 'utf8');
473
+ const profileConfig = safeParseJSON(content);
474
+ if (profileConfig) {
475
+ this.profiles.set(profileName, profileConfig);
476
+ }
477
+ } catch (error) {
478
+ console.warn(`Failed to load profile ${profileName}: ${error.message}`);
479
+ }
480
+ }
481
+ }
482
+ } catch (error) {
483
+ // Profiles directory doesn't exist - this is okay
484
+ }
485
+ }
486
+ /**
487
+ * Applies a named profile
488
+ */ async applyProfile(profileName) {
489
+ await this.loadProfiles();
490
+ const profile = this.profiles.get(profileName);
491
+ if (!profile) {
492
+ throw new ConfigError(`Profile '${profileName}' not found`);
493
+ }
494
+ this.config = deepMergeConfig(this.config, profile);
495
+ this.currentProfile = profileName;
496
+ this.validate(this.config);
497
+ }
498
+ /**
499
+ * Saves current configuration as a profile
500
+ */ async saveProfile(profileName, config) {
501
+ await this.ensureUserConfigDir();
502
+ const profilesDir = join(this.userConfigDir, 'profiles');
503
+ await fs.mkdir(profilesDir, {
504
+ recursive: true
505
+ });
506
+ const profileConfig = config || this.config;
507
+ const profilePath = join(profilesDir, `${profileName}.json`);
508
+ const content = JSON.stringify(profileConfig, null, 2);
509
+ await fs.writeFile(profilePath, content, 'utf8');
510
+ this.profiles.set(profileName, profileConfig);
511
+ }
512
+ /**
513
+ * Deletes a profile
514
+ */ async deleteProfile(profileName) {
515
+ const profilePath = join(this.userConfigDir, 'profiles', `${profileName}.json`);
516
+ try {
517
+ await fs.unlink(profilePath);
518
+ this.profiles.delete(profileName);
519
+ } catch (error) {
520
+ if (error.code === 'ENOENT') {
521
+ throw new ConfigError(`Profile '${profileName}' not found`);
522
+ }
523
+ throw new ConfigError(`Failed to delete profile: ${error.message}`);
524
+ }
525
+ }
526
+ /**
527
+ * Lists all available profiles
528
+ */ async listProfiles() {
529
+ await this.loadProfiles();
530
+ return Array.from(this.profiles.keys());
531
+ }
532
+ /**
533
+ * Gets a specific profile configuration
534
+ */ async getProfile(profileName) {
535
+ await this.loadProfiles();
536
+ return this.profiles.get(profileName);
537
+ }
538
+ /**
539
+ * Gets the current active profile name
540
+ */ getCurrentProfile() {
541
+ return this.currentProfile;
542
+ }
543
+ /**
544
+ * Sets a configuration value by path with change tracking and validation
545
+ */ set(path, value, options = {}) {
546
+ const oldValue = this.getValue(path);
547
+ // Record the change
548
+ this.recordChange({
549
+ timestamp: new Date().toISOString(),
550
+ path,
551
+ oldValue,
552
+ newValue: value,
553
+ user: options.user,
554
+ reason: options.reason,
555
+ source: options.source || 'cli'
556
+ });
557
+ // Encrypt sensitive values
558
+ if (this.isSensitivePath(path) && this.config.security?.encryptionEnabled) {
559
+ value = this.encryptValue(value);
560
+ }
561
+ const keys = path.split('.');
562
+ let current = this.config;
563
+ for(let i = 0; i < keys.length - 1; i++){
564
+ const key = keys[i];
565
+ if (!(key in current)) {
566
+ current[key] = {};
567
+ }
568
+ current = current[key];
569
+ }
570
+ current[keys[keys.length - 1]] = value;
571
+ // Validate the path-specific rule and dependencies
572
+ this.validatePath(path, value);
573
+ this.validateWithDependencies(this.config);
574
+ }
575
+ /**
576
+ * Gets a configuration value by path with decryption for sensitive values
577
+ */ getValue(path, decrypt = true) {
578
+ const keys = path.split('.');
579
+ let current = this.config;
580
+ for (const key of keys){
581
+ if (current && typeof current === 'object' && key in current) {
582
+ current = current[key];
583
+ } else {
584
+ return undefined;
585
+ }
586
+ }
587
+ // Decrypt sensitive values if requested
588
+ if (decrypt && this.isSensitivePath(path) && this.isEncryptedValue(current)) {
589
+ try {
590
+ return this.decryptValue(current);
591
+ } catch (error) {
592
+ console.warn(`Failed to decrypt value at path ${path}:`, error.message);
593
+ return current;
594
+ }
595
+ }
596
+ return current;
597
+ }
598
+ /**
599
+ * Resets configuration to defaults
600
+ */ reset() {
601
+ this.config = deepClone(DEFAULT_CONFIG);
602
+ delete this.currentProfile;
603
+ }
604
+ /**
605
+ * Gets configuration schema for validation
606
+ */ getSchema() {
607
+ return {
608
+ orchestrator: {
609
+ maxConcurrentAgents: {
610
+ type: 'number',
611
+ min: 1,
612
+ max: 100
613
+ },
614
+ taskQueueSize: {
615
+ type: 'number',
616
+ min: 1,
617
+ max: 10000
618
+ },
619
+ healthCheckInterval: {
620
+ type: 'number',
621
+ min: 1000,
622
+ max: 300000
623
+ },
624
+ shutdownTimeout: {
625
+ type: 'number',
626
+ min: 1000,
627
+ max: 300000
628
+ }
629
+ },
630
+ terminal: {
631
+ type: {
632
+ type: 'string',
633
+ values: [
634
+ 'auto',
635
+ 'vscode',
636
+ 'native'
637
+ ]
638
+ },
639
+ poolSize: {
640
+ type: 'number',
641
+ min: 1,
642
+ max: 50
643
+ },
644
+ recycleAfter: {
645
+ type: 'number',
646
+ min: 1,
647
+ max: 1000
648
+ },
649
+ healthCheckInterval: {
650
+ type: 'number',
651
+ min: 1000,
652
+ max: 3600000
653
+ },
654
+ commandTimeout: {
655
+ type: 'number',
656
+ min: 1000,
657
+ max: 3600000
658
+ }
659
+ },
660
+ memory: {
661
+ backend: {
662
+ type: 'string',
663
+ values: [
664
+ 'sqlite',
665
+ 'markdown',
666
+ 'hybrid'
667
+ ]
668
+ },
669
+ cacheSizeMB: {
670
+ type: 'number',
671
+ min: 1,
672
+ max: 10000
673
+ },
674
+ syncInterval: {
675
+ type: 'number',
676
+ min: 1000,
677
+ max: 300000
678
+ },
679
+ conflictResolution: {
680
+ type: 'string',
681
+ values: [
682
+ 'crdt',
683
+ 'timestamp',
684
+ 'manual'
685
+ ]
686
+ },
687
+ retentionDays: {
688
+ type: 'number',
689
+ min: 1,
690
+ max: 3650
691
+ }
692
+ },
693
+ coordination: {
694
+ maxRetries: {
695
+ type: 'number',
696
+ min: 0,
697
+ max: 100
698
+ },
699
+ retryDelay: {
700
+ type: 'number',
701
+ min: 100,
702
+ max: 60000
703
+ },
704
+ deadlockDetection: {
705
+ type: 'boolean'
706
+ },
707
+ resourceTimeout: {
708
+ type: 'number',
709
+ min: 1000,
710
+ max: 3600000
711
+ },
712
+ messageTimeout: {
713
+ type: 'number',
714
+ min: 1000,
715
+ max: 300000
716
+ }
717
+ },
718
+ mcp: {
719
+ transport: {
720
+ type: 'string',
721
+ values: [
722
+ 'stdio',
723
+ 'http',
724
+ 'websocket'
725
+ ]
726
+ },
727
+ port: {
728
+ type: 'number',
729
+ min: 1,
730
+ max: 65535
731
+ },
732
+ tlsEnabled: {
733
+ type: 'boolean'
734
+ }
735
+ },
736
+ logging: {
737
+ level: {
738
+ type: 'string',
739
+ values: [
740
+ 'debug',
741
+ 'info',
742
+ 'warn',
743
+ 'error'
744
+ ]
745
+ },
746
+ format: {
747
+ type: 'string',
748
+ values: [
749
+ 'json',
750
+ 'text'
751
+ ]
752
+ },
753
+ destination: {
754
+ type: 'string',
755
+ values: [
756
+ 'console',
757
+ 'file'
758
+ ]
759
+ }
760
+ }
761
+ };
762
+ }
763
+ /**
764
+ * Validates a value against schema
765
+ */ validateValue(value, schema, path) {
766
+ if (schema.type === 'number') {
767
+ if (typeof value !== 'number' || isNaN(value)) {
768
+ throw new ValidationError(`${path}: must be a number`);
769
+ }
770
+ if (schema.min !== undefined && value < schema.min) {
771
+ throw new ValidationError(`${path}: must be at least ${schema.min}`);
772
+ }
773
+ if (schema.max !== undefined && value > schema.max) {
774
+ throw new ValidationError(`${path}: must be at most ${schema.max}`);
775
+ }
776
+ } else if (schema.type === 'string') {
777
+ if (typeof value !== 'string') {
778
+ throw new ValidationError(`${path}: must be a string`);
779
+ }
780
+ if (schema.values && !schema.values.includes(value)) {
781
+ throw new ValidationError(`${path}: must be one of [${schema.values.join(', ')}]`);
782
+ }
783
+ } else if (schema.type === 'boolean') {
784
+ if (typeof value !== 'boolean') {
785
+ throw new ValidationError(`${path}: must be a boolean`);
786
+ }
787
+ }
788
+ }
789
+ /**
790
+ * Gets configuration diff between current and default
791
+ */ getDiff() {
792
+ const defaultConfig = DEFAULT_CONFIG;
793
+ const diff = {};
794
+ const findDifferences = (current, defaults, path = '')=>{
795
+ for(const key in current){
796
+ const currentValue = current[key];
797
+ const defaultValue = defaults[key];
798
+ const fullPath = path ? `${path}.${key}` : key;
799
+ if (typeof currentValue === 'object' && currentValue !== null && !Array.isArray(currentValue)) {
800
+ if (typeof defaultValue === 'object' && defaultValue !== null) {
801
+ const nestedDiff = {};
802
+ findDifferences(currentValue, defaultValue, fullPath);
803
+ if (Object.keys(nestedDiff).length > 0) {
804
+ if (!path) {
805
+ diff[key] = nestedDiff;
806
+ }
807
+ }
808
+ }
809
+ } else if (currentValue !== defaultValue) {
810
+ const pathParts = fullPath.split('.');
811
+ let target = diff;
812
+ for(let i = 0; i < pathParts.length - 1; i++){
813
+ if (!target[pathParts[i]]) {
814
+ target[pathParts[i]] = {};
815
+ }
816
+ target = target[pathParts[i]];
817
+ }
818
+ target[pathParts[pathParts.length - 1]] = currentValue;
819
+ }
820
+ }
821
+ };
822
+ findDifferences(this.config, defaultConfig);
823
+ return diff;
824
+ }
825
+ /**
826
+ * Exports configuration with metadata
827
+ */ export() {
828
+ return {
829
+ version: '1.0.0',
830
+ exported: new Date().toISOString(),
831
+ profile: this.currentProfile,
832
+ config: this.config,
833
+ diff: this.getDiff()
834
+ };
835
+ }
836
+ /**
837
+ * Imports configuration from export
838
+ */ import(data) {
839
+ if (!data.config) {
840
+ throw new ConfigError('Invalid configuration export format');
841
+ }
842
+ this.validateWithDependencies(data.config);
843
+ this.config = data.config;
844
+ this.currentProfile = data.profile;
845
+ // Record the import operation
846
+ this.recordChange({
847
+ timestamp: new Date().toISOString(),
848
+ path: 'CONFIG_IMPORTED',
849
+ oldValue: null,
850
+ newValue: data.version || 'unknown',
851
+ source: 'file'
852
+ });
853
+ }
854
+ /**
855
+ * Loads configuration from file with format detection
856
+ */ async loadFromFile(path) {
857
+ try {
858
+ const content = await fs.readFile(path, 'utf8');
859
+ const format = this.detectFormat(path, content);
860
+ const parser = this.formatParsers[format];
861
+ if (!parser) {
862
+ throw new ConfigError(`Unsupported configuration format: ${format}`);
863
+ }
864
+ const config = parser.parse(content);
865
+ if (!config) {
866
+ throw new ConfigError(`Invalid ${format.toUpperCase()} in configuration file: ${path}`);
867
+ }
868
+ return config;
869
+ } catch (error) {
870
+ if (error.code === 'ENOENT') {
871
+ // File doesn't exist, use defaults
872
+ return {};
873
+ }
874
+ throw new ConfigError(`Failed to load configuration from ${path}: ${error.message}`);
875
+ }
876
+ }
877
+ /**
878
+ * Detects configuration file format
879
+ */ detectFormat(path, content) {
880
+ const ext = path.split('.').pop()?.toLowerCase();
881
+ if (ext === 'yaml' || ext === 'yml') return 'yaml';
882
+ if (ext === 'toml') return 'toml';
883
+ if (ext === 'json') return 'json';
884
+ // Try to detect from content
885
+ if (content) {
886
+ const trimmed = content.trim();
887
+ if (trimmed.startsWith('{') || trimmed.startsWith('[')) return 'json';
888
+ if (trimmed.includes('=') && trimmed.includes('[')) return 'toml';
889
+ if (trimmed.includes(':') && !trimmed.includes('=')) return 'yaml';
890
+ }
891
+ // Default to JSON
892
+ return 'json';
893
+ }
894
+ /**
895
+ * Loads configuration from environment variables
896
+ */ loadFromEnv() {
897
+ const config = {};
898
+ // Orchestrator settings
899
+ const maxAgents = process.env.CLAUDE_FLOW_MAX_AGENTS;
900
+ if (maxAgents) {
901
+ if (!config.orchestrator) {
902
+ config.orchestrator = {};
903
+ }
904
+ config.orchestrator = {
905
+ ...DEFAULT_CONFIG.orchestrator,
906
+ ...config.orchestrator,
907
+ maxConcurrentAgents: parseInt(maxAgents, 10)
908
+ };
909
+ }
910
+ // Terminal settings
911
+ const terminalType = process.env.CLAUDE_FLOW_TERMINAL_TYPE;
912
+ if (terminalType === 'vscode' || terminalType === 'native' || terminalType === 'auto') {
913
+ config.terminal = {
914
+ ...DEFAULT_CONFIG.terminal,
915
+ ...config.terminal,
916
+ type: terminalType
917
+ };
918
+ }
919
+ // Memory settings
920
+ const memoryBackend = process.env.CLAUDE_FLOW_MEMORY_BACKEND;
921
+ if (memoryBackend === 'sqlite' || memoryBackend === 'markdown' || memoryBackend === 'hybrid') {
922
+ config.memory = {
923
+ ...DEFAULT_CONFIG.memory,
924
+ ...config.memory,
925
+ backend: memoryBackend
926
+ };
927
+ }
928
+ // MCP settings
929
+ const mcpTransport = process.env.CLAUDE_FLOW_MCP_TRANSPORT;
930
+ if (mcpTransport === 'stdio' || mcpTransport === 'http' || mcpTransport === 'websocket') {
931
+ config.mcp = {
932
+ ...DEFAULT_CONFIG.mcp,
933
+ ...config.mcp,
934
+ transport: mcpTransport
935
+ };
936
+ }
937
+ const mcpPort = process.env.CLAUDE_FLOW_MCP_PORT;
938
+ if (mcpPort) {
939
+ config.mcp = {
940
+ ...DEFAULT_CONFIG.mcp,
941
+ ...config.mcp,
942
+ port: parseInt(mcpPort, 10)
943
+ };
944
+ }
945
+ // Logging settings
946
+ const logLevel = process.env.CLAUDE_FLOW_LOG_LEVEL;
947
+ if (logLevel === 'debug' || logLevel === 'info' || logLevel === 'warn' || logLevel === 'error') {
948
+ config.logging = {
949
+ ...DEFAULT_CONFIG.logging,
950
+ ...config.logging,
951
+ level: logLevel
952
+ };
953
+ }
954
+ return config;
955
+ }
956
+ /**
957
+ * Validates configuration with dependency checking
958
+ */ validateWithDependencies(config) {
959
+ const errors = [];
960
+ const warnings = [];
961
+ // Validate all paths with rules
962
+ for (const [path, rule] of this.validationRules.entries()){
963
+ const value = this.getValueByPath(config, path);
964
+ try {
965
+ this.validatePath(path, value, config);
966
+ } catch (error) {
967
+ errors.push(error.message);
968
+ }
969
+ }
970
+ // Additional cross-field validations
971
+ if (config.orchestrator.maxConcurrentAgents > config.terminal.poolSize * 3) {
972
+ warnings.push('High agent-to-terminal ratio may cause resource contention');
973
+ }
974
+ if (config.memory.cacheSizeMB > 1000 && config.memory.backend === 'sqlite') {
975
+ warnings.push('Large cache size with SQLite backend may impact performance');
976
+ }
977
+ if (config.mcp.transport === 'http' && !config.mcp.tlsEnabled) {
978
+ warnings.push('HTTP transport without TLS is not recommended for production');
979
+ }
980
+ // Log warnings
981
+ if (warnings.length > 0 && config.logging?.level === 'debug') {
982
+ console.warn('Configuration warnings:', warnings);
983
+ }
984
+ // Throw errors
985
+ if (errors.length > 0) {
986
+ throw new ValidationError(`Configuration validation failed:\n${errors.join('\n')}`);
987
+ }
988
+ }
989
+ /**
990
+ * Validates a specific configuration path
991
+ */ validatePath(path, value, config) {
992
+ const rule = this.validationRules.get(path);
993
+ if (!rule) return;
994
+ const currentConfig = config || this.config;
995
+ // Required validation
996
+ if (rule.required && (value === undefined || value === null)) {
997
+ throw new ValidationError(`${path} is required`);
998
+ }
999
+ if (value === undefined || value === null) return;
1000
+ // Type validation
1001
+ if (rule.type === 'number' && (typeof value !== 'number' || isNaN(value))) {
1002
+ throw new ValidationError(`${path} must be a number`);
1003
+ }
1004
+ if (rule.type === 'string' && typeof value !== 'string') {
1005
+ throw new ValidationError(`${path} must be a string`);
1006
+ }
1007
+ if (rule.type === 'boolean' && typeof value !== 'boolean') {
1008
+ throw new ValidationError(`${path} must be a boolean`);
1009
+ }
1010
+ // Range validation
1011
+ if (typeof value === 'number') {
1012
+ if (rule.min !== undefined && value < rule.min) {
1013
+ throw new ValidationError(`${path} must be at least ${rule.min}`);
1014
+ }
1015
+ if (rule.max !== undefined && value > rule.max) {
1016
+ throw new ValidationError(`${path} must be at most ${rule.max}`);
1017
+ }
1018
+ }
1019
+ // Values validation
1020
+ if (rule.values && !rule.values.includes(value)) {
1021
+ throw new ValidationError(`${path} must be one of: ${rule.values.join(', ')}`);
1022
+ }
1023
+ // Pattern validation
1024
+ if (rule.pattern && typeof value === 'string' && !rule.pattern.test(value)) {
1025
+ throw new ValidationError(`${path} does not match required pattern`);
1026
+ }
1027
+ // Custom validator
1028
+ if (rule.validator) {
1029
+ const result = rule.validator(value, currentConfig);
1030
+ if (result) {
1031
+ throw new ValidationError(`${path}: ${result}`);
1032
+ }
1033
+ }
1034
+ }
1035
+ /**
1036
+ * Gets a value from a configuration object by path
1037
+ */ getValueByPath(obj, path) {
1038
+ const keys = path.split('.');
1039
+ let current = obj;
1040
+ for (const key of keys){
1041
+ if (current && typeof current === 'object' && key in current) {
1042
+ current = current[key];
1043
+ } else {
1044
+ return undefined;
1045
+ }
1046
+ }
1047
+ return current;
1048
+ }
1049
+ /**
1050
+ * Legacy validate method for backward compatibility
1051
+ */ validate(config) {
1052
+ this.validateWithDependencies(config);
1053
+ }
1054
+ /**
1055
+ * Masks sensitive values in configuration
1056
+ */ maskSensitiveValues(config) {
1057
+ const maskedConfig = deepClone(config);
1058
+ // Recursively mask sensitive paths
1059
+ const maskObject = (obj, path = '')=>{
1060
+ if (!obj || typeof obj !== 'object') return obj;
1061
+ const masked = {};
1062
+ for (const [key, value] of Object.entries(obj)){
1063
+ const currentPath = path ? `${path}.${key}` : key;
1064
+ if (this.isSensitivePath(currentPath)) {
1065
+ const classification = SECURITY_CLASSIFICATIONS[currentPath];
1066
+ masked[key] = classification?.maskPattern || '****';
1067
+ } else if (typeof value === 'object' && value !== null) {
1068
+ masked[key] = maskObject(value, currentPath);
1069
+ } else {
1070
+ masked[key] = value;
1071
+ }
1072
+ }
1073
+ return masked;
1074
+ };
1075
+ return maskObject(maskedConfig);
1076
+ }
1077
+ /**
1078
+ * Tracks changes to configuration
1079
+ */ trackChanges(oldConfig, updates, options) {
1080
+ // Simple implementation for tracking changes
1081
+ for (const [key, value] of Object.entries(updates)){
1082
+ this.recordChange({
1083
+ timestamp: new Date().toISOString(),
1084
+ path: key,
1085
+ oldValue: oldConfig[key],
1086
+ newValue: value,
1087
+ user: options.user,
1088
+ reason: options.reason,
1089
+ source: options.source || 'cli'
1090
+ });
1091
+ }
1092
+ }
1093
+ /**
1094
+ * Records a configuration change
1095
+ */ recordChange(change) {
1096
+ this.changeHistory.push(change);
1097
+ // Keep only last 1000 changes
1098
+ if (this.changeHistory.length > 1000) {
1099
+ this.changeHistory.shift();
1100
+ }
1101
+ }
1102
+ /**
1103
+ * Checks if a path contains sensitive information
1104
+ */ isSensitivePath(path) {
1105
+ return SENSITIVE_PATHS.some((sensitive)=>path.toLowerCase().includes(sensitive.toLowerCase()));
1106
+ }
1107
+ /**
1108
+ * Encrypts a sensitive value
1109
+ */ encryptValue(value) {
1110
+ if (!this.encryptionKey) {
1111
+ return value; // Return original if encryption not available
1112
+ }
1113
+ try {
1114
+ // Simplified encryption - in production use proper encryption
1115
+ const iv = randomBytes(16);
1116
+ const key = createHash('sha256').update(this.encryptionKey).digest();
1117
+ const cipher = createCipheriv('aes-256-cbc', key, iv);
1118
+ let encrypted = cipher.update(JSON.stringify(value), 'utf8', 'hex');
1119
+ encrypted += cipher.final('hex');
1120
+ return `encrypted:${iv.toString('hex')}:${encrypted}`;
1121
+ } catch (error) {
1122
+ console.warn('Failed to encrypt value:', error.message);
1123
+ return value;
1124
+ }
1125
+ }
1126
+ /**
1127
+ * Decrypts a sensitive value
1128
+ */ decryptValue(encryptedValue) {
1129
+ if (!this.encryptionKey || !this.isEncryptedValue(encryptedValue)) {
1130
+ return encryptedValue;
1131
+ }
1132
+ try {
1133
+ const parts = encryptedValue.replace('encrypted:', '').split(':');
1134
+ if (parts.length !== 2) return encryptedValue; // Handle old format
1135
+ const iv = Buffer.from(parts[0], 'hex');
1136
+ const encrypted = parts[1];
1137
+ const key = createHash('sha256').update(this.encryptionKey).digest();
1138
+ const decipher = createDecipheriv('aes-256-cbc', key, iv);
1139
+ let decrypted = decipher.update(encrypted, 'hex', 'utf8');
1140
+ decrypted += decipher.final('utf8');
1141
+ return JSON.parse(decrypted);
1142
+ } catch (error) {
1143
+ console.warn('Failed to decrypt value:', error.message);
1144
+ return encryptedValue;
1145
+ }
1146
+ }
1147
+ /**
1148
+ * Checks if a value is encrypted
1149
+ */ isEncryptedValue(value) {
1150
+ return typeof value === 'string' && value.startsWith('encrypted:');
1151
+ }
1152
+ }
1153
+ // Export singleton instance
1154
+ export const configManager = ConfigManager.getInstance();
1155
+ // Helper function to load configuration
1156
+ export async function loadConfig(path) {
1157
+ return await configManager.load(path);
1158
+ }
1159
+ function deepClone(obj) {
1160
+ return JSON.parse(JSON.stringify(obj));
1161
+ }
1162
+ export { SENSITIVE_PATHS, SECURITY_CLASSIFICATIONS };
1163
+ // Custom deepMerge for Config type
1164
+ function deepMergeConfig(target, ...sources) {
1165
+ const result = deepClone(target);
1166
+ for (const source of sources){
1167
+ if (!source) continue;
1168
+ // Merge each section
1169
+ if (source.orchestrator) {
1170
+ result.orchestrator = {
1171
+ ...result.orchestrator,
1172
+ ...source.orchestrator
1173
+ };
1174
+ }
1175
+ if (source.terminal) {
1176
+ result.terminal = {
1177
+ ...result.terminal,
1178
+ ...source.terminal
1179
+ };
1180
+ }
1181
+ if (source.memory) {
1182
+ result.memory = {
1183
+ ...result.memory,
1184
+ ...source.memory
1185
+ };
1186
+ }
1187
+ if (source.coordination) {
1188
+ result.coordination = {
1189
+ ...result.coordination,
1190
+ ...source.coordination
1191
+ };
1192
+ }
1193
+ if (source.mcp) {
1194
+ result.mcp = {
1195
+ ...result.mcp,
1196
+ ...source.mcp
1197
+ };
1198
+ }
1199
+ if (source.logging) {
1200
+ result.logging = {
1201
+ ...result.logging,
1202
+ ...source.logging
1203
+ };
1204
+ }
1205
+ if (source.credentials) {
1206
+ result.credentials = {
1207
+ ...result.credentials,
1208
+ ...source.credentials
1209
+ };
1210
+ }
1211
+ if (source.security) {
1212
+ result.security = {
1213
+ ...result.security,
1214
+ ...source.security
1215
+ };
1216
+ }
1217
+ }
1218
+ return result;
1219
+ }
1220
+
1221
+ //# sourceMappingURL=config.js.map