fraim-framework 2.0.146 → 2.0.148

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.
@@ -426,6 +426,10 @@ body {
426
426
  gap: 12px;
427
427
  }
428
428
 
429
+ .start-container {
430
+ gap: 14px;
431
+ }
432
+
429
433
  .user-type-card {
430
434
  background: var(--surface);
431
435
  border: 1.5px solid var(--line);
@@ -448,6 +452,20 @@ body {
448
452
  box-shadow: 0 4px 20px rgba(61, 138, 110, 0.18);
449
453
  }
450
454
 
455
+ .user-type-card--ide-default {
456
+ padding: 24px;
457
+ gap: 18px;
458
+ }
459
+
460
+ .user-type-card--compact {
461
+ padding: 16px 18px;
462
+ gap: 12px;
463
+ }
464
+
465
+ .user-type-card--alpha {
466
+ background: linear-gradient(180deg, #fbfcfa 0%, #f4f7f2 100%);
467
+ }
468
+
451
469
  .card-header {
452
470
  display: flex;
453
471
  gap: 14px;
@@ -483,6 +501,30 @@ body {
483
501
  }
484
502
  .card-desc { margin: 0; font-size: 13px; color: var(--muted); line-height: 1.45; }
485
503
 
504
+ .card-eyebrow {
505
+ display: inline-flex;
506
+ align-self: flex-start;
507
+ padding: 4px 10px;
508
+ border-radius: 999px;
509
+ background: var(--accent-soft);
510
+ color: var(--accent-strong);
511
+ font-size: 11px;
512
+ font-weight: 700;
513
+ letter-spacing: 0.06em;
514
+ text-transform: uppercase;
515
+ }
516
+
517
+ .route-note {
518
+ margin: 0;
519
+ color: var(--muted);
520
+ font-size: 13px;
521
+ line-height: 1.5;
522
+ }
523
+
524
+ .route-note--compact {
525
+ font-size: 12px;
526
+ }
527
+
486
528
  .btn {
487
529
  display: inline-flex;
488
530
  align-items: center;
@@ -530,28 +572,6 @@ body {
530
572
  border-color: var(--line);
531
573
  }
532
574
 
533
- /* "or" divider between route cards */
534
- .route-or {
535
- text-align: center;
536
- color: var(--muted);
537
- font-size: 12px;
538
- letter-spacing: 0.05em;
539
- text-transform: uppercase;
540
- padding: 8px 0;
541
- position: relative;
542
- }
543
- .route-or::before,
544
- .route-or::after {
545
- content: '';
546
- position: absolute;
547
- top: 50%;
548
- width: calc(50% - 22px);
549
- height: 1px;
550
- background: var(--line);
551
- }
552
- .route-or::before { left: 0; }
553
- .route-or::after { right: 0; }
554
-
555
575
  /* IDE command display — shown after choosing "In my IDE". */
556
576
  .cmd-block {
557
577
  background: #0d1410;
@@ -600,4 +620,8 @@ body {
600
620
  .row .change-link { margin-left: auto; }
601
621
  .row .project-picker { gap: 8px; }
602
622
  .row .project-picker input { width: 100%; min-width: 0; }
623
+ .user-type-card--ide-default,
624
+ .user-type-card--compact {
625
+ padding: 18px 16px;
626
+ }
603
627
  }
@@ -1,171 +0,0 @@
1
- "use strict";
2
- var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
- if (k2 === undefined) k2 = k;
4
- var desc = Object.getOwnPropertyDescriptor(m, k);
5
- if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
- desc = { enumerable: true, get: function() { return m[k]; } };
7
- }
8
- Object.defineProperty(o, k2, desc);
9
- }) : (function(o, m, k, k2) {
10
- if (k2 === undefined) k2 = k;
11
- o[k2] = m[k];
12
- }));
13
- var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
- Object.defineProperty(o, "default", { enumerable: true, value: v });
15
- }) : function(o, v) {
16
- o["default"] = v;
17
- });
18
- var __importStar = (this && this.__importStar) || (function () {
19
- var ownKeys = function(o) {
20
- ownKeys = Object.getOwnPropertyNames || function (o) {
21
- var ar = [];
22
- for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
- return ar;
24
- };
25
- return ownKeys(o);
26
- };
27
- return function (mod) {
28
- if (mod && mod.__esModule) return mod;
29
- var result = {};
30
- if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
- __setModuleDefault(result, mod);
32
- return result;
33
- };
34
- })();
35
- var __importDefault = (this && this.__importDefault) || function (mod) {
36
- return (mod && mod.__esModule) ? mod : { "default": mod };
37
- };
38
- Object.defineProperty(exports, "__esModule", { value: true });
39
- exports.testMCPCommand = exports.runTestMCP = void 0;
40
- const commander_1 = require("commander");
41
- const chalk_1 = __importDefault(require("chalk"));
42
- const fs_1 = __importDefault(require("fs"));
43
- const path_1 = __importDefault(require("path"));
44
- const ide_detector_1 = require("../setup/ide-detector");
45
- const script_sync_utils_1 = require("../utils/script-sync-utils");
46
- const testIDEConfig = async (ide) => {
47
- const result = {
48
- ide: ide.name,
49
- configExists: false,
50
- configValid: false,
51
- mcpServers: [],
52
- errors: []
53
- };
54
- const configPath = (0, ide_detector_1.expandPath)(ide.configPath);
55
- if (!fs_1.default.existsSync(configPath)) {
56
- result.errors.push('Config file does not exist');
57
- return result;
58
- }
59
- result.configExists = true;
60
- try {
61
- if (ide.configFormat === 'json') {
62
- const configContent = fs_1.default.readFileSync(configPath, 'utf8');
63
- const config = JSON.parse(configContent);
64
- const servers = ide.configType === 'vscode' ? config.servers : config.mcpServers;
65
- if (servers) {
66
- result.configValid = true;
67
- result.mcpServers = Object.keys(servers);
68
- }
69
- else {
70
- const expectedKey = ide.configType === 'vscode' ? 'servers' : 'mcpServers';
71
- result.errors.push(`No ${expectedKey} section found`);
72
- }
73
- }
74
- else if (ide.configFormat === 'toml') {
75
- const configContent = fs_1.default.readFileSync(configPath, 'utf8');
76
- // Simple TOML parsing for MCP servers
77
- const serverMatches = configContent.match(/\[mcp_servers\.(\w+)\]/g);
78
- if (serverMatches) {
79
- result.configValid = true;
80
- result.mcpServers = serverMatches.map(match => match.replace(/\[mcp_servers\.(\w+)\]/, '$1'));
81
- }
82
- else {
83
- result.errors.push('No mcp_servers sections found');
84
- }
85
- }
86
- }
87
- catch (error) {
88
- result.errors.push(`Failed to parse config: ${error instanceof Error ? error.message : 'Unknown error'}`);
89
- }
90
- return result;
91
- };
92
- const checkGlobalSetup = () => {
93
- const globalConfigPath = path_1.default.join((0, script_sync_utils_1.getUserFraimDir)(), 'config.json');
94
- return fs_1.default.existsSync(globalConfigPath);
95
- };
96
- const runTestMCP = async () => {
97
- console.log(chalk_1.default.blue('🔍 Testing MCP configuration...\n'));
98
- // Check global setup
99
- if (!checkGlobalSetup()) {
100
- console.log(chalk_1.default.red('❌ Global FRAIM setup not found.'));
101
- console.log(chalk_1.default.yellow('Please run: fraim setup --key=<your-fraim-key>'));
102
- return;
103
- }
104
- console.log(chalk_1.default.green('✅ Global FRAIM setup found'));
105
- // Detect IDEs
106
- const detectedIDEs = (0, ide_detector_1.detectInstalledIDEs)();
107
- if (detectedIDEs.length === 0) {
108
- console.log(chalk_1.default.yellow('⚠️ No supported IDEs detected.'));
109
- return;
110
- }
111
- console.log(chalk_1.default.blue(`\n🔍 Testing ${detectedIDEs.length} detected IDEs...\n`));
112
- const results = await Promise.all(detectedIDEs.map(ide => testIDEConfig(ide)));
113
- let totalConfigured = 0;
114
- let totalWithFRAIM = 0;
115
- for (const result of results) {
116
- console.log(chalk_1.default.white(`📱 ${result.ide}`));
117
- if (!result.configExists) {
118
- console.log(chalk_1.default.red(' ❌ No MCP config found'));
119
- console.log(chalk_1.default.gray(` 💡 Run: fraim setup --ide=${result.ide.toLowerCase()}`));
120
- }
121
- else if (!result.configValid) {
122
- console.log(chalk_1.default.yellow(' ⚠️ Config exists but invalid'));
123
- result.errors.forEach(error => {
124
- console.log(chalk_1.default.red(` ❌ ${error}`));
125
- });
126
- }
127
- else {
128
- totalConfigured++;
129
- console.log(chalk_1.default.green(` ✅ MCP config valid (${result.mcpServers.length} servers)`));
130
- // Check for essential servers
131
- const { BASE_MCP_SERVERS } = await Promise.resolve().then(() => __importStar(require('../mcp/mcp-server-registry')));
132
- const essentialServers = BASE_MCP_SERVERS.map(s => s.id); // fraim, git, playwright
133
- const hasEssential = essentialServers.filter(server => result.mcpServers.includes(server));
134
- if (hasEssential.includes('fraim')) {
135
- totalWithFRAIM++;
136
- console.log(chalk_1.default.green(' ✅ FRAIM server configured'));
137
- }
138
- else {
139
- console.log(chalk_1.default.yellow(' ⚠️ FRAIM server missing'));
140
- }
141
- if (hasEssential.length > 1) {
142
- console.log(chalk_1.default.green(` ✅ ${hasEssential.length - 1} additional servers: ${hasEssential.filter(s => s !== 'fraim').join(', ')}`));
143
- }
144
- const missingEssential = essentialServers.filter(server => !result.mcpServers.includes(server));
145
- if (missingEssential.length > 0) {
146
- console.log(chalk_1.default.yellow(` ⚠️ Missing servers: ${missingEssential.join(', ')}`));
147
- }
148
- }
149
- console.log(); // Empty line
150
- }
151
- // Summary
152
- console.log(chalk_1.default.blue('📊 Summary:'));
153
- console.log(chalk_1.default.green(` ✅ ${totalConfigured}/${detectedIDEs.length} IDEs have valid MCP configs`));
154
- console.log(chalk_1.default.green(` ✅ ${totalWithFRAIM}/${detectedIDEs.length} IDEs have FRAIM configured`));
155
- if (totalWithFRAIM === 0) {
156
- console.log(chalk_1.default.red('\n❌ No IDEs have FRAIM configured!'));
157
- console.log(chalk_1.default.yellow('💡 Run: fraim setup --key=<your-fraim-key>'));
158
- }
159
- else if (totalWithFRAIM < detectedIDEs.length) {
160
- console.log(chalk_1.default.yellow(`\n⚠️ ${detectedIDEs.length - totalWithFRAIM} IDEs missing FRAIM configuration`));
161
- console.log(chalk_1.default.yellow('💡 Run: fraim setup to configure remaining IDEs'));
162
- }
163
- else {
164
- console.log(chalk_1.default.green('\n🎉 All detected IDEs have FRAIM configured!'));
165
- console.log(chalk_1.default.blue('💡 Try running: fraim init-project in any project'));
166
- }
167
- };
168
- exports.runTestMCP = runTestMCP;
169
- exports.testMCPCommand = new commander_1.Command('test-mcp')
170
- .description('Test MCP server configurations for all detected IDEs')
171
- .action(exports.runTestMCP);
@@ -1,242 +0,0 @@
1
- "use strict";
2
- var __createBinding = (this && this.__createBinding) || (Object.create ? (function(o, m, k, k2) {
3
- if (k2 === undefined) k2 = k;
4
- var desc = Object.getOwnPropertyDescriptor(m, k);
5
- if (!desc || ("get" in desc ? !m.__esModule : desc.writable || desc.configurable)) {
6
- desc = { enumerable: true, get: function() { return m[k]; } };
7
- }
8
- Object.defineProperty(o, k2, desc);
9
- }) : (function(o, m, k, k2) {
10
- if (k2 === undefined) k2 = k;
11
- o[k2] = m[k];
12
- }));
13
- var __setModuleDefault = (this && this.__setModuleDefault) || (Object.create ? (function(o, v) {
14
- Object.defineProperty(o, "default", { enumerable: true, value: v });
15
- }) : function(o, v) {
16
- o["default"] = v;
17
- });
18
- var __importStar = (this && this.__importStar) || (function () {
19
- var ownKeys = function(o) {
20
- ownKeys = Object.getOwnPropertyNames || function (o) {
21
- var ar = [];
22
- for (var k in o) if (Object.prototype.hasOwnProperty.call(o, k)) ar[ar.length] = k;
23
- return ar;
24
- };
25
- return ownKeys(o);
26
- };
27
- return function (mod) {
28
- if (mod && mod.__esModule) return mod;
29
- var result = {};
30
- if (mod != null) for (var k = ownKeys(mod), i = 0; i < k.length; i++) if (k[i] !== "default") __createBinding(result, mod, k[i]);
31
- __setModuleDefault(result, mod);
32
- return result;
33
- };
34
- })();
35
- var __importDefault = (this && this.__importDefault) || function (mod) {
36
- return (mod && mod.__esModule) ? mod : { "default": mod };
37
- };
38
- Object.defineProperty(exports, "__esModule", { value: true });
39
- exports.runFirstRunExperience = void 0;
40
- const prompts_1 = __importDefault(require("prompts"));
41
- const chalk_1 = __importDefault(require("chalk"));
42
- const fs_1 = __importDefault(require("fs"));
43
- const path_1 = __importDefault(require("path"));
44
- const os_1 = __importDefault(require("os"));
45
- const script_sync_utils_1 = require("../utils/script-sync-utils");
46
- const project_fraim_paths_1 = require("../../core/utils/project-fraim-paths");
47
- const loadGlobalConfig = () => {
48
- const globalConfigPath = path_1.default.join((0, script_sync_utils_1.getUserFraimDir)(), 'config.json');
49
- if (!fs_1.default.existsSync(globalConfigPath)) {
50
- return null;
51
- }
52
- try {
53
- const config = JSON.parse(fs_1.default.readFileSync(globalConfigPath, 'utf8'));
54
- return {
55
- fraimKey: config.apiKey,
56
- githubToken: config.githubToken
57
- };
58
- }
59
- catch (e) {
60
- return null;
61
- }
62
- };
63
- const checkMCPConfigurations = () => {
64
- const sources = [];
65
- // Check Kiro MCP config
66
- const kiroConfigPath = path_1.default.join(os_1.default.homedir(), '.kiro', 'settings', 'mcp.json');
67
- if (fs_1.default.existsSync(kiroConfigPath)) {
68
- try {
69
- const kiroConfig = JSON.parse(fs_1.default.readFileSync(kiroConfigPath, 'utf8'));
70
- if (kiroConfig.mcpServers?.fraim) {
71
- sources.push('Kiro');
72
- }
73
- }
74
- catch (e) {
75
- // Ignore parsing errors
76
- }
77
- }
78
- // Check Claude Desktop / Cowork config
79
- const claudeConfigPath = path_1.default.join(os_1.default.homedir(), 'AppData', 'Roaming', 'Claude', 'claude_desktop_config.json');
80
- if (fs_1.default.existsSync(claudeConfigPath)) {
81
- try {
82
- const claudeConfig = JSON.parse(fs_1.default.readFileSync(claudeConfigPath, 'utf8'));
83
- if (claudeConfig.mcpServers?.fraim) {
84
- sources.push('Claude Desktop / Cowork');
85
- }
86
- }
87
- catch (e) {
88
- // Ignore parsing errors
89
- }
90
- }
91
- // Check macOS Claude Desktop / Cowork config path
92
- const claudeMacPath = path_1.default.join(os_1.default.homedir(), 'Library', 'Application Support', 'Claude', 'claude_desktop_config.json');
93
- if (fs_1.default.existsSync(claudeMacPath)) {
94
- try {
95
- const claudeConfig = JSON.parse(fs_1.default.readFileSync(claudeMacPath, 'utf8'));
96
- if (claudeConfig.mcpServers?.fraim) {
97
- sources.push('Claude Desktop / Cowork (macOS)');
98
- }
99
- }
100
- catch (e) {
101
- // Ignore parsing errors
102
- }
103
- }
104
- // Check VS Code MCP config (uses "servers" key)
105
- const vscodePaths = [
106
- path_1.default.join(os_1.default.homedir(), 'AppData', 'Roaming', 'Code', 'User', 'mcp.json'),
107
- path_1.default.join(os_1.default.homedir(), 'Library', 'Application Support', 'Code', 'User', 'mcp.json'),
108
- path_1.default.join(os_1.default.homedir(), '.config', 'Code', 'User', 'mcp.json')
109
- ];
110
- for (const vscodePath of vscodePaths) {
111
- if (fs_1.default.existsSync(vscodePath)) {
112
- try {
113
- const vscodeConfig = JSON.parse(fs_1.default.readFileSync(vscodePath, 'utf8'));
114
- if (vscodeConfig.servers?.fraim) {
115
- sources.push('VSCode');
116
- break;
117
- }
118
- }
119
- catch (e) {
120
- // Ignore parsing errors
121
- }
122
- }
123
- }
124
- return {
125
- found: sources.length > 0,
126
- sources
127
- };
128
- };
129
- const runFirstRunExperience = async () => {
130
- // Skip interactive setup in CI/Test environments
131
- if (process.env.CI === 'true' || process.env.TEST_MODE === 'true') {
132
- console.log(chalk_1.default.yellow('ℹ️ Skipping interactive first-run experience (CI/TEST_MODE detected)'));
133
- return;
134
- }
135
- console.log(chalk_1.default.blue('\n👋 Welcome to FRAIM! Let\'s get you set up.\n'));
136
- // Check for existing configuration
137
- const globalConfig = loadGlobalConfig();
138
- const mcpCheck = checkMCPConfigurations();
139
- if (globalConfig && globalConfig.fraimKey) {
140
- console.log(chalk_1.default.green('✅ Found existing FRAIM configuration'));
141
- if (mcpCheck.found) {
142
- console.log(chalk_1.default.green(`✅ Found FRAIM MCP configuration in: ${mcpCheck.sources.join(', ')}`));
143
- console.log(chalk_1.default.blue('🎉 You\'re all set! FRAIM is ready to use.\n'));
144
- }
145
- else {
146
- console.log(chalk_1.default.yellow('⚠️ FRAIM key found but no MCP configuration detected.'));
147
- console.log(chalk_1.default.cyan('💡 Consider running "fraim add-ide" to configure your IDE.\n'));
148
- }
149
- }
150
- else if (mcpCheck.found) {
151
- console.log(chalk_1.default.green(`✅ Found FRAIM MCP configuration in: ${mcpCheck.sources.join(', ')}`));
152
- console.log(chalk_1.default.yellow('⚠️ But no global FRAIM configuration found.'));
153
- console.log(chalk_1.default.cyan('💡 Your MCP setup looks good! Skipping key setup.\n'));
154
- }
155
- else {
156
- // No existing configuration found - ask for FRAIM key
157
- console.log(chalk_1.default.yellow('🔍 No existing FRAIM configuration detected.\n'));
158
- let response;
159
- try {
160
- response = await (0, prompts_1.default)({
161
- type: 'text',
162
- name: 'fraimKey',
163
- message: 'Do you have a FRAIM key? (Input key or press Enter to skip)',
164
- validate: (value) => true // Optional input
165
- });
166
- }
167
- catch (e) {
168
- console.warn(chalk_1.default.yellow('\n⚠️ Interactive prompts experienced an issue. Skipping setup.\n'));
169
- return;
170
- }
171
- if (response.fraimKey && response.fraimKey.trim().length > 0) {
172
- const key = response.fraimKey.trim();
173
- console.log(chalk_1.default.green('\n✅ Key received.'));
174
- console.log(chalk_1.default.yellow('\nPlease add the following to your IDE\'s MCP config (e.g. claude_desktop_config.json):'));
175
- const mcpConfig = {
176
- mcpServers: {
177
- fraim: {
178
- serverUrl: "https://fraim.wellnessatwork.me",
179
- headers: {
180
- "x-api-key": key
181
- }
182
- }
183
- }
184
- };
185
- console.log(chalk_1.default.cyan(JSON.stringify(mcpConfig, null, 2)));
186
- console.log('\n');
187
- }
188
- else {
189
- console.log(chalk_1.default.yellow('\nℹ️ If you need a key, please email sid.mathur@gmail.com to request one.\n'));
190
- }
191
- }
192
- // 2. Ask about Architecture Document
193
- const archResponse = await (0, prompts_1.default)({
194
- type: 'confirm',
195
- name: 'hasArchDoc',
196
- message: 'Do you have an architecture document for this project?',
197
- initial: false
198
- });
199
- if (!archResponse.hasArchDoc) {
200
- console.log(chalk_1.default.yellow('\n💡 To create an architecture document, please ask your IDE Agent:'));
201
- console.log(chalk_1.default.cyan(' "Run the bootstrap/create-architecture workflow"'));
202
- console.log(chalk_1.default.gray(' (This ensures your agent understands your codebase structure)\n'));
203
- }
204
- else {
205
- const pathResponse = await (0, prompts_1.default)({
206
- type: 'text',
207
- name: 'archDocPath',
208
- message: 'Please provide the relative path to your architecture document:',
209
- initial: 'docs/architecture.md',
210
- validate: (value) => value.length > 0 ? true : 'Path cannot be empty'
211
- });
212
- if (pathResponse.archDocPath) {
213
- const fs = await Promise.resolve().then(() => __importStar(require('fs')));
214
- const path = await Promise.resolve().then(() => __importStar(require('path')));
215
- const resolvedPath = path.resolve(process.cwd(), pathResponse.archDocPath);
216
- if (fs.existsSync(resolvedPath)) {
217
- try {
218
- const configPath = (0, project_fraim_paths_1.getWorkspaceConfigPath)(process.cwd());
219
- if (fs.existsSync(configPath)) {
220
- const configContent = JSON.parse(fs.readFileSync(configPath, 'utf-8'));
221
- if (!configContent.customizations)
222
- configContent.customizations = {};
223
- configContent.customizations.architectureDoc = pathResponse.archDocPath;
224
- fs.writeFileSync(configPath, JSON.stringify(configContent, null, 2));
225
- console.log(chalk_1.default.green(`\n✅ Linked architecture document: ${pathResponse.archDocPath}`));
226
- }
227
- else {
228
- console.log(chalk_1.default.red(`\n❌ ${(0, project_fraim_paths_1.getWorkspaceFraimDisplayPath)('config.json')} not found. Could not save preference.`));
229
- }
230
- }
231
- catch (e) {
232
- console.error(chalk_1.default.red('\n❌ Failed to update config:'), e);
233
- }
234
- }
235
- else {
236
- console.log(chalk_1.default.yellow(`\n⚠️ File not found at ${pathResponse.archDocPath}. We'll skip linking it for now.`));
237
- }
238
- }
239
- }
240
- console.log(chalk_1.default.green(`\n✅ Jobs are installed in ${(0, project_fraim_paths_1.getWorkspaceFraimDisplayPath)('ai-employee/jobs')} and ${(0, project_fraim_paths_1.getWorkspaceFraimDisplayPath)('ai-manager/jobs')}.\n`));
241
- };
242
- exports.runFirstRunExperience = runFirstRunExperience;
@@ -1,75 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.readFraimConfigRaw = readFraimConfigRaw;
4
- exports.mergeFraimConfig = mergeFraimConfig;
5
- exports.writeFraimConfigUpdate = writeFraimConfigUpdate;
6
- exports.writeFraimConfig = writeFraimConfig;
7
- const fs_1 = require("fs");
8
- const path_1 = require("path");
9
- const config_loader_1 = require("./config-loader");
10
- const types_1 = require("./types");
11
- function isPlainObject(value) {
12
- return typeof value === 'object' && value !== null && !Array.isArray(value);
13
- }
14
- function deepMerge(baseValue, updateValue) {
15
- if (updateValue === undefined) {
16
- return baseValue;
17
- }
18
- if (Array.isArray(updateValue)) {
19
- return [...updateValue];
20
- }
21
- if (!isPlainObject(updateValue)) {
22
- return updateValue;
23
- }
24
- const baseObject = isPlainObject(baseValue) ? baseValue : {};
25
- const merged = { ...baseObject };
26
- for (const [key, value] of Object.entries(updateValue)) {
27
- merged[key] = deepMerge(baseObject[key], value);
28
- }
29
- return merged;
30
- }
31
- function ensureWritableFraimConfigShape(rawConfig) {
32
- const config = { ...rawConfig };
33
- if (typeof config.version !== 'string' || config.version.trim().length === 0) {
34
- config.version = types_1.DEFAULT_FRAIM_CONFIG.version;
35
- }
36
- const projectConfig = isPlainObject(config.project) ? config.project : {};
37
- config.project = {
38
- ...types_1.DEFAULT_FRAIM_CONFIG.project,
39
- ...projectConfig
40
- };
41
- const customizationsConfig = isPlainObject(config.customizations) ? config.customizations : {};
42
- config.customizations = {
43
- ...types_1.DEFAULT_FRAIM_CONFIG.customizations,
44
- ...customizationsConfig
45
- };
46
- return config;
47
- }
48
- function readFraimConfigRaw(configPath) {
49
- if (!(0, fs_1.existsSync)(configPath)) {
50
- return {};
51
- }
52
- const parsed = JSON.parse((0, fs_1.readFileSync)(configPath, 'utf8'));
53
- if (!isPlainObject(parsed)) {
54
- throw new Error('FRAIM config must contain a JSON object at the top level.');
55
- }
56
- return parsed;
57
- }
58
- function mergeFraimConfig(existingRawConfig, update) {
59
- const rawConfig = ensureWritableFraimConfigShape(deepMerge(existingRawConfig, update));
60
- return {
61
- config: (0, config_loader_1.normalizeFraimConfig)(rawConfig),
62
- created: Object.keys(existingRawConfig).length === 0,
63
- rawConfig
64
- };
65
- }
66
- function writeFraimConfigUpdate(configPath, update) {
67
- const existingRawConfig = readFraimConfigRaw(configPath);
68
- const result = mergeFraimConfig(existingRawConfig, update);
69
- (0, fs_1.mkdirSync)((0, path_1.dirname)(configPath), { recursive: true });
70
- (0, fs_1.writeFileSync)(configPath, JSON.stringify(result.rawConfig, null, 2));
71
- return result;
72
- }
73
- function writeFraimConfig(configPath, config) {
74
- return writeFraimConfigUpdate(configPath, config);
75
- }
@@ -1,47 +0,0 @@
1
- "use strict";
2
- Object.defineProperty(exports, "__esModule", { value: true });
3
- exports.DEPRECATED_FRAIM_JOB_NAMES = void 0;
4
- exports.resolveFraimJobName = resolveFraimJobName;
5
- exports.isListableFraimJob = isListableFraimJob;
6
- const DEPRECATED_TO_CANONICAL_JOB_MAP = {
7
- 'learn-and-scale': 'upskill-employee',
8
- 'model-behavior': 'upskill-employee',
9
- 'promote-learning': 'upskill-employee',
10
- 'refine-jobs': 'upskill-employee',
11
- 'refine-skills': 'upskill-employee'
12
- };
13
- const DIRECT_JOB_ALIASES = {
14
- 'sleep on learnings': 'sleep-on-learnings'
15
- };
16
- exports.DEPRECATED_FRAIM_JOB_NAMES = new Set(Object.keys(DEPRECATED_TO_CANONICAL_JOB_MAP));
17
- function normalizeJobLookupInput(input) {
18
- return input.trim().toLowerCase().replace(/[_\s]+/g, '-');
19
- }
20
- function resolveFraimJobName(input) {
21
- const normalizedJobName = normalizeJobLookupInput(input);
22
- const directAliasTarget = DIRECT_JOB_ALIASES[input.trim().toLowerCase()];
23
- if (directAliasTarget) {
24
- return {
25
- requestedJobName: input,
26
- normalizedJobName,
27
- canonicalJobName: directAliasTarget
28
- };
29
- }
30
- const deprecatedAliasTarget = DEPRECATED_TO_CANONICAL_JOB_MAP[normalizedJobName];
31
- if (deprecatedAliasTarget) {
32
- return {
33
- requestedJobName: input,
34
- normalizedJobName,
35
- canonicalJobName: deprecatedAliasTarget,
36
- deprecatedAliasTarget
37
- };
38
- }
39
- return {
40
- requestedJobName: input,
41
- normalizedJobName,
42
- canonicalJobName: normalizedJobName
43
- };
44
- }
45
- function isListableFraimJob(jobName) {
46
- return !exports.DEPRECATED_FRAIM_JOB_NAMES.has(normalizeJobLookupInput(jobName));
47
- }