vibecodingmachine-core 1.0.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.
- package/.babelrc +13 -0
- package/README.md +28 -0
- package/__tests__/applescript-manager-claude-fix.test.js +286 -0
- package/__tests__/requirement-2-auto-start-looping.test.js +69 -0
- package/__tests__/requirement-3-auto-start-looping.test.js +69 -0
- package/__tests__/requirement-4-auto-start-looping.test.js +69 -0
- package/__tests__/requirement-6-auto-start-looping.test.js +73 -0
- package/__tests__/requirement-7-status-tracking.test.js +332 -0
- package/jest.config.js +18 -0
- package/jest.setup.js +12 -0
- package/package.json +46 -0
- package/src/auth/access-denied.html +119 -0
- package/src/auth/shared-auth-storage.js +230 -0
- package/src/autonomous-mode/feature-implementer.cjs +70 -0
- package/src/autonomous-mode/feature-implementer.js +425 -0
- package/src/chat-management/chat-manager.cjs +71 -0
- package/src/chat-management/chat-manager.js +342 -0
- package/src/ide-integration/__tests__/applescript-manager-thread-closure.test.js +227 -0
- package/src/ide-integration/aider-cli-manager.cjs +850 -0
- package/src/ide-integration/applescript-diagnostics.js +0 -0
- package/src/ide-integration/applescript-manager.cjs +1088 -0
- package/src/ide-integration/applescript-manager.js +2803 -0
- package/src/ide-integration/applescript-open-apps.js +0 -0
- package/src/ide-integration/applescript-read-response.js +0 -0
- package/src/ide-integration/applescript-send-text.js +0 -0
- package/src/ide-integration/applescript-thread-closure.js +0 -0
- package/src/ide-integration/applescript-utils.js +306 -0
- package/src/ide-integration/cdp-manager.cjs +221 -0
- package/src/ide-integration/cdp-manager.js +321 -0
- package/src/ide-integration/claude-code-cli-manager.cjs +301 -0
- package/src/ide-integration/cline-cli-manager.cjs +2252 -0
- package/src/ide-integration/continue-cli-manager.js +431 -0
- package/src/ide-integration/provider-manager.cjs +354 -0
- package/src/ide-integration/quota-detector.cjs +34 -0
- package/src/ide-integration/quota-detector.js +349 -0
- package/src/ide-integration/windows-automation-manager.js +262 -0
- package/src/index.cjs +43 -0
- package/src/index.js +17 -0
- package/src/llm/direct-llm-manager.cjs +609 -0
- package/src/ui/ButtonComponents.js +247 -0
- package/src/ui/ChatInterface.js +499 -0
- package/src/ui/StateManager.js +259 -0
- package/src/ui/StateManager.test.js +0 -0
- package/src/utils/audit-logger.cjs +116 -0
- package/src/utils/config-helpers.cjs +94 -0
- package/src/utils/config-helpers.js +94 -0
- package/src/utils/electron-update-checker.js +78 -0
- package/src/utils/gcloud-auth.cjs +394 -0
- package/src/utils/logger.cjs +193 -0
- package/src/utils/logger.js +191 -0
- package/src/utils/repo-helpers.cjs +120 -0
- package/src/utils/repo-helpers.js +120 -0
- package/src/utils/requirement-helpers.js +432 -0
- package/src/utils/update-checker.js +167 -0
|
@@ -0,0 +1,342 @@
|
|
|
1
|
+
// @vibecodingmachine/core - Chat Manager
|
|
2
|
+
// Handles chat message management, polling, and response detection
|
|
3
|
+
|
|
4
|
+
/**
|
|
5
|
+
* Chat Manager for handling chat interactions
|
|
6
|
+
* Manages message sending, response polling, and state management
|
|
7
|
+
*/
|
|
8
|
+
export class ChatManager {
|
|
9
|
+
constructor(options = {}) {
|
|
10
|
+
this.logger = options.logger || console;
|
|
11
|
+
this.electronAPI = options.electronAPI || null;
|
|
12
|
+
this.onMessageUpdate = options.onMessageUpdate || (() => {});
|
|
13
|
+
this.onStatusUpdate = options.onStatusUpdate || (() => {});
|
|
14
|
+
this.onProgressUpdate = options.onProgressUpdate || (() => {});
|
|
15
|
+
|
|
16
|
+
// State
|
|
17
|
+
this.isPolling = false;
|
|
18
|
+
this.isPaused = false;
|
|
19
|
+
this.isStopped = false;
|
|
20
|
+
this.responseWaiting = false;
|
|
21
|
+
this.currentProgress = 0;
|
|
22
|
+
this.pollingTimer = null;
|
|
23
|
+
this.lastChangeAt = 0;
|
|
24
|
+
}
|
|
25
|
+
|
|
26
|
+
/**
|
|
27
|
+
* Send a chat message to an IDE
|
|
28
|
+
* @param {string} message - The message to send
|
|
29
|
+
* @param {string} ide - The IDE to send to
|
|
30
|
+
* @param {string} tabId - The tab ID for tracking
|
|
31
|
+
* @returns {Promise<Object>} Result of the send operation
|
|
32
|
+
*/
|
|
33
|
+
async sendMessage(message, ide, tabId) {
|
|
34
|
+
this.logger.log('sendMessage called with:', { message: message.substring(0, 50) + '...', ide, tabId });
|
|
35
|
+
|
|
36
|
+
if (!message || !message.trim() || this.isPaused || this.isStopped) {
|
|
37
|
+
this.logger.log('sendMessage early return - message:', !!message, 'isPaused:', this.isPaused, 'isStopped:', this.isStopped);
|
|
38
|
+
return {
|
|
39
|
+
success: false,
|
|
40
|
+
error: 'Cannot send message - invalid state or empty message'
|
|
41
|
+
};
|
|
42
|
+
}
|
|
43
|
+
|
|
44
|
+
try {
|
|
45
|
+
this.onStatusUpdate(tabId, 'Sending message...');
|
|
46
|
+
|
|
47
|
+
let result;
|
|
48
|
+
if (this.electronAPI && this.electronAPI.sendChat) {
|
|
49
|
+
result = await this.electronAPI.sendChat(message, ide);
|
|
50
|
+
} else {
|
|
51
|
+
// Fallback for non-Electron environments
|
|
52
|
+
result = { success: true, method: 'simulated', message: 'Simulated message sent' };
|
|
53
|
+
}
|
|
54
|
+
|
|
55
|
+
if (result && result.success) {
|
|
56
|
+
this.logger.log(`Message sent successfully via ${result.method || 'chat'}`);
|
|
57
|
+
this.onStatusUpdate(tabId, `✅ Message sent successfully via ${result.method || 'chat'}`);
|
|
58
|
+
|
|
59
|
+
if (result.method !== 'terminal') {
|
|
60
|
+
this.startPolling(ide, tabId);
|
|
61
|
+
}
|
|
62
|
+
|
|
63
|
+
return {
|
|
64
|
+
success: true,
|
|
65
|
+
method: result.method,
|
|
66
|
+
message: result.message
|
|
67
|
+
};
|
|
68
|
+
} else {
|
|
69
|
+
this.logger.log('Failed to send message:', result);
|
|
70
|
+
this.onStatusUpdate(tabId, `❌ Failed to send message: ${result?.error || 'Unknown error'}`);
|
|
71
|
+
|
|
72
|
+
return {
|
|
73
|
+
success: false,
|
|
74
|
+
error: result?.error || 'Unknown error'
|
|
75
|
+
};
|
|
76
|
+
}
|
|
77
|
+
} catch (error) {
|
|
78
|
+
this.logger.error('Error sending message:', error);
|
|
79
|
+
this.onStatusUpdate(tabId, `❌ Error sending message: ${error.message}`);
|
|
80
|
+
|
|
81
|
+
return {
|
|
82
|
+
success: false,
|
|
83
|
+
error: error.message
|
|
84
|
+
};
|
|
85
|
+
}
|
|
86
|
+
}
|
|
87
|
+
|
|
88
|
+
/**
|
|
89
|
+
* Start polling for responses from an IDE
|
|
90
|
+
* @param {string} ide - The IDE to poll
|
|
91
|
+
* @param {string} tabId - The tab ID for tracking
|
|
92
|
+
*/
|
|
93
|
+
startPolling(ide, tabId) {
|
|
94
|
+
if (this.isPolling) {
|
|
95
|
+
this.logger.log('Polling already in progress');
|
|
96
|
+
return;
|
|
97
|
+
}
|
|
98
|
+
|
|
99
|
+
this.isPolling = true;
|
|
100
|
+
this.lastChangeAt = Date.now();
|
|
101
|
+
|
|
102
|
+
this.logger.log(`Starting polling for ${ide} (tab: ${tabId})`);
|
|
103
|
+
this.onStatusUpdate(tabId, 'Polling for response...');
|
|
104
|
+
|
|
105
|
+
this.pollingTimer = setInterval(async () => {
|
|
106
|
+
if (this.isStopped || this.isPaused) {
|
|
107
|
+
this.stopPolling();
|
|
108
|
+
return;
|
|
109
|
+
}
|
|
110
|
+
|
|
111
|
+
try {
|
|
112
|
+
const response = await this.readResponse(ide);
|
|
113
|
+
|
|
114
|
+
if (response && response.length > 0) {
|
|
115
|
+
this.lastChangeAt = Date.now();
|
|
116
|
+
|
|
117
|
+
// Check if response is complete
|
|
118
|
+
const completionResult = await this.detectResponseCompletion(ide, response);
|
|
119
|
+
|
|
120
|
+
if (completionResult.completed) {
|
|
121
|
+
this.logger.log('Response completed, stopping polling');
|
|
122
|
+
this.onStatusUpdate(tabId, '✅ Response received');
|
|
123
|
+
this.stopPolling();
|
|
124
|
+
|
|
125
|
+
// Update messages with the complete response
|
|
126
|
+
this.onMessageUpdate(tabId, {
|
|
127
|
+
role: 'assistant',
|
|
128
|
+
text: completionResult.response,
|
|
129
|
+
timestamp: Date.now(),
|
|
130
|
+
progress: completionResult.progress || 100
|
|
131
|
+
});
|
|
132
|
+
}
|
|
133
|
+
}
|
|
134
|
+
|
|
135
|
+
// Stop polling if no change for 30 seconds
|
|
136
|
+
if (Date.now() - this.lastChangeAt > 30000) {
|
|
137
|
+
this.logger.log('No response change for 30 seconds, stopping polling');
|
|
138
|
+
this.onStatusUpdate(tabId, '⏰ Response timeout');
|
|
139
|
+
this.stopPolling();
|
|
140
|
+
}
|
|
141
|
+
} catch (error) {
|
|
142
|
+
this.logger.error('Error during polling:', error);
|
|
143
|
+
this.onStatusUpdate(tabId, `❌ Polling error: ${error.message}`);
|
|
144
|
+
this.stopPolling();
|
|
145
|
+
}
|
|
146
|
+
}, 2000); // Poll every 2 seconds
|
|
147
|
+
}
|
|
148
|
+
|
|
149
|
+
/**
|
|
150
|
+
* Stop polling for responses
|
|
151
|
+
*/
|
|
152
|
+
stopPolling() {
|
|
153
|
+
if (this.pollingTimer) {
|
|
154
|
+
clearInterval(this.pollingTimer);
|
|
155
|
+
this.pollingTimer = null;
|
|
156
|
+
}
|
|
157
|
+
this.isPolling = false;
|
|
158
|
+
this.responseWaiting = false;
|
|
159
|
+
this.logger.log('Polling stopped');
|
|
160
|
+
}
|
|
161
|
+
|
|
162
|
+
/**
|
|
163
|
+
* Read response from an IDE
|
|
164
|
+
* @param {string} ide - The IDE to read from
|
|
165
|
+
* @returns {Promise<string>} The response text
|
|
166
|
+
*/
|
|
167
|
+
async readResponse(ide) {
|
|
168
|
+
try {
|
|
169
|
+
if (this.electronAPI && this.electronAPI.readChat) {
|
|
170
|
+
return await this.electronAPI.readChat(ide);
|
|
171
|
+
} else {
|
|
172
|
+
// Fallback for non-Electron environments
|
|
173
|
+
return 'Simulated response from ' + ide;
|
|
174
|
+
}
|
|
175
|
+
} catch (error) {
|
|
176
|
+
this.logger.error('Error reading response:', error);
|
|
177
|
+
return '';
|
|
178
|
+
}
|
|
179
|
+
}
|
|
180
|
+
|
|
181
|
+
/**
|
|
182
|
+
* Detect if a response is complete
|
|
183
|
+
* @param {string} ide - The IDE name
|
|
184
|
+
* @param {string} response - The current response text
|
|
185
|
+
* @returns {Promise<Object>} Completion detection result
|
|
186
|
+
*/
|
|
187
|
+
async detectResponseCompletion(ide, response) {
|
|
188
|
+
if (ide !== 'windsurf') {
|
|
189
|
+
// For non-Windsurf IDEs, assume response is complete after a delay
|
|
190
|
+
return { completed: true, response };
|
|
191
|
+
}
|
|
192
|
+
|
|
193
|
+
this.logger.log('🔍 Detecting Windsurf response completion...');
|
|
194
|
+
this.responseWaiting = true;
|
|
195
|
+
|
|
196
|
+
try {
|
|
197
|
+
const responseText = response.toLowerCase();
|
|
198
|
+
|
|
199
|
+
// Check for ongoing status indicators - if these are present, response is NOT complete
|
|
200
|
+
const hasOngoingStatus = responseText.includes('surfing') ||
|
|
201
|
+
responseText.includes('navigating') ||
|
|
202
|
+
responseText.includes('swimming') ||
|
|
203
|
+
responseText.includes('floating') ||
|
|
204
|
+
responseText.includes('proposing code') ||
|
|
205
|
+
responseText.includes('reading file') ||
|
|
206
|
+
responseText.includes('searching') ||
|
|
207
|
+
responseText.includes('updating todo list');
|
|
208
|
+
|
|
209
|
+
if (hasOngoingStatus) {
|
|
210
|
+
this.logger.log('⏳ Response still in progress - waiting for completion...');
|
|
211
|
+
return { completed: false, response };
|
|
212
|
+
}
|
|
213
|
+
|
|
214
|
+
// Check for completion indicators
|
|
215
|
+
const hasCompletionIndicators = responseText.includes('thumbs up') ||
|
|
216
|
+
responseText.includes('thumbs down') ||
|
|
217
|
+
responseText.includes('👍') ||
|
|
218
|
+
responseText.includes('👎') ||
|
|
219
|
+
responseText.includes('copy icon') ||
|
|
220
|
+
responseText.includes('bookmark icon') ||
|
|
221
|
+
responseText.includes('bar chart icon') ||
|
|
222
|
+
responseText.includes('view response summary') ||
|
|
223
|
+
responseText.includes('ask anything (⌘l)') ||
|
|
224
|
+
responseText.includes('microphone icon') ||
|
|
225
|
+
responseText.includes('send button') ||
|
|
226
|
+
responseText.includes('+ chat') ||
|
|
227
|
+
responseText.includes('+ chat swe-1') ||
|
|
228
|
+
responseText.includes('paper plane') ||
|
|
229
|
+
responseText.includes('send icon') ||
|
|
230
|
+
responseText.includes('chat swe-1') ||
|
|
231
|
+
responseText.includes('ask anything') ||
|
|
232
|
+
responseText.includes('⌘l') ||
|
|
233
|
+
responseText.includes('command+l');
|
|
234
|
+
|
|
235
|
+
if (hasCompletionIndicators) {
|
|
236
|
+
this.logger.log('🎯 Completion indicators found!');
|
|
237
|
+
this.responseWaiting = false;
|
|
238
|
+
return { completed: true, response };
|
|
239
|
+
}
|
|
240
|
+
|
|
241
|
+
// Check for percentage in response
|
|
242
|
+
const percentageMatch = response.match(/(\d+)%/);
|
|
243
|
+
if (percentageMatch) {
|
|
244
|
+
const percentage = parseInt(percentageMatch[1]);
|
|
245
|
+
this.currentProgress = percentage;
|
|
246
|
+
this.onProgressUpdate(percentage);
|
|
247
|
+
this.logger.log(`📊 Progress detected: ${percentage}%`);
|
|
248
|
+
|
|
249
|
+
if (percentage === 100) {
|
|
250
|
+
this.logger.log('🎉 100% completion detected!');
|
|
251
|
+
this.responseWaiting = false;
|
|
252
|
+
return { completed: true, response, progress: 100 };
|
|
253
|
+
}
|
|
254
|
+
}
|
|
255
|
+
|
|
256
|
+
// Response is not complete
|
|
257
|
+
return { completed: false, response };
|
|
258
|
+
|
|
259
|
+
} catch (error) {
|
|
260
|
+
this.logger.error('Error detecting response completion:', error);
|
|
261
|
+
this.responseWaiting = false;
|
|
262
|
+
return { completed: false, response, error: error.message };
|
|
263
|
+
}
|
|
264
|
+
}
|
|
265
|
+
|
|
266
|
+
/**
|
|
267
|
+
* Pause chat operations
|
|
268
|
+
*/
|
|
269
|
+
pause() {
|
|
270
|
+
this.isPaused = true;
|
|
271
|
+
this.logger.log('Chat operations paused');
|
|
272
|
+
}
|
|
273
|
+
|
|
274
|
+
/**
|
|
275
|
+
* Resume chat operations
|
|
276
|
+
*/
|
|
277
|
+
resume() {
|
|
278
|
+
this.isPaused = false;
|
|
279
|
+
this.logger.log('Chat operations resumed');
|
|
280
|
+
}
|
|
281
|
+
|
|
282
|
+
/**
|
|
283
|
+
* Stop chat operations
|
|
284
|
+
*/
|
|
285
|
+
stop() {
|
|
286
|
+
this.isStopped = true;
|
|
287
|
+
this.stopPolling();
|
|
288
|
+
this.logger.log('Chat operations stopped');
|
|
289
|
+
}
|
|
290
|
+
|
|
291
|
+
/**
|
|
292
|
+
* Reset chat state
|
|
293
|
+
*/
|
|
294
|
+
reset() {
|
|
295
|
+
this.isStopped = false;
|
|
296
|
+
this.isPaused = false;
|
|
297
|
+
this.isPolling = false;
|
|
298
|
+
this.responseWaiting = false;
|
|
299
|
+
this.currentProgress = 0;
|
|
300
|
+
this.stopPolling();
|
|
301
|
+
this.logger.log('Chat state reset');
|
|
302
|
+
}
|
|
303
|
+
|
|
304
|
+
/**
|
|
305
|
+
* Get current chat state
|
|
306
|
+
* @returns {Object} Current chat state
|
|
307
|
+
*/
|
|
308
|
+
getState() {
|
|
309
|
+
return {
|
|
310
|
+
isPolling: this.isPolling,
|
|
311
|
+
isPaused: this.isPaused,
|
|
312
|
+
isStopped: this.isStopped,
|
|
313
|
+
responseWaiting: this.responseWaiting,
|
|
314
|
+
currentProgress: this.currentProgress,
|
|
315
|
+
lastChangeAt: this.lastChangeAt
|
|
316
|
+
};
|
|
317
|
+
}
|
|
318
|
+
|
|
319
|
+
/**
|
|
320
|
+
* Set electron API for Electron environment
|
|
321
|
+
* @param {Object} electronAPI - The electron API object
|
|
322
|
+
*/
|
|
323
|
+
setElectronAPI(electronAPI) {
|
|
324
|
+
this.electronAPI = electronAPI;
|
|
325
|
+
}
|
|
326
|
+
|
|
327
|
+
/**
|
|
328
|
+
* Set callback functions
|
|
329
|
+
* @param {Object} callbacks - Object containing callback functions
|
|
330
|
+
*/
|
|
331
|
+
setCallbacks(callbacks) {
|
|
332
|
+
if (callbacks.onMessageUpdate) {
|
|
333
|
+
this.onMessageUpdate = callbacks.onMessageUpdate;
|
|
334
|
+
}
|
|
335
|
+
if (callbacks.onStatusUpdate) {
|
|
336
|
+
this.onStatusUpdate = callbacks.onStatusUpdate;
|
|
337
|
+
}
|
|
338
|
+
if (callbacks.onProgressUpdate) {
|
|
339
|
+
this.onProgressUpdate = callbacks.onProgressUpdate;
|
|
340
|
+
}
|
|
341
|
+
}
|
|
342
|
+
}
|
|
@@ -0,0 +1,227 @@
|
|
|
1
|
+
/**
|
|
2
|
+
* Jest tests for AppleScript Manager thread closure functionality
|
|
3
|
+
* Tests the ability to close previous chat threads when starting new ones
|
|
4
|
+
*/
|
|
5
|
+
|
|
6
|
+
import { AppleScriptManager } from '../applescript-manager.js';
|
|
7
|
+
import { execSync } from 'child_process';
|
|
8
|
+
|
|
9
|
+
// Mock child_process
|
|
10
|
+
jest.mock('child_process', () => ({
|
|
11
|
+
execSync: jest.fn()
|
|
12
|
+
}));Windsurf: Open Chat
|
|
13
|
+
|
|
14
|
+
|
|
15
|
+
|
|
16
|
+
|
|
17
|
+
|
|
18
|
+
// Mock fs
|
|
19
|
+
jest.mock('fs', () => ({
|
|
20
|
+
writeFileSync: jest.fn(),
|
|
21
|
+
unlinkSync: jest.fn()
|
|
22
|
+
}));
|
|
23
|
+
|
|
24
|
+
// Mock os
|
|
25
|
+
jest.mock('os', () => ({
|
|
26
|
+
tmpdir: () => '/tmp'
|
|
27
|
+
}));
|
|
28
|
+
|
|
29
|
+
describe('AppleScriptManager - Thread Closure', () => {
|
|
30
|
+
let appleScriptManager;
|
|
31
|
+
let mockExecSync;
|
|
32
|
+
|
|
33
|
+
beforeEach(() => {
|
|
34
|
+
appleScriptManager = new AppleScriptManager();
|
|
35
|
+
mockExecSync = execSync;
|
|
36
|
+
mockExecSync.mockClear();
|
|
37
|
+
});
|
|
38
|
+
|
|
39
|
+
afterEach(() => {
|
|
40
|
+
jest.clearAllMocks();
|
|
41
|
+
});
|
|
42
|
+
|
|
43
|
+
describe('closePreviousChatThread', () => {
|
|
44
|
+
it('should close previous chat thread in Cursor', async () => {
|
|
45
|
+
// Mock successful execution
|
|
46
|
+
mockExecSync.mockReturnValue('Previous chat thread closed');
|
|
47
|
+
|
|
48
|
+
const result = await appleScriptManager.closePreviousChatThread('cursor');
|
|
49
|
+
|
|
50
|
+
expect(result.success).toBe(true);
|
|
51
|
+
expect(result.message).toContain('Previous chat thread closed');
|
|
52
|
+
expect(result.method).toBe('applescript');
|
|
53
|
+
expect(mockExecSync).toHaveBeenCalled();
|
|
54
|
+
});
|
|
55
|
+
|
|
56
|
+
it('should close previous chat thread in VS Code', async () => {
|
|
57
|
+
// Mock successful execution
|
|
58
|
+
mockExecSync.mockReturnValue('Previous chat thread closed');
|
|
59
|
+
|
|
60
|
+
const result = await appleScriptManager.closePreviousChatThread('vscode');
|
|
61
|
+
|
|
62
|
+
expect(result.success).toBe(true);
|
|
63
|
+
expect(result.message).toContain('Previous chat thread closed');
|
|
64
|
+
expect(result.method).toBe('applescript');
|
|
65
|
+
expect(mockExecSync).toHaveBeenCalled();
|
|
66
|
+
});
|
|
67
|
+
|
|
68
|
+
it('should close previous chat thread in Windsurf', async () => {
|
|
69
|
+
// Mock successful execution
|
|
70
|
+
mockExecSync.mockReturnValue('Previous chat thread closed');
|
|
71
|
+
|
|
72
|
+
const result = await appleScriptManager.closePreviousChatThread('windsurf');
|
|
73
|
+
|
|
74
|
+
expect(result.success).toBe(true);
|
|
75
|
+
expect(result.message).toContain('Previous chat thread closed');
|
|
76
|
+
expect(result.method).toBe('applescript');
|
|
77
|
+
expect(mockExecSync).toHaveBeenCalled();
|
|
78
|
+
});
|
|
79
|
+
|
|
80
|
+
it('should handle errors gracefully when closing threads', async () => {
|
|
81
|
+
// Mock execution error
|
|
82
|
+
mockExecSync.mockImplementation(() => {
|
|
83
|
+
throw new Error('Failed to close thread');
|
|
84
|
+
});
|
|
85
|
+
|
|
86
|
+
const result = await appleScriptManager.closePreviousChatThread('cursor');
|
|
87
|
+
|
|
88
|
+
expect(result.success).toBe(false);
|
|
89
|
+
expect(result.error).toContain('Failed to close thread');
|
|
90
|
+
expect(result.method).toBe('applescript');
|
|
91
|
+
});
|
|
92
|
+
|
|
93
|
+
it('should return error for unsupported IDE', async () => {
|
|
94
|
+
const result = await appleScriptManager.closePreviousChatThread('unsupported');
|
|
95
|
+
|
|
96
|
+
expect(result.success).toBe(false);
|
|
97
|
+
expect(result.error).toContain('Unsupported IDE');
|
|
98
|
+
expect(mockExecSync).not.toHaveBeenCalled();
|
|
99
|
+
});
|
|
100
|
+
});
|
|
101
|
+
|
|
102
|
+
describe('sendTextWithThreadClosure', () => {
|
|
103
|
+
it('should close previous thread before sending new message to Cursor', async () => {
|
|
104
|
+
// Mock successful executions
|
|
105
|
+
mockExecSync
|
|
106
|
+
.mockReturnValueOnce('Previous chat thread closed')
|
|
107
|
+
.mockReturnValueOnce('Message sent to Cursor: Test message')
|
|
108
|
+
.mockReturnValueOnce('Message sent to Cursor: Test message');
|
|
109
|
+
|
|
110
|
+
const result = await appleScriptManager.sendTextWithThreadClosure('Test message', 'cursor');
|
|
111
|
+
|
|
112
|
+
expect(result.success).toBe(true);
|
|
113
|
+
expect(result.message).toContain('Test message');
|
|
114
|
+
expect(mockExecSync).toHaveBeenCalledTimes(3); // Once for close, twice for send (sendText calls execSync twice)
|
|
115
|
+
});
|
|
116
|
+
|
|
117
|
+
it('should close previous thread before sending new message to VS Code', async () => {
|
|
118
|
+
// Mock successful executions
|
|
119
|
+
mockExecSync
|
|
120
|
+
.mockReturnValueOnce('Previous chat thread closed')
|
|
121
|
+
.mockImplementationOnce(() => {
|
|
122
|
+
// Simulate successful AppleScript execution for text sending
|
|
123
|
+
return 'Message sent via AppleScript to VS Code GitHub Copilot Chat';
|
|
124
|
+
});
|
|
125
|
+
|
|
126
|
+
const result = await appleScriptManager.sendTextWithThreadClosure('Test message', 'vscode');
|
|
127
|
+
|
|
128
|
+
expect(result.success).toBe(true); // VS Code is now supported by AppleScript manager
|
|
129
|
+
expect(result.method).toBe('applescript');
|
|
130
|
+
expect(result.threadClosure).toContain('Previous thread closed');
|
|
131
|
+
expect(mockExecSync).toHaveBeenCalledTimes(2); // Once for thread closure, once for sending
|
|
132
|
+
});
|
|
133
|
+
|
|
134
|
+
it('should close previous thread before sending new message to Windsurf', async () => {
|
|
135
|
+
// Mock successful executions
|
|
136
|
+
mockExecSync
|
|
137
|
+
.mockReturnValueOnce('Previous chat thread closed')
|
|
138
|
+
.mockImplementationOnce(() => {
|
|
139
|
+
throw new Error('AppleScript failed');
|
|
140
|
+
});
|
|
141
|
+
|
|
142
|
+
const result = await appleScriptManager.sendTextWithThreadClosure('Test message', 'windsurf');
|
|
143
|
+
|
|
144
|
+
expect(result.success).toBe(true);
|
|
145
|
+
expect(result.message).toContain('Test message');
|
|
146
|
+
expect(mockExecSync).toHaveBeenCalled(); // Should be called at least once for thread closure
|
|
147
|
+
});
|
|
148
|
+
|
|
149
|
+
it('should handle thread closure failure gracefully', async () => {
|
|
150
|
+
// Mock thread closure failure but successful message send
|
|
151
|
+
mockExecSync
|
|
152
|
+
.mockImplementationOnce(() => {
|
|
153
|
+
throw new Error('Failed to close thread');
|
|
154
|
+
})
|
|
155
|
+
.mockImplementationOnce(() => {
|
|
156
|
+
throw new Error('AppleScript failed');
|
|
157
|
+
});
|
|
158
|
+
|
|
159
|
+
const result = await appleScriptManager.sendTextWithThreadClosure('Test message', 'cursor');
|
|
160
|
+
|
|
161
|
+
expect(result.success).toBe(true);
|
|
162
|
+
expect(result.message).toContain('Test message');
|
|
163
|
+
expect(result.warning).toContain('Failed to close previous thread');
|
|
164
|
+
expect(mockExecSync).toHaveBeenCalled(); // Should be called at least once
|
|
165
|
+
});
|
|
166
|
+
|
|
167
|
+
it('should handle both thread closure and message send failures', async () => {
|
|
168
|
+
// Mock both failures
|
|
169
|
+
mockExecSync
|
|
170
|
+
.mockImplementationOnce(() => {
|
|
171
|
+
throw new Error('Failed to close thread');
|
|
172
|
+
})
|
|
173
|
+
.mockImplementationOnce(() => {
|
|
174
|
+
throw new Error('AppleScript failed');
|
|
175
|
+
});
|
|
176
|
+
|
|
177
|
+
const result = await appleScriptManager.sendTextWithThreadClosure('Test message', 'cursor');
|
|
178
|
+
|
|
179
|
+
expect(result.success).toBe(true); // Should still succeed due to simulated fallback
|
|
180
|
+
expect(result.message).toContain('Test message');
|
|
181
|
+
expect(mockExecSync).toHaveBeenCalled(); // Should be called at least once
|
|
182
|
+
});
|
|
183
|
+
|
|
184
|
+
it('should validate input parameters', async () => {
|
|
185
|
+
const result = await appleScriptManager.sendTextWithThreadClosure('', 'cursor');
|
|
186
|
+
|
|
187
|
+
expect(result.success).toBe(false);
|
|
188
|
+
expect(result.error).toContain('Invalid text');
|
|
189
|
+
expect(mockExecSync).not.toHaveBeenCalled();
|
|
190
|
+
});
|
|
191
|
+
});
|
|
192
|
+
|
|
193
|
+
describe('AppleScript Content Validation', () => {
|
|
194
|
+
it('should generate correct AppleScript for Cursor thread closure', async () => {
|
|
195
|
+
mockExecSync.mockReturnValue('Success');
|
|
196
|
+
|
|
197
|
+
await appleScriptManager.closePreviousChatThread('cursor');
|
|
198
|
+
|
|
199
|
+
const callArgs = mockExecSync.mock.calls[0][0];
|
|
200
|
+
expect(callArgs).toContain('osascript');
|
|
201
|
+
// The script content is written to a temp file, so we check the osascript call
|
|
202
|
+
expect(mockExecSync).toHaveBeenCalledWith(expect.stringContaining('osascript'), expect.any(Object));
|
|
203
|
+
});
|
|
204
|
+
|
|
205
|
+
it('should generate correct AppleScript for VS Code thread closure', async () => {
|
|
206
|
+
mockExecSync.mockReturnValue('Success');
|
|
207
|
+
|
|
208
|
+
await appleScriptManager.closePreviousChatThread('vscode');
|
|
209
|
+
|
|
210
|
+
const callArgs = mockExecSync.mock.calls[0][0];
|
|
211
|
+
expect(callArgs).toContain('osascript');
|
|
212
|
+
// The script content is written to a temp file, so we check the osascript call
|
|
213
|
+
expect(mockExecSync).toHaveBeenCalledWith(expect.stringContaining('osascript'), expect.any(Object));
|
|
214
|
+
});
|
|
215
|
+
|
|
216
|
+
it('should generate correct AppleScript for Windsurf thread closure', async () => {
|
|
217
|
+
mockExecSync.mockReturnValue('Success');
|
|
218
|
+
|
|
219
|
+
await appleScriptManager.closePreviousChatThread('windsurf');
|
|
220
|
+
|
|
221
|
+
const callArgs = mockExecSync.mock.calls[0][0];
|
|
222
|
+
expect(callArgs).toContain('osascript');
|
|
223
|
+
// The script content is written to a temp file, so we check the osascript call
|
|
224
|
+
expect(mockExecSync).toHaveBeenCalledWith(expect.stringContaining('osascript'), expect.any(Object));
|
|
225
|
+
});
|
|
226
|
+
});
|
|
227
|
+
});
|