vibecodingmachine-cli 2026.1.24-1641 → 2026.1.29-1432

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 (41) hide show
  1. package/package.json +12 -4
  2. package/.allnightai/REQUIREMENTS.md +0 -11
  3. package/.eslintrc.js +0 -16
  4. package/__tests__/antigravity-js-handler.test.js +0 -23
  5. package/__tests__/provider-manager.test.js +0 -84
  6. package/__tests__/provider-rate-cache.test.js +0 -27
  7. package/jest.config.js +0 -8
  8. package/logs/audit/2025-11-07.jsonl +0 -2
  9. package/logs/audit/2025-12-24.jsonl +0 -2
  10. package/logs/audit/2025-12-27.jsonl +0 -1
  11. package/logs/audit/2026-01-03.jsonl +0 -2
  12. package/logs/audit/2026-01-23.jsonl +0 -2
  13. package/repro_open.js +0 -13
  14. package/reproduce_issue.js +0 -160
  15. package/reset_provider_order.js +0 -21
  16. package/scripts/README.md +0 -128
  17. package/scripts/auto-start-wrapper.sh +0 -92
  18. package/scripts/convert-requirements.js +0 -35
  19. package/scripts/debug-parse.js +0 -24
  20. package/src/commands/auto.js.bak +0 -710
  21. package/src/utils/auto-mode-ui.js.bak.blessed +0 -207
  22. package/tests/antigravity-js-handler.test.js +0 -23
  23. package/tests/auto-mode.test.js +0 -37
  24. package/tests/config.test.js +0 -34
  25. package/tests/home-bootstrap.test.js +0 -76
  26. package/tests/integration/health-tracking.integration.test.js +0 -284
  27. package/tests/provider-manager.test.js +0 -92
  28. package/tests/rate-limit-display.test.js +0 -44
  29. package/tests/requirements-bullet-parsing.test.js +0 -15
  30. package/tests/requirements-converter.test.js +0 -42
  31. package/tests/requirements-heading-count.test.js +0 -27
  32. package/tests/requirements-legacy-parsing.test.js +0 -15
  33. package/tests/requirements-navigator-buildtree-await.test.js +0 -28
  34. package/tests/requirements-parse-integration.test.js +0 -44
  35. package/tests/wait-for-ide-completion.test.js +0 -56
  36. package/tests/wait-for-ide-quota-detection-cursor-screenshot.test.js +0 -61
  37. package/tests/wait-for-ide-quota-detection-cursor.test.js +0 -60
  38. package/tests/wait-for-ide-quota-detection-negative.test.js +0 -45
  39. package/tests/wait-for-ide-quota-detection.test.js +0 -59
  40. package/verify_fix.js +0 -36
  41. package/verify_ui.js +0 -38
package/package.json CHANGED
@@ -1,12 +1,20 @@
1
1
  {
2
2
  "name": "vibecodingmachine-cli",
3
- "version": "2026.01.24-1641",
3
+ "version": "2026.01.29-1432",
4
4
  "description": "Command-line interface for Vibe Coding Machine - Autonomous development",
5
5
  "main": "src/index.js",
6
6
  "bin": {
7
- "vibecodingmachine": "./bin/vibecodingmachine.js",
8
- "vcm": "./bin/vibecodingmachine.js"
7
+ "vibecodingmachine": "./bin/vibecodingmachine.js"
9
8
  },
9
+ "files": [
10
+ "bin/**/*.js",
11
+ "src/**/*.js",
12
+ "src/**/*.json",
13
+ "scripts/postinstall.js",
14
+ "!src/**/*.test.js",
15
+ "!src/**/__tests__/**",
16
+ "!src/**/tests/**"
17
+ ],
10
18
  "scripts": {
11
19
  "postinstall": "node scripts/postinstall.js",
12
20
  "test": "jest",
@@ -44,7 +52,7 @@
44
52
  "react": "^19.2.0",
45
53
  "screenshot-desktop": "^1.15.3",
46
54
  "table": "^6.8.1",
47
- "vibecodingmachine-core": "^2026.01.24-1641"
55
+ "vibecodingmachine-core": "^2026.01.29-1432"
48
56
  },
49
57
  "devDependencies": {
50
58
  "eslint": "^8.57.0",
@@ -1,11 +0,0 @@
1
-
2
-
3
- ## ✅ Requirements completed
4
- - TEST
5
- - Add Cline CLI IDE
6
-
7
- ### Add Cline CLI to list of IDEs
8
-
9
- ### NEW REQUIREMENT
10
-
11
- ### Add Cline CLI to list of IDEs
package/.eslintrc.js DELETED
@@ -1,16 +0,0 @@
1
- module.exports = {
2
- env: {
3
- node: true,
4
- es2021: true,
5
- jest: true
6
- },
7
- extends: 'eslint:recommended',
8
- parserOptions: {
9
- ecmaVersion: 2021,
10
- sourceType: 'module'
11
- },
12
- rules: {
13
- 'no-unused-vars': ['error', { argsIgnorePattern: '^_' }],
14
- 'no-console': 'off'
15
- }
16
- };
@@ -1,23 +0,0 @@
1
- const { handleAntigravityRateLimit } = require('../src/utils/antigravity-js-handler');
2
- const providerRegistry = require('../src/utils/provider-registry');
3
-
4
- jest.mock('../src/utils/provider-registry');
5
-
6
- describe('handleAntigravityRateLimit', () => {
7
- beforeEach(() => {
8
- jest.resetAllMocks();
9
- });
10
-
11
- test('suggests next provider and does not persistently disable antigravity', async () => {
12
- providerRegistry.getProviderPreferences.mockResolvedValue({
13
- order: ['antigravity', 'vscode'],
14
- enabled: { antigravity: true, vscode: true }
15
- });
16
-
17
- const result = await handleAntigravityRateLimit();
18
-
19
- expect(result.success).toBe(true);
20
- expect(result.nextProvider).toBe('vscode');
21
- expect(providerRegistry.saveProviderPreferences).not.toHaveBeenCalled();
22
- });
23
- });
@@ -1,84 +0,0 @@
1
- const providerRegistry = require('../src/utils/provider-registry');
2
- const interactive = require('../src/utils/interactive');
3
-
4
- jest.mock('../src/utils/provider-registry');
5
-
6
- describe('showProviderManagerMenu', () => {
7
- const origIsTTY = process.stdin.isTTY;
8
- const origSetRawMode = process.stdin.setRawMode;
9
-
10
- beforeAll(() => {
11
- // Ensure stdin behaves like a TTY for the menu
12
- process.stdin.isTTY = true;
13
- process.stdin.setRawMode = () => {};
14
- });
15
-
16
- afterAll(() => {
17
- process.stdin.isTTY = origIsTTY;
18
- process.stdin.setRawMode = origSetRawMode;
19
- });
20
-
21
- beforeEach(() => {
22
- jest.resetAllMocks();
23
- });
24
-
25
- test('pressing left after reordering calls saveProviderPreferences', async () => {
26
- providerRegistry.getProviderDefinitions.mockReturnValue([
27
- { id: 'groq', name: 'Groq' },
28
- { id: 'antigravity', name: 'Antigravity' }
29
- ]);
30
-
31
- providerRegistry.getProviderPreferences.mockResolvedValue({
32
- order: ['groq', 'antigravity'],
33
- enabled: { groq: true, antigravity: true }
34
- });
35
-
36
- providerRegistry.saveProviderPreferences.mockResolvedValue();
37
-
38
- // Start the menu
39
- const menuPromise = interactive.showProviderManagerMenu();
40
-
41
- // Allow the menu to initialize
42
- await new Promise(resolve => setImmediate(resolve));
43
-
44
- // Simulate 'j' (reorder downward)
45
- process.stdin.emit('keypress', 'j', { name: 'j' });
46
- await new Promise(resolve => setImmediate(resolve));
47
-
48
- // Simulate left arrow to save and exit
49
- process.stdin.emit('keypress', undefined, { name: 'left' });
50
-
51
- await menuPromise; // wait for menu to finish
52
-
53
- expect(providerRegistry.saveProviderPreferences).toHaveBeenCalledTimes(1);
54
- expect(providerRegistry.saveProviderPreferences).toHaveBeenCalledWith(['antigravity', 'groq'], { groq: true, antigravity: true });
55
- });
56
-
57
- test('pressing escape after reordering does NOT call saveProviderPreferences', async () => {
58
- providerRegistry.getProviderDefinitions.mockReturnValue([
59
- { id: 'groq', name: 'Groq' },
60
- { id: 'antigravity', name: 'Antigravity' }
61
- ]);
62
-
63
- providerRegistry.getProviderPreferences.mockResolvedValue({
64
- order: ['groq', 'antigravity'],
65
- enabled: { groq: true, antigravity: true }
66
- });
67
-
68
- providerRegistry.saveProviderPreferences.mockResolvedValue();
69
-
70
- const menuPromise = interactive.showProviderManagerMenu();
71
- await new Promise(resolve => setImmediate(resolve));
72
-
73
- // Make a change
74
- process.stdin.emit('keypress', 'j', { name: 'j' });
75
- await new Promise(resolve => setImmediate(resolve));
76
-
77
- // Press escape to cancel (should not persist)
78
- process.stdin.emit('keypress', undefined, { name: 'escape' });
79
-
80
- await menuPromise;
81
-
82
- expect(providerRegistry.saveProviderPreferences).not.toHaveBeenCalled();
83
- });
84
- });
@@ -1,27 +0,0 @@
1
- const { getProviderRateLimitedQuotas } = require('../src/utils/provider-rate-cache');
2
- const ProviderManager = require('vibecodingmachine-core/src/ide-integration/provider-manager.cjs');
3
-
4
- describe('getProviderRateLimitedQuotas', () => {
5
- let pm;
6
-
7
- beforeEach(() => {
8
- pm = new ProviderManager();
9
- pm.clearAllRateLimits();
10
- });
11
-
12
- afterEach(() => {
13
- pm.clearAllRateLimits();
14
- });
15
-
16
- test('returns rate-limited entry when ProviderManager has a limit', () => {
17
- pm.markRateLimited('antigravity', undefined, 'Quota limit reached');
18
-
19
- const defs = [{ id: 'antigravity' }, { id: 'vscode' }];
20
- const map = getProviderRateLimitedQuotas(defs);
21
-
22
- expect(map.has('antigravity')).toBe(true);
23
- const q = map.get('antigravity');
24
- expect(q).toHaveProperty('type', 'rate-limit');
25
- expect(q).toHaveProperty('resetsAt');
26
- });
27
- });
package/jest.config.js DELETED
@@ -1,8 +0,0 @@
1
- module.exports = {
2
- testEnvironment: 'node',
3
- testMatch: ['**/tests/**/*.test.js'],
4
- testPathIgnorePatterns: ['/node_modules/'],
5
- collectCoverageFrom: ['src/**/*.js'],
6
- coverageDirectory: 'coverage',
7
- verbose: true
8
- };
@@ -1,2 +0,0 @@
1
- {"timestamp":"2025-11-07T12:33:10.966Z","type":"auto-mode-stop","reason":"startup","message":"Auto Mode stopped (startup)"}
2
- {"timestamp":"2025-11-07T12:33:47.590Z","type":"auto-mode-stop","reason":"startup","message":"Auto Mode stopped (startup)"}
@@ -1,2 +0,0 @@
1
- {"timestamp":"2025-12-25T00:38:27.990Z","type":"auto-mode-stop","reason":"startup","message":"Auto Mode stopped (startup)"}
2
- {"timestamp":"2025-12-25T00:39:00.886Z","type":"auto-mode-stop","reason":"startup","message":"Auto Mode stopped (startup)"}
@@ -1 +0,0 @@
1
- {"timestamp":"2025-12-28T01:17:10.393Z","type":"auto-mode-stop","reason":"startup","message":"Auto Mode stopped (startup)"}
@@ -1,2 +0,0 @@
1
- {"timestamp":"2026-01-04T04:29:09.497Z","type":"auto-mode-stop","reason":"startup","message":"Auto Mode stopped (startup)"}
2
- {"timestamp":"2026-01-04T04:46:49.126Z","type":"auto-mode-stop","reason":"startup","message":"Auto Mode stopped (startup)"}
@@ -1,2 +0,0 @@
1
- {"timestamp":"2026-01-24T03:42:12.706Z","type":"auto-mode-stop","reason":"startup","message":"Auto Mode stopped (startup)"}
2
- {"timestamp":"2026-01-24T03:42:41.636Z","type":"auto-mode-stop","reason":"startup","message":"Auto Mode stopped (startup)"}
package/repro_open.js DELETED
@@ -1,13 +0,0 @@
1
- try {
2
- const open = require('open');
3
- console.log('Type of open:', typeof open);
4
- console.log('open:', open);
5
- if (typeof open !== 'function') {
6
- console.log('Exports:', Object.keys(open));
7
- if (open.default) {
8
- console.log('Type of open.default:', typeof open.default);
9
- }
10
- }
11
- } catch (e) {
12
- console.error('Require failed:', e.message);
13
- }
@@ -1,160 +0,0 @@
1
- const fs = require('fs-extra');
2
- const path = require('path');
3
- const { getRequirementsPath } = require('vibecodingmachine-core');
4
- const { getRepoPath } = require('./src/utils/config');
5
- const requirements = require('./src/commands/requirements');
6
-
7
- // Mock specific functions we need from interactive.js's logic
8
- // helping function to move requirement to recycled (deletion logic)
9
- async function moveRequirementToRecycled(reqPath, requirementTitle, fromSection) {
10
- const content = await fs.readFile(reqPath, 'utf8');
11
- const lines = content.split('\n');
12
-
13
- let requirementStartIndex = -1;
14
- let requirementEndIndex = -1;
15
-
16
- for (let i = 0; i < lines.length; i++) {
17
- const line = lines[i].trim();
18
- if (line.startsWith('###')) {
19
- const title = line.replace(/^###\s*/, '').trim();
20
- // Logic from interactive.js
21
- if (title && title.includes(requirementTitle)) {
22
- requirementStartIndex = i;
23
- for (let j = i + 1; j < lines.length; j++) {
24
- const nextLine = lines[j].trim();
25
- if (nextLine.startsWith('###') || (nextLine.startsWith('##') && !nextLine.startsWith('###'))) {
26
- requirementEndIndex = j;
27
- break;
28
- }
29
- }
30
- if (requirementEndIndex === -1) {
31
- requirementEndIndex = lines.length;
32
- }
33
- break;
34
- }
35
- }
36
- }
37
-
38
- if (requirementStartIndex === -1) {
39
- console.log('⚠️ Could not find requirement to recycle');
40
- return false;
41
- }
42
-
43
- console.log(`Found requirement at lines ${requirementStartIndex}-${requirementEndIndex}`);
44
-
45
- const requirementBlock = lines.slice(requirementStartIndex, requirementEndIndex);
46
- lines.splice(requirementStartIndex, requirementEndIndex - requirementStartIndex);
47
-
48
- if (requirementStartIndex < lines.length) {
49
- const nextLine = lines[requirementStartIndex]?.trim();
50
- const packageNames = ['cli', 'core', 'electron-app', 'web', 'mobile', 'vscode-extension', 'sync-server'];
51
- if (nextLine && packageNames.includes(nextLine.toLowerCase()) &&
52
- !nextLine.startsWith('###') && !nextLine.startsWith('PACKAGE:')) {
53
- lines.splice(requirementStartIndex, 1);
54
- }
55
- while (requirementStartIndex < lines.length && lines[requirementStartIndex]?.trim() === '') {
56
- lines.splice(requirementStartIndex, 1);
57
- }
58
- }
59
-
60
- let recycledIndex = -1;
61
- for (let i = 0; i < lines.length; i++) {
62
- if (lines[i].includes('♻️ Recycled') || lines[i].includes('🗑️ Recycled')) {
63
- recycledIndex = i;
64
- break;
65
- }
66
- }
67
-
68
- if (recycledIndex === -1) {
69
- let lastSectionIndex = -1;
70
- for (let i = lines.length - 1; i >= 0; i--) {
71
- if (lines[i].startsWith('##') && !lines[i].startsWith('###')) {
72
- lastSectionIndex = i;
73
- break;
74
- }
75
- }
76
- const insertIndex = lastSectionIndex > 0 ? lastSectionIndex : lines.length;
77
- lines.splice(insertIndex, 0, '', '## ♻️ Recycled', '');
78
- recycledIndex = insertIndex + 1;
79
- }
80
-
81
- let insertIndex = recycledIndex + 1;
82
- while (insertIndex < lines.length && lines[insertIndex].trim() === '') {
83
- insertIndex++;
84
- }
85
- lines.splice(insertIndex, 0, ...requirementBlock);
86
-
87
- if (insertIndex + requirementBlock.length < lines.length && lines[insertIndex + requirementBlock.length].trim() !== '') {
88
- lines.splice(insertIndex + requirementBlock.length, 0, '');
89
- }
90
-
91
- await fs.writeFile(reqPath, lines.join('\n'));
92
- return true;
93
- }
94
-
95
- async function run() {
96
- try {
97
- const repoPath = '/Users/jesse/code/mediawink/vibecodingmachine'; // Hardcoded valid repo path
98
- console.log('Repo path:', repoPath);
99
-
100
- const reqPath = await getRequirementsPath(repoPath);
101
- console.log('<<< PATH >>>', reqPath);
102
-
103
- if (await fs.pathExists(reqPath)) {
104
- const content = await fs.readFile(reqPath, 'utf8');
105
- console.log('Current content length:', content.length);
106
- console.log('Current content PRE-TEST:\n', content);
107
- } else {
108
- console.log('Requirements file does not exist');
109
- }
110
-
111
- console.log('\n--- Adding TESTREQ1 ---');
112
- await requirements.add('TESTREQ1', 'all', 'Description 1');
113
-
114
- console.log('\n--- Adding TESTREQ2 ---');
115
- await requirements.add('TESTREQ2', 'all', 'Description 2');
116
-
117
- let content = await fs.readFile(reqPath, 'utf8');
118
- console.log('Content after adding:\n', content);
119
-
120
- // Verify order
121
- // We expect TESTREQ2 to be above TESTREQ1 if it inserts at the top of the section
122
- const lines = content.split('\n');
123
- let idx1 = -1, idx2 = -1;
124
- for (let i = 0; i < lines.length; i++) {
125
- if (lines[i].includes('### TESTREQ1')) idx1 = i;
126
- if (lines[i].includes('### TESTREQ2')) idx2 = i;
127
- }
128
- console.log(`TESTREQ1 line: ${idx1}, TESTREQ2 line: ${idx2}`);
129
- if (idx2 < idx1 && idx2 > -1) {
130
- console.log('SUCCESS: TESTREQ2 is above TESTREQ1 (Correct LIFO/Stack behavior for "Top of list")');
131
- } else {
132
- console.log('FAIL: Order is not correct for "Top of list" insertion');
133
- }
134
-
135
- console.log('\n--- Deleting TESTREQ1 ---');
136
- const success = await moveRequirementToRecycled(reqPath, 'TESTREQ1', 'todo');
137
- console.log('Delete success:', success);
138
-
139
- content = await fs.readFile(reqPath, 'utf8');
140
- console.log('Content after delete:\n', content);
141
-
142
- if (content.includes('### TESTREQ1')) {
143
- // It should be in recycled section
144
- const recycledIndex = content.indexOf('## ♻️ Recycled');
145
- const reqIndex = content.indexOf('### TESTREQ1');
146
- if (reqIndex > recycledIndex) {
147
- console.log('SUCCESS: TESTREQ1 moved to Recycled');
148
- } else {
149
- console.log('FAIL: TESTREQ1 is still in TODO or wrong place');
150
- }
151
- } else {
152
- console.log('FAIL: TESTREQ1 disappeared completely (should be recycled)');
153
- }
154
-
155
- } catch (error) {
156
- console.error('Error:', error);
157
- }
158
- }
159
-
160
- run();
@@ -1,21 +0,0 @@
1
- const { getDefaultProviderOrder, saveProviderPreferences, getProviderPreferences } = require('./src/utils/provider-registry');
2
- const chalk = require('chalk');
3
-
4
- async function resetOrder() {
5
- try {
6
- const defaultOrder = getDefaultProviderOrder();
7
- console.log('New Default Order:', defaultOrder);
8
-
9
- const currentPrefs = await getProviderPreferences();
10
- console.log('Current User Order:', currentPrefs.order);
11
-
12
- // Force update the order to match default
13
- await saveProviderPreferences(defaultOrder, currentPrefs.enabled);
14
- console.log(chalk.green('Successfully reset provider order to default (Cloud -> IDE -> Local).'));
15
- } catch (error) {
16
- console.error('Failed to reset order:', error);
17
- process.exit(1);
18
- }
19
- }
20
-
21
- resetOrder();
package/scripts/README.md DELETED
@@ -1,128 +0,0 @@
1
- # Auto Start Wrapper Script
2
-
3
- ## Overview
4
-
5
- The `auto-start-wrapper.sh` script provides **reliable Ctrl+C handling** for `vcm auto:start`.
6
-
7
- ## Problem
8
-
9
- Due to a Node.js limitation, SIGINT (Ctrl+C) signals don't reliably reach the event loop when Node.js is blocked waiting for child processes (like Aider). **This means Ctrl+C does NOT work when running `vcm auto:start` directly.**
10
-
11
- ## Recommended Usage
12
-
13
- **If you want Ctrl+C to work**, use the wrapper script instead of running `vcm auto:start` directly:
14
-
15
- ```bash
16
- # From anywhere (recommended)
17
- ~/.asdf/installs/nodejs/20.19.5/lib/node_modules/@vibecodingmachine/cli/scripts/auto-start-wrapper.sh
18
-
19
- # Or if you know the package location
20
- /path/to/vibecodingmachine/packages/cli/scripts/auto-start-wrapper.sh
21
- ```
22
-
23
- ### Creating a Convenient Alias
24
-
25
- Add this to your `~/.zshrc` or `~/.bashrc`:
26
-
27
- ```bash
28
- # Alias for vcm auto:start with Ctrl+C support
29
- alias vcm-start='~/.asdf/installs/nodejs/20.19.5/lib/node_modules/@vibecodingmachine/cli/scripts/auto-start-wrapper.sh'
30
- ```
31
-
32
- Then you can simply run:
33
- ```bash
34
- vcm-start
35
- vcm-start --max-chats 10
36
- ```
37
-
38
- ## Solution
39
-
40
- This wrapper script:
41
- 1. Runs `vcm auto:start` in the background
42
- 2. Monitors keyboard input in the foreground (in the wrapper's process)
43
- 3. When Ctrl+C, Esc, or 'x' is pressed, creates a `.stop` file
44
- 4. The running `vcm` process detects the file (via watchdog timer) and exits gracefully
45
-
46
- ## Usage
47
-
48
- ### Direct Usage
49
- ```bash
50
- # From the project root
51
- ./packages/cli/scripts/auto-start-wrapper.sh
52
-
53
- # Or with options
54
- ./packages/cli/scripts/auto-start-wrapper.sh --max-chats 5
55
- ```
56
-
57
- ### How to Stop
58
- - **Ctrl+C**: Stops the process (most common)
59
- - **Esc key**: Stops the process
60
- - **'x' key**: Stops the process
61
- - **`vcm auto:stop`**: From another terminal
62
-
63
- ## How It Works
64
-
65
- 1. **Wrapper starts**: Spawns `node vcm auto:start` as a background process
66
- 2. **Wrapper monitors**: Uses `read -t 0.5 -n 1` to check for key presses every 500ms
67
- 3. **Key pressed**: Creates `~/.config/vibecodingmachine/.stop` file
68
- 4. **Watchdog detects**: The running vcm process has a watchdog that checks for this file every 500ms
69
- 5. **Graceful exit**: When detected, vcm kills Aider processes and exits the main loop
70
- 6. **Cleanup**: Wrapper removes the stop file and exits
71
-
72
- ## Alternative: Manual Stop File
73
-
74
- You can also create the stop file manually:
75
- ```bash
76
- # Create stop file
77
- touch ~/.config/vibecodingmachine/.stop
78
-
79
- # Or use the stop command
80
- vcm auto:stop
81
- ```
82
-
83
- ## Technical Details
84
-
85
- ### Watchdog Timer
86
- The watchdog runs every 500ms in the Node.js process:
87
- ```javascript
88
- const watchdog = setInterval(async () => {
89
- if (await fs.pathExists(stopFilePath)) {
90
- console.log('🛑 Stop signal detected (via .stop file)');
91
- shouldExit = true;
92
- exitReason = 'user-stop';
93
- await fs.unlink(stopFilePath);
94
- }
95
-
96
- if (shouldExit) {
97
- clearInterval(watchdog);
98
- aiderManager.killAllProcesses();
99
- // Loop breaks naturally
100
- }
101
- }, 500);
102
- ```
103
-
104
- ### Why Not Direct SIGINT?
105
- Node.js SIGINT handlers aren't called when:
106
- - `await` is blocked on a child process
107
- - The event loop is not running (waiting for I/O)
108
- - Child processes intercept signals first
109
-
110
- The wrapper script solves this by handling signals at the shell level (outside Node.js).
111
-
112
- ## Testing
113
-
114
- Test the wrapper with Ctrl+C:
115
- ```bash
116
- ./packages/cli/scripts/auto-start-wrapper.sh &
117
- sleep 5
118
- # Press Ctrl+C
119
- # Should see: "🛑 Stop signal detected" and process exits
120
- ```
121
-
122
- Test with stop command:
123
- ```bash
124
- ./packages/cli/scripts/auto-start-wrapper.sh &
125
- sleep 5
126
- vcm auto:stop
127
- # Should see process exit within 2 seconds
128
- ```
@@ -1,92 +0,0 @@
1
- #!/bin/bash
2
- #
3
- # Wrapper script for 'vcm auto:start' that properly handles Ctrl+C and other key presses
4
- # This script runs vcm auto:start in the background and monitors keyboard input
5
- #
6
-
7
- set -e
8
-
9
- # Colors
10
- RED='\033[0;31m'
11
- GREEN='\033[0;32m'
12
- YELLOW='\033[1;33m'
13
- GRAY='\033[0;37m'
14
- NC='\033[0m' # No Color
15
-
16
- # Get the directory where this script is located
17
- SCRIPT_DIR="$(cd "$(dirname "${BASH_SOURCE[0]}")" && pwd)"
18
- CLI_DIR="$(dirname "$SCRIPT_DIR")"
19
-
20
- # Path to vcm command
21
- ANA_CMD="$CLI_DIR/bin/vibecodingmachine.js"
22
-
23
- # Path to stop file
24
- STOP_FILE="$HOME/.config/vibecodingmachine/.stop"
25
-
26
- # Cleanup function
27
- cleanup() {
28
- echo -e "\n${YELLOW}Stopping auto mode...${NC}"
29
-
30
- # Create stop file to signal the process
31
- mkdir -p "$(dirname "$STOP_FILE")"
32
- echo "Stop requested at $(date)" > "$STOP_FILE"
33
-
34
- # Send SIGTERM to the vcm process if it's running
35
- if [ -n "$ANA_PID" ] && kill -0 "$ANA_PID" 2>/dev/null; then
36
- kill -TERM "$ANA_PID" 2>/dev/null || true
37
- fi
38
-
39
- # Wait a moment for the process to exit gracefully
40
- sleep 2
41
-
42
- # Force kill if still running
43
- if [ -n "$ANA_PID" ] && kill -0 "$ANA_PID" 2>/dev/null; then
44
- echo -e "${YELLOW}Force stopping...${NC}"
45
- kill -9 "$ANA_PID" 2>/dev/null || true
46
- fi
47
-
48
- # Also kill any remaining aider processes
49
- pkill -9 -f "aider.*vibecodingmachine" 2>/dev/null || true
50
-
51
- echo -e "${GREEN}Auto mode stopped${NC}"
52
- exit 0
53
- }
54
-
55
- # Set up trap for Ctrl+C
56
- trap cleanup INT TERM
57
-
58
- # Start vcm auto:start in background
59
- echo -e "${GRAY}Starting auto mode...${NC}"
60
- echo -e "${GRAY}Press Ctrl+C, Esc, or 'x' to stop${NC}"
61
- echo ""
62
-
63
- # Start vcm in background, saving its PID
64
- # Set ANA_WRAPPER_RUNNING to prevent infinite recursion
65
- export ANA_WRAPPER_RUNNING=1
66
- node "$ANA_CMD" auto:start "$@" &
67
- ANA_PID=$!
68
-
69
- # Monitor for keyboard input AND stop file in the foreground
70
- # Use a non-blocking read to check for key presses
71
- while kill -0 "$ANA_PID" 2>/dev/null; do
72
- # Check for stop file first (created by 'vcm auto:stop' or Ctrl+C signal)
73
- if [ -f "$STOP_FILE" ]; then
74
- echo -e "\n${YELLOW}Stop signal detected${NC}"
75
- cleanup
76
- fi
77
-
78
- # Read a single character with 0.5 second timeout
79
- if read -t 0.5 -n 1 -s key 2>/dev/null; then
80
- # Check if key is 'x', Esc (ASCII 27), or Ctrl+C
81
- if [ "$key" = "x" ] || [ "$key" = "X" ] || [ "$(printf '%d' "'$key")" = "27" ]; then
82
- echo -e "\n${YELLOW}Stop key pressed${NC}"
83
- cleanup
84
- fi
85
- fi
86
- done
87
-
88
- # Process exited naturally
89
- echo -e "\n${GREEN}Auto mode completed${NC}"
90
- wait "$ANA_PID"
91
- exit_code=$?
92
- exit $exit_code