aios-core 4.1.0 → 4.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 (145) hide show
  1. package/.aios-core/.session/current-session.json +14 -0
  2. package/.aios-core/core/registry/registry-schema.json +166 -166
  3. package/.aios-core/core/registry/service-registry.json +6585 -6585
  4. package/.aios-core/data/entity-registry.yaml +208 -8
  5. package/.aios-core/data/registry-update-log.jsonl +165 -0
  6. package/.aios-core/development/scripts/approval-workflow.js +642 -642
  7. package/.aios-core/development/scripts/backup-manager.js +606 -606
  8. package/.aios-core/development/scripts/branch-manager.js +389 -389
  9. package/.aios-core/development/scripts/code-quality-improver.js +1311 -1311
  10. package/.aios-core/development/scripts/commit-message-generator.js +849 -849
  11. package/.aios-core/development/scripts/conflict-resolver.js +674 -674
  12. package/.aios-core/development/scripts/dependency-analyzer.js +637 -637
  13. package/.aios-core/development/scripts/diff-generator.js +351 -351
  14. package/.aios-core/development/scripts/elicitation-engine.js +384 -384
  15. package/.aios-core/development/scripts/elicitation-session-manager.js +299 -299
  16. package/.aios-core/development/scripts/git-wrapper.js +461 -461
  17. package/.aios-core/development/scripts/manifest-preview.js +244 -244
  18. package/.aios-core/development/scripts/metrics-tracker.js +775 -775
  19. package/.aios-core/development/scripts/modification-validator.js +554 -554
  20. package/.aios-core/development/scripts/pattern-learner.js +1224 -1224
  21. package/.aios-core/development/scripts/performance-analyzer.js +757 -757
  22. package/.aios-core/development/scripts/refactoring-suggester.js +1138 -1138
  23. package/.aios-core/development/scripts/rollback-handler.js +530 -530
  24. package/.aios-core/development/scripts/security-checker.js +358 -358
  25. package/.aios-core/development/scripts/template-engine.js +239 -239
  26. package/.aios-core/development/scripts/template-validator.js +278 -278
  27. package/.aios-core/development/scripts/test-generator.js +843 -843
  28. package/.aios-core/development/scripts/transaction-manager.js +589 -589
  29. package/.aios-core/development/scripts/usage-tracker.js +673 -673
  30. package/.aios-core/development/scripts/validate-filenames.js +226 -226
  31. package/.aios-core/development/scripts/version-tracker.js +526 -526
  32. package/.aios-core/development/scripts/yaml-validator.js +396 -396
  33. package/.aios-core/development/tasks/validate-next-story.md +99 -2
  34. package/.aios-core/development/templates/service-template/README.md.hbs +158 -158
  35. package/.aios-core/development/templates/service-template/__tests__/index.test.ts.hbs +237 -237
  36. package/.aios-core/development/templates/service-template/client.ts.hbs +403 -403
  37. package/.aios-core/development/templates/service-template/errors.ts.hbs +182 -182
  38. package/.aios-core/development/templates/service-template/index.ts.hbs +120 -120
  39. package/.aios-core/development/templates/service-template/package.json.hbs +87 -87
  40. package/.aios-core/development/templates/service-template/types.ts.hbs +145 -145
  41. package/.aios-core/development/templates/squad-template/LICENSE +21 -21
  42. package/.aios-core/docs/SHARD-TRANSLATION-GUIDE.md +335 -0
  43. package/.aios-core/docs/component-creation-guide.md +458 -0
  44. package/.aios-core/docs/session-update-pattern.md +307 -0
  45. package/.aios-core/docs/standards/AIOS-FRAMEWORK-MASTER.md +1963 -0
  46. package/.aios-core/docs/standards/AIOS-LIVRO-DE-OURO-V2.1-SUMMARY.md +1190 -0
  47. package/.aios-core/docs/standards/AIOS-LIVRO-DE-OURO-V2.1.md +439 -0
  48. package/.aios-core/docs/standards/AIOS-LIVRO-DE-OURO.md +5398 -0
  49. package/.aios-core/docs/standards/V3-ARCHITECTURAL-DECISIONS.md +523 -0
  50. package/.aios-core/docs/template-syntax.md +267 -0
  51. package/.aios-core/docs/troubleshooting-guide.md +625 -0
  52. package/.aios-core/infrastructure/templates/aios-sync.yaml.template +193 -193
  53. package/.aios-core/infrastructure/templates/coderabbit.yaml.template +279 -279
  54. package/.aios-core/infrastructure/templates/github-workflows/ci.yml.template +169 -169
  55. package/.aios-core/infrastructure/templates/github-workflows/pr-automation.yml.template +330 -330
  56. package/.aios-core/infrastructure/templates/github-workflows/release.yml.template +196 -196
  57. package/.aios-core/infrastructure/templates/gitignore/gitignore-aios-base.tmpl +63 -63
  58. package/.aios-core/infrastructure/templates/gitignore/gitignore-brownfield-merge.tmpl +18 -18
  59. package/.aios-core/infrastructure/templates/gitignore/gitignore-node.tmpl +85 -85
  60. package/.aios-core/infrastructure/templates/gitignore/gitignore-python.tmpl +145 -145
  61. package/.aios-core/infrastructure/tests/utilities-audit-results.json +501 -0
  62. package/.aios-core/install-manifest.yaml +101 -101
  63. package/.aios-core/local-config.yaml.template +70 -70
  64. package/.aios-core/manifests/agents.csv +29 -0
  65. package/.aios-core/manifests/schema/manifest-schema.json +190 -190
  66. package/.aios-core/manifests/tasks.csv +198 -0
  67. package/.aios-core/manifests/workers.csv +204 -0
  68. package/.aios-core/monitor/hooks/lib/__init__.py +1 -1
  69. package/.aios-core/monitor/hooks/lib/enrich.py +58 -58
  70. package/.aios-core/monitor/hooks/lib/send_event.py +47 -47
  71. package/.aios-core/monitor/hooks/notification.py +29 -29
  72. package/.aios-core/monitor/hooks/post_tool_use.py +45 -45
  73. package/.aios-core/monitor/hooks/pre_compact.py +29 -29
  74. package/.aios-core/monitor/hooks/pre_tool_use.py +40 -40
  75. package/.aios-core/monitor/hooks/stop.py +29 -29
  76. package/.aios-core/monitor/hooks/subagent_stop.py +29 -29
  77. package/.aios-core/monitor/hooks/user_prompt_submit.py +38 -38
  78. package/.aios-core/product/templates/adr.hbs +125 -125
  79. package/.aios-core/product/templates/component-react-tmpl.tsx +98 -98
  80. package/.aios-core/product/templates/dbdr.hbs +241 -241
  81. package/.aios-core/product/templates/engine/schemas/adr.schema.json +102 -102
  82. package/.aios-core/product/templates/engine/schemas/dbdr.schema.json +205 -205
  83. package/.aios-core/product/templates/engine/schemas/epic.schema.json +175 -175
  84. package/.aios-core/product/templates/engine/schemas/pmdr.schema.json +175 -175
  85. package/.aios-core/product/templates/engine/schemas/prd-v2.schema.json +300 -300
  86. package/.aios-core/product/templates/engine/schemas/prd.schema.json +152 -152
  87. package/.aios-core/product/templates/engine/schemas/story.schema.json +222 -222
  88. package/.aios-core/product/templates/engine/schemas/task.schema.json +154 -154
  89. package/.aios-core/product/templates/epic.hbs +212 -212
  90. package/.aios-core/product/templates/eslintrc-security.json +32 -32
  91. package/.aios-core/product/templates/github-actions-cd.yml +212 -212
  92. package/.aios-core/product/templates/github-actions-ci.yml +172 -172
  93. package/.aios-core/product/templates/pmdr.hbs +186 -186
  94. package/.aios-core/product/templates/prd-v2.0.hbs +216 -216
  95. package/.aios-core/product/templates/prd.hbs +201 -201
  96. package/.aios-core/product/templates/shock-report-tmpl.html +502 -502
  97. package/.aios-core/product/templates/story.hbs +263 -263
  98. package/.aios-core/product/templates/task.hbs +170 -170
  99. package/.aios-core/product/templates/tmpl-comment-on-examples.sql +158 -158
  100. package/.aios-core/product/templates/tmpl-migration-script.sql +91 -91
  101. package/.aios-core/product/templates/tmpl-rls-granular-policies.sql +104 -104
  102. package/.aios-core/product/templates/tmpl-rls-kiss-policy.sql +10 -10
  103. package/.aios-core/product/templates/tmpl-rls-roles.sql +135 -135
  104. package/.aios-core/product/templates/tmpl-rls-simple.sql +77 -77
  105. package/.aios-core/product/templates/tmpl-rls-tenant.sql +152 -152
  106. package/.aios-core/product/templates/tmpl-rollback-script.sql +77 -77
  107. package/.aios-core/product/templates/tmpl-seed-data.sql +140 -140
  108. package/.aios-core/product/templates/tmpl-smoke-test.sql +16 -16
  109. package/.aios-core/product/templates/tmpl-staging-copy-merge.sql +139 -139
  110. package/.aios-core/product/templates/tmpl-stored-proc.sql +140 -140
  111. package/.aios-core/product/templates/tmpl-trigger.sql +152 -152
  112. package/.aios-core/product/templates/tmpl-view-materialized.sql +133 -133
  113. package/.aios-core/product/templates/tmpl-view.sql +177 -177
  114. package/.aios-core/product/templates/token-exports-css-tmpl.css +240 -240
  115. package/.aios-core/quality/schemas/quality-metrics.schema.json +233 -233
  116. package/.aios-core/scripts/migrate-framework-docs.sh +300 -300
  117. package/.aios-core/scripts/pm.sh +0 -0
  118. package/.claude/hooks/enforce-architecture-first.py +196 -196
  119. package/.claude/hooks/mind-clone-governance.py +192 -192
  120. package/.claude/hooks/read-protection.py +151 -151
  121. package/.claude/hooks/slug-validation.py +176 -176
  122. package/.claude/hooks/sql-governance.py +182 -182
  123. package/.claude/hooks/write-path-validation.py +194 -194
  124. package/.claude/rules/agent-authority.md +105 -0
  125. package/.claude/rules/coderabbit-integration.md +93 -0
  126. package/.claude/rules/ids-principles.md +112 -0
  127. package/.claude/rules/story-lifecycle.md +139 -0
  128. package/.claude/rules/workflow-execution.md +150 -0
  129. package/LICENSE +48 -48
  130. package/bin/aios-minimal.js +0 -0
  131. package/bin/aios.js +0 -0
  132. package/package.json +1 -1
  133. package/packages/aios-install/bin/aios-install.js +0 -0
  134. package/packages/aios-install/bin/edmcp.js +0 -0
  135. package/packages/aios-pro-cli/bin/aios-pro.js +0 -0
  136. package/packages/installer/src/wizard/pro-setup.js +433 -49
  137. package/scripts/check-markdown-links.py +352 -352
  138. package/scripts/code-intel-health-check.js +343 -0
  139. package/scripts/dashboard-parallel-dev.sh +0 -0
  140. package/scripts/dashboard-parallel-phase3.sh +0 -0
  141. package/scripts/dashboard-parallel-phase4.sh +0 -0
  142. package/scripts/glue/README.md +355 -0
  143. package/scripts/glue/compose-agent-prompt.cjs +362 -0
  144. package/scripts/install-monitor-hooks.sh +0 -0
  145. package/.aios-core/lib/build.json +0 -1
@@ -1,390 +1,390 @@
1
- const GitWrapper = require('./git-wrapper');
2
- const chalk = require('chalk');
3
- const inquirer = require('inquirer');
4
- const path = require('path');
5
-
6
- /**
7
- * Manages git branches for meta-agent modifications
8
- */
9
- class BranchManager {
10
- constructor(options = {}) {
11
- this.git = new GitWrapper(options);
12
- this.branchPrefix = options.branchPrefix || 'meta-agent/';
13
- this.maxBranches = options.maxBranches || 10;
14
- this.autoCleanup = options.autoCleanup !== false;
15
- }
16
-
17
- /**
18
- * Create a modification branch with proper naming
19
- * @param {Object} modification - Modification details
20
- * @returns {Promise<Object>} Branch creation result
21
- */
22
- async createModificationBranch(modification) {
23
- const {
24
- type,
25
- target,
26
- action,
27
- ticketId
28
- } = modification;
29
-
30
- // Generate branch name
31
- const timestamp = Date.now();
32
- const sanitizedTarget = target.replace(/[^a-zA-Z0-9-]/g, '-').toLowerCase();
33
- const sanitizedAction = action.replace(/[^a-zA-Z0-9-]/g, '-').toLowerCase();
34
-
35
- let branchName = `${this.branchPrefix}${type}/${sanitizedTarget}-${sanitizedAction}-${timestamp}`;
36
-
37
- if (ticketId) {
38
- branchName = `${this.branchPrefix}${ticketId}/${type}-${sanitizedTarget}`;
39
- }
40
-
41
- try {
42
- // Ensure we're on the default branch first
43
- const currentBranch = await this.git.getCurrentBranch();
44
- if (currentBranch !== this.git.defaultBranch) {
45
- console.log(chalk.yellow(`Switching to ${this.git.defaultBranch} before creating new branch`));
46
- await this.git.checkoutBranch(this.git.defaultBranch);
47
- }
48
-
49
- // Create and checkout the new branch
50
- await this.git.createBranch(branchName, true);
51
-
52
- return {
53
- success: true,
54
- branchName,
55
- baseBranch: this.git.defaultBranch,
56
- timestamp: new Date(timestamp).toISOString()
57
- };
58
- } catch (error) {
59
- console.error(chalk.red(`Failed to create branch: ${error.message}`));
60
- return {
61
- success: false,
62
- error: error.message
63
- };
64
- }
65
- }
66
-
67
- /**
68
- * Get all modification branches
69
- * @returns {Promise<Array>} List of modification branches
70
- */
71
- async getModificationBranches() {
72
- try {
73
- const output = await this.git.execGit('branch -a');
74
- const branches = output.split('\n')
75
- .map(line => line.trim().replace('* ', ''))
76
- .filter(branch => branch.startsWith(this.branchPrefix));
77
-
78
- const branchDetails = [];
79
- for (const branch of branches) {
80
- const lastCommit = await this.git.execGit(`log -1 --format="%H|%at|%s" ${branch}`);
81
- const [hash, timestamp, subject] = lastCommit.split('|');
82
-
83
- branchDetails.push({
84
- name: branch,
85
- lastCommitHash: hash,
86
- lastCommitDate: new Date(parseInt(timestamp) * 1000),
87
- lastCommitMessage: subject,
88
- age: this.calculateAge(parseInt(timestamp) * 1000)
89
- });
90
- }
91
-
92
- return branchDetails.sort((a, b) => b.lastCommitDate - a.lastCommitDate);
93
- } catch (error) {
94
- console.error(chalk.red(`Failed to get modification branches: ${error.message}`));
95
- return [];
96
- }
97
- }
98
-
99
- /**
100
- * Switch to a modification branch
101
- * @param {string} branchName - Branch to switch to
102
- * @returns {Promise<Object>} Switch result
103
- */
104
- async switchToBranch(branchName) {
105
- try {
106
- // Check for uncommitted changes
107
- const status = await this.git.getStatus();
108
- if (!status.clean) {
109
- const { action } = await inquirer.prompt([{
110
- type: 'list',
111
- name: 'action',
112
- message: 'You have uncommitted changes. What would you like to do?',
113
- choices: [
114
- { name: 'Stash changes and switch', value: 'stash' },
115
- { name: 'Commit changes first', value: 'commit' },
116
- { name: 'Cancel', value: 'cancel' }
117
- ]
118
- }]);
119
-
120
- if (action === 'cancel') {
121
- return { success: false, reason: 'User cancelled' };
122
- }
123
-
124
- if (action === 'stash') {
125
- await this.git.stash(`Auto-stash before switching to ${branchName}`);
126
- } else if (action === 'commit') {
127
- await this.git.stageFiles(['.']);
128
- await this.git.commit('WIP: Auto-commit before branch switch');
129
- }
130
- }
131
-
132
- await this.git.checkoutBranch(branchName);
133
- return { success: true, branchName };
134
- } catch (error) {
135
- return { success: false, error: error.message };
136
- }
137
- }
138
-
139
- /**
140
- * Merge modification branch back to main
141
- * @param {string} branchName - Branch to merge
142
- * @param {Object} options - Merge options
143
- * @returns {Promise<Object>} Merge result
144
- */
145
- async mergeModificationBranch(branchName, options = {}) {
146
- try {
147
- // Switch to target branch
148
- const targetBranch = options.targetBranch || this.git.defaultBranch;
149
- await this.git.checkoutBranch(targetBranch);
150
-
151
- // Attempt merge
152
- const mergeResult = await this.git.mergeBranch(branchName, {
153
- message: options.message || `Merge modification branch '${branchName}'`,
154
- noFastForward: true
155
- });
156
-
157
- if (mergeResult.success) {
158
- // Optionally delete the branch after successful merge
159
- if (options.deleteBranch) {
160
- await this.deleteBranch(branchName);
161
- }
162
-
163
- return {
164
- success: true,
165
- message: 'Branch merged successfully',
166
- targetBranch
167
- };
168
- } else {
169
- return {
170
- success: false,
171
- conflicts: mergeResult.conflicts,
172
- message: 'Merge conflicts detected'
173
- };
174
- }
175
- } catch (error) {
176
- return {
177
- success: false,
178
- error: error.message
179
- };
180
- }
181
- }
182
-
183
- /**
184
- * Delete a modification branch
185
- * @param {string} branchName - Branch to delete
186
- * @param {boolean} force - Force delete even if not merged
187
- * @returns {Promise<Object>} Deletion result
188
- */
189
- async deleteBranch(branchName, force = false) {
190
- try {
191
- const flag = force ? '-D' : '-d';
192
- await this.git.execGit(`branch ${flag} ${branchName}`);
193
-
194
- console.log(chalk.green(`✅ Deleted branch: ${branchName}`));
195
- return { success: true };
196
- } catch (error) {
197
- if (error.message.includes('not fully merged')) {
198
- console.error(chalk.red('Branch not fully merged. Use force=true to delete anyway.'));
199
- }
200
- return { success: false, error: error.message };
201
- }
202
- }
203
-
204
- /**
205
- * Clean up old modification branches
206
- * @param {number} daysOld - Delete branches older than this many days
207
- * @returns {Promise<Object>} Cleanup result
208
- */
209
- async cleanupOldBranches(daysOld = 30) {
210
- const branches = await this.getModificationBranches();
211
- const cutoffDate = new Date();
212
- cutoffDate.setDate(cutoffDate.getDate() - daysOld);
213
-
214
- const toDelete = branches.filter(branch =>
215
- branch.lastCommitDate < cutoffDate &&
216
- branch.name !== await this.git.getCurrentBranch()
217
- );
218
-
219
- if (toDelete.length === 0) {
220
- console.log(chalk.yellow('No old branches to clean up'));
221
- return { deleted: 0 };
222
- }
223
-
224
- console.log(chalk.blue(`Found ${toDelete.length} branches older than ${daysOld} days:`));
225
- toDelete.forEach(branch => {
226
- console.log(chalk.gray(` - ${branch.name} (${branch.age})`));
227
- });
228
-
229
- const { confirm } = await inquirer.prompt([{
230
- type: 'confirm',
231
- name: 'confirm',
232
- message: `Delete ${toDelete.length} old branches?`,
233
- default: false
234
- }]);
235
-
236
- if (!confirm) {
237
- return { deleted: 0, cancelled: true };
238
- }
239
-
240
- let deleted = 0;
241
- for (const branch of toDelete) {
242
- const result = await this.deleteBranch(branch.name, true);
243
- if (result.success) deleted++;
244
- }
245
-
246
- return { deleted, total: toDelete.length };
247
- }
248
-
249
- /**
250
- * Create a branch protection strategy
251
- * @param {string} branchName - Branch to protect
252
- * @returns {Promise<Object>} Protection result
253
- */
254
- async protectBranch(branchName) {
255
- // This would integrate with GitHub/GitLab API for real protection
256
- // For now, we'll track it locally
257
- const protectionFile = path.join(
258
- this.git.rootPath,
259
- '.git',
260
- 'aios-branch-protection.json'
261
- );
262
-
263
- try {
264
- let protections = {};
265
- try {
266
- const content = await require('fs').promises.readFile(protectionFile, 'utf-8');
267
- protections = JSON.parse(content);
268
- } catch (_e) {
269
- // File doesn't exist yet
270
- }
271
-
272
- protections[branchName] = {
273
- protected: true,
274
- requiredReviews: 1,
275
- dismissStaleReviews: true,
276
- requireUpToDate: true,
277
- protectedAt: new Date().toISOString()
278
- };
279
-
280
- await require('fs').promises.writeFile(
281
- protectionFile,
282
- JSON.stringify(protections, null, 2)
283
- );
284
-
285
- console.log(chalk.green(`✅ Branch protection enabled for: ${branchName}`));
286
- return { success: true, protection: protections[branchName] };
287
- } catch (error) {
288
- return { success: false, error: error.message };
289
- }
290
- }
291
-
292
- /**
293
- * Get branch comparison
294
- * @param {string} branch1 - First branch
295
- * @param {string} branch2 - Second branch (default: main)
296
- * @returns {Promise<Object>} Comparison result
297
- */
298
- async compareBranches(branch1, branch2 = null) {
299
- const targetBranch = branch2 || this.git.defaultBranch;
300
-
301
- try {
302
- // Get commits ahead/behind
303
- const ahead = await this.git.execGit(
304
- `rev-list --count ${targetBranch}..${branch1}`
305
- );
306
- const behind = await this.git.execGit(
307
- `rev-list --count ${branch1}..${targetBranch}`
308
- );
309
-
310
- // Get changed files
311
- const changedFiles = await this.git.getDiff({
312
- from: targetBranch,
313
- to: branch1,
314
- nameOnly: true
315
- });
316
-
317
- return {
318
- branch: branch1,
319
- compareTo: targetBranch,
320
- ahead: parseInt(ahead),
321
- behind: parseInt(behind),
322
- changedFiles: changedFiles.split('\n').filter(Boolean),
323
- canFastForward: parseInt(behind) === 0
324
- };
325
- } catch (error) {
326
- return {
327
- success: false,
328
- error: error.message
329
- };
330
- }
331
- }
332
-
333
- /**
334
- * Calculate age string from timestamp
335
- * @private
336
- */
337
- calculateAge(timestamp) {
338
- const now = Date.now();
339
- const diff = now - timestamp;
340
- const days = Math.floor(diff / (1000 * 60 * 60 * 24));
341
- const hours = Math.floor((diff % (1000 * 60 * 60 * 24)) / (1000 * 60 * 60));
342
-
343
- if (days > 0) {
344
- return `${days} day${days > 1 ? 's' : ''} ago`;
345
- } else if (hours > 0) {
346
- return `${hours} hour${hours > 1 ? 's' : ''} ago`;
347
- } else {
348
- return 'recently';
349
- }
350
- }
351
-
352
- /**
353
- * Create branch strategy for different modification types
354
- * @param {string} modificationType - Type of modification
355
- * @returns {Object} Branch strategy
356
- */
357
- getBranchStrategy(modificationType) {
358
- const strategies = {
359
- enhancement: {
360
- prefix: 'feature/',
361
- baseFrom: 'main',
362
- protectByDefault: false,
363
- autoMerge: false
364
- },
365
- bugfix: {
366
- prefix: 'fix/',
367
- baseFrom: 'main',
368
- protectByDefault: false,
369
- autoMerge: true
370
- },
371
- experiment: {
372
- prefix: 'experiment/',
373
- baseFrom: 'develop',
374
- protectByDefault: false,
375
- autoMerge: false
376
- },
377
- 'self-modification': {
378
- prefix: 'self-mod/',
379
- baseFrom: 'main',
380
- protectByDefault: true,
381
- autoMerge: false,
382
- requireApproval: true
383
- }
384
- };
385
-
386
- return strategies[modificationType] || strategies.enhancement;
387
- }
388
- }
389
-
1
+ const GitWrapper = require('./git-wrapper');
2
+ const chalk = require('chalk');
3
+ const inquirer = require('inquirer');
4
+ const path = require('path');
5
+
6
+ /**
7
+ * Manages git branches for meta-agent modifications
8
+ */
9
+ class BranchManager {
10
+ constructor(options = {}) {
11
+ this.git = new GitWrapper(options);
12
+ this.branchPrefix = options.branchPrefix || 'meta-agent/';
13
+ this.maxBranches = options.maxBranches || 10;
14
+ this.autoCleanup = options.autoCleanup !== false;
15
+ }
16
+
17
+ /**
18
+ * Create a modification branch with proper naming
19
+ * @param {Object} modification - Modification details
20
+ * @returns {Promise<Object>} Branch creation result
21
+ */
22
+ async createModificationBranch(modification) {
23
+ const {
24
+ type,
25
+ target,
26
+ action,
27
+ ticketId
28
+ } = modification;
29
+
30
+ // Generate branch name
31
+ const timestamp = Date.now();
32
+ const sanitizedTarget = target.replace(/[^a-zA-Z0-9-]/g, '-').toLowerCase();
33
+ const sanitizedAction = action.replace(/[^a-zA-Z0-9-]/g, '-').toLowerCase();
34
+
35
+ let branchName = `${this.branchPrefix}${type}/${sanitizedTarget}-${sanitizedAction}-${timestamp}`;
36
+
37
+ if (ticketId) {
38
+ branchName = `${this.branchPrefix}${ticketId}/${type}-${sanitizedTarget}`;
39
+ }
40
+
41
+ try {
42
+ // Ensure we're on the default branch first
43
+ const currentBranch = await this.git.getCurrentBranch();
44
+ if (currentBranch !== this.git.defaultBranch) {
45
+ console.log(chalk.yellow(`Switching to ${this.git.defaultBranch} before creating new branch`));
46
+ await this.git.checkoutBranch(this.git.defaultBranch);
47
+ }
48
+
49
+ // Create and checkout the new branch
50
+ await this.git.createBranch(branchName, true);
51
+
52
+ return {
53
+ success: true,
54
+ branchName,
55
+ baseBranch: this.git.defaultBranch,
56
+ timestamp: new Date(timestamp).toISOString()
57
+ };
58
+ } catch (error) {
59
+ console.error(chalk.red(`Failed to create branch: ${error.message}`));
60
+ return {
61
+ success: false,
62
+ error: error.message
63
+ };
64
+ }
65
+ }
66
+
67
+ /**
68
+ * Get all modification branches
69
+ * @returns {Promise<Array>} List of modification branches
70
+ */
71
+ async getModificationBranches() {
72
+ try {
73
+ const output = await this.git.execGit('branch -a');
74
+ const branches = output.split('\n')
75
+ .map(line => line.trim().replace('* ', ''))
76
+ .filter(branch => branch.startsWith(this.branchPrefix));
77
+
78
+ const branchDetails = [];
79
+ for (const branch of branches) {
80
+ const lastCommit = await this.git.execGit(`log -1 --format="%H|%at|%s" ${branch}`);
81
+ const [hash, timestamp, subject] = lastCommit.split('|');
82
+
83
+ branchDetails.push({
84
+ name: branch,
85
+ lastCommitHash: hash,
86
+ lastCommitDate: new Date(parseInt(timestamp) * 1000),
87
+ lastCommitMessage: subject,
88
+ age: this.calculateAge(parseInt(timestamp) * 1000)
89
+ });
90
+ }
91
+
92
+ return branchDetails.sort((a, b) => b.lastCommitDate - a.lastCommitDate);
93
+ } catch (error) {
94
+ console.error(chalk.red(`Failed to get modification branches: ${error.message}`));
95
+ return [];
96
+ }
97
+ }
98
+
99
+ /**
100
+ * Switch to a modification branch
101
+ * @param {string} branchName - Branch to switch to
102
+ * @returns {Promise<Object>} Switch result
103
+ */
104
+ async switchToBranch(branchName) {
105
+ try {
106
+ // Check for uncommitted changes
107
+ const status = await this.git.getStatus();
108
+ if (!status.clean) {
109
+ const { action } = await inquirer.prompt([{
110
+ type: 'list',
111
+ name: 'action',
112
+ message: 'You have uncommitted changes. What would you like to do?',
113
+ choices: [
114
+ { name: 'Stash changes and switch', value: 'stash' },
115
+ { name: 'Commit changes first', value: 'commit' },
116
+ { name: 'Cancel', value: 'cancel' }
117
+ ]
118
+ }]);
119
+
120
+ if (action === 'cancel') {
121
+ return { success: false, reason: 'User cancelled' };
122
+ }
123
+
124
+ if (action === 'stash') {
125
+ await this.git.stash(`Auto-stash before switching to ${branchName}`);
126
+ } else if (action === 'commit') {
127
+ await this.git.stageFiles(['.']);
128
+ await this.git.commit('WIP: Auto-commit before branch switch');
129
+ }
130
+ }
131
+
132
+ await this.git.checkoutBranch(branchName);
133
+ return { success: true, branchName };
134
+ } catch (error) {
135
+ return { success: false, error: error.message };
136
+ }
137
+ }
138
+
139
+ /**
140
+ * Merge modification branch back to main
141
+ * @param {string} branchName - Branch to merge
142
+ * @param {Object} options - Merge options
143
+ * @returns {Promise<Object>} Merge result
144
+ */
145
+ async mergeModificationBranch(branchName, options = {}) {
146
+ try {
147
+ // Switch to target branch
148
+ const targetBranch = options.targetBranch || this.git.defaultBranch;
149
+ await this.git.checkoutBranch(targetBranch);
150
+
151
+ // Attempt merge
152
+ const mergeResult = await this.git.mergeBranch(branchName, {
153
+ message: options.message || `Merge modification branch '${branchName}'`,
154
+ noFastForward: true
155
+ });
156
+
157
+ if (mergeResult.success) {
158
+ // Optionally delete the branch after successful merge
159
+ if (options.deleteBranch) {
160
+ await this.deleteBranch(branchName);
161
+ }
162
+
163
+ return {
164
+ success: true,
165
+ message: 'Branch merged successfully',
166
+ targetBranch
167
+ };
168
+ } else {
169
+ return {
170
+ success: false,
171
+ conflicts: mergeResult.conflicts,
172
+ message: 'Merge conflicts detected'
173
+ };
174
+ }
175
+ } catch (error) {
176
+ return {
177
+ success: false,
178
+ error: error.message
179
+ };
180
+ }
181
+ }
182
+
183
+ /**
184
+ * Delete a modification branch
185
+ * @param {string} branchName - Branch to delete
186
+ * @param {boolean} force - Force delete even if not merged
187
+ * @returns {Promise<Object>} Deletion result
188
+ */
189
+ async deleteBranch(branchName, force = false) {
190
+ try {
191
+ const flag = force ? '-D' : '-d';
192
+ await this.git.execGit(`branch ${flag} ${branchName}`);
193
+
194
+ console.log(chalk.green(`✅ Deleted branch: ${branchName}`));
195
+ return { success: true };
196
+ } catch (error) {
197
+ if (error.message.includes('not fully merged')) {
198
+ console.error(chalk.red('Branch not fully merged. Use force=true to delete anyway.'));
199
+ }
200
+ return { success: false, error: error.message };
201
+ }
202
+ }
203
+
204
+ /**
205
+ * Clean up old modification branches
206
+ * @param {number} daysOld - Delete branches older than this many days
207
+ * @returns {Promise<Object>} Cleanup result
208
+ */
209
+ async cleanupOldBranches(daysOld = 30) {
210
+ const branches = await this.getModificationBranches();
211
+ const cutoffDate = new Date();
212
+ cutoffDate.setDate(cutoffDate.getDate() - daysOld);
213
+
214
+ const toDelete = branches.filter(branch =>
215
+ branch.lastCommitDate < cutoffDate &&
216
+ branch.name !== await this.git.getCurrentBranch()
217
+ );
218
+
219
+ if (toDelete.length === 0) {
220
+ console.log(chalk.yellow('No old branches to clean up'));
221
+ return { deleted: 0 };
222
+ }
223
+
224
+ console.log(chalk.blue(`Found ${toDelete.length} branches older than ${daysOld} days:`));
225
+ toDelete.forEach(branch => {
226
+ console.log(chalk.gray(` - ${branch.name} (${branch.age})`));
227
+ });
228
+
229
+ const { confirm } = await inquirer.prompt([{
230
+ type: 'confirm',
231
+ name: 'confirm',
232
+ message: `Delete ${toDelete.length} old branches?`,
233
+ default: false
234
+ }]);
235
+
236
+ if (!confirm) {
237
+ return { deleted: 0, cancelled: true };
238
+ }
239
+
240
+ let deleted = 0;
241
+ for (const branch of toDelete) {
242
+ const result = await this.deleteBranch(branch.name, true);
243
+ if (result.success) deleted++;
244
+ }
245
+
246
+ return { deleted, total: toDelete.length };
247
+ }
248
+
249
+ /**
250
+ * Create a branch protection strategy
251
+ * @param {string} branchName - Branch to protect
252
+ * @returns {Promise<Object>} Protection result
253
+ */
254
+ async protectBranch(branchName) {
255
+ // This would integrate with GitHub/GitLab API for real protection
256
+ // For now, we'll track it locally
257
+ const protectionFile = path.join(
258
+ this.git.rootPath,
259
+ '.git',
260
+ 'aios-branch-protection.json'
261
+ );
262
+
263
+ try {
264
+ let protections = {};
265
+ try {
266
+ const content = await require('fs').promises.readFile(protectionFile, 'utf-8');
267
+ protections = JSON.parse(content);
268
+ } catch (_e) {
269
+ // File doesn't exist yet
270
+ }
271
+
272
+ protections[branchName] = {
273
+ protected: true,
274
+ requiredReviews: 1,
275
+ dismissStaleReviews: true,
276
+ requireUpToDate: true,
277
+ protectedAt: new Date().toISOString()
278
+ };
279
+
280
+ await require('fs').promises.writeFile(
281
+ protectionFile,
282
+ JSON.stringify(protections, null, 2)
283
+ );
284
+
285
+ console.log(chalk.green(`✅ Branch protection enabled for: ${branchName}`));
286
+ return { success: true, protection: protections[branchName] };
287
+ } catch (error) {
288
+ return { success: false, error: error.message };
289
+ }
290
+ }
291
+
292
+ /**
293
+ * Get branch comparison
294
+ * @param {string} branch1 - First branch
295
+ * @param {string} branch2 - Second branch (default: main)
296
+ * @returns {Promise<Object>} Comparison result
297
+ */
298
+ async compareBranches(branch1, branch2 = null) {
299
+ const targetBranch = branch2 || this.git.defaultBranch;
300
+
301
+ try {
302
+ // Get commits ahead/behind
303
+ const ahead = await this.git.execGit(
304
+ `rev-list --count ${targetBranch}..${branch1}`
305
+ );
306
+ const behind = await this.git.execGit(
307
+ `rev-list --count ${branch1}..${targetBranch}`
308
+ );
309
+
310
+ // Get changed files
311
+ const changedFiles = await this.git.getDiff({
312
+ from: targetBranch,
313
+ to: branch1,
314
+ nameOnly: true
315
+ });
316
+
317
+ return {
318
+ branch: branch1,
319
+ compareTo: targetBranch,
320
+ ahead: parseInt(ahead),
321
+ behind: parseInt(behind),
322
+ changedFiles: changedFiles.split('\n').filter(Boolean),
323
+ canFastForward: parseInt(behind) === 0
324
+ };
325
+ } catch (error) {
326
+ return {
327
+ success: false,
328
+ error: error.message
329
+ };
330
+ }
331
+ }
332
+
333
+ /**
334
+ * Calculate age string from timestamp
335
+ * @private
336
+ */
337
+ calculateAge(timestamp) {
338
+ const now = Date.now();
339
+ const diff = now - timestamp;
340
+ const days = Math.floor(diff / (1000 * 60 * 60 * 24));
341
+ const hours = Math.floor((diff % (1000 * 60 * 60 * 24)) / (1000 * 60 * 60));
342
+
343
+ if (days > 0) {
344
+ return `${days} day${days > 1 ? 's' : ''} ago`;
345
+ } else if (hours > 0) {
346
+ return `${hours} hour${hours > 1 ? 's' : ''} ago`;
347
+ } else {
348
+ return 'recently';
349
+ }
350
+ }
351
+
352
+ /**
353
+ * Create branch strategy for different modification types
354
+ * @param {string} modificationType - Type of modification
355
+ * @returns {Object} Branch strategy
356
+ */
357
+ getBranchStrategy(modificationType) {
358
+ const strategies = {
359
+ enhancement: {
360
+ prefix: 'feature/',
361
+ baseFrom: 'main',
362
+ protectByDefault: false,
363
+ autoMerge: false
364
+ },
365
+ bugfix: {
366
+ prefix: 'fix/',
367
+ baseFrom: 'main',
368
+ protectByDefault: false,
369
+ autoMerge: true
370
+ },
371
+ experiment: {
372
+ prefix: 'experiment/',
373
+ baseFrom: 'develop',
374
+ protectByDefault: false,
375
+ autoMerge: false
376
+ },
377
+ 'self-modification': {
378
+ prefix: 'self-mod/',
379
+ baseFrom: 'main',
380
+ protectByDefault: true,
381
+ autoMerge: false,
382
+ requireApproval: true
383
+ }
384
+ };
385
+
386
+ return strategies[modificationType] || strategies.enhancement;
387
+ }
388
+ }
389
+
390
390
  module.exports = BranchManager;