@lobehub/chat 1.99.5 → 1.99.6
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/.cursor/rules/code-review.mdc +38 -34
- package/.cursor/rules/system-role.mdc +8 -3
- package/.cursor/rules/testing-guide/testing-guide.mdc +155 -233
- package/CHANGELOG.md +25 -0
- package/apps/desktop/package.json +1 -1
- package/changelog/v1.json +9 -0
- package/package.json +1 -1
- package/src/services/__tests__/chat.test.ts +998 -62
- package/src/services/chat.ts +104 -58
- package/src/utils/url.test.ts +42 -1
- package/src/utils/url.ts +28 -0
@@ -2,7 +2,7 @@ import { LobeChatPluginManifest } from '@lobehub/chat-plugin-sdk';
|
|
2
2
|
import { act } from '@testing-library/react';
|
3
3
|
import { merge } from 'lodash-es';
|
4
4
|
import OpenAI from 'openai';
|
5
|
-
import { beforeEach, describe, expect, it, vi } from 'vitest';
|
5
|
+
import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
|
6
6
|
|
7
7
|
import { DEFAULT_AGENT_CONFIG } from '@/const/settings';
|
8
8
|
import {
|
@@ -36,7 +36,8 @@ import { modelConfigSelectors } from '@/store/user/selectors';
|
|
36
36
|
import { UserSettingsState, initialSettingsState } from '@/store/user/slices/settings/initialState';
|
37
37
|
import { DalleManifest } from '@/tools/dalle';
|
38
38
|
import { WebBrowsingManifest } from '@/tools/web-browsing';
|
39
|
-
import {
|
39
|
+
import { ChatErrorType } from '@/types/fetch';
|
40
|
+
import { ChatImageItem, ChatMessage } from '@/types/message';
|
40
41
|
import { ChatStreamPayload, type OpenAIChatMessage } from '@/types/openai/chat';
|
41
42
|
import { LobeTool } from '@/types/tool';
|
42
43
|
|
@@ -58,15 +59,48 @@ vi.mock('@/utils/fetch', async (importOriginal) => {
|
|
58
59
|
return { ...(module as any), getMessageError: vi.fn() };
|
59
60
|
});
|
60
61
|
|
61
|
-
|
62
|
+
// Mock image processing utilities
|
63
|
+
vi.mock('@/utils/url', () => ({
|
64
|
+
isLocalUrl: vi.fn(),
|
65
|
+
}));
|
66
|
+
|
67
|
+
vi.mock('@/utils/imageToBase64', () => ({
|
68
|
+
imageUrlToBase64: vi.fn(),
|
69
|
+
}));
|
70
|
+
|
71
|
+
vi.mock('@/libs/model-runtime/utils/uriParser', () => ({
|
72
|
+
parseDataUri: vi.fn(),
|
73
|
+
}));
|
74
|
+
|
75
|
+
afterEach(() => {
|
76
|
+
vi.restoreAllMocks();
|
77
|
+
});
|
78
|
+
|
79
|
+
beforeEach(async () => {
|
62
80
|
// 清除所有模块的缓存
|
63
81
|
vi.resetModules();
|
82
|
+
|
64
83
|
// 默认设置 isServerMode 为 false
|
65
84
|
vi.mock('@/const/version', () => ({
|
66
85
|
isServerMode: false,
|
67
86
|
isDeprecatedEdition: true,
|
68
87
|
isDesktop: false,
|
69
88
|
}));
|
89
|
+
|
90
|
+
// Reset all mocks
|
91
|
+
vi.clearAllMocks();
|
92
|
+
|
93
|
+
// Set default mock return values for image processing utilities
|
94
|
+
const { isLocalUrl } = await import('@/utils/url');
|
95
|
+
const { imageUrlToBase64 } = await import('@/utils/imageToBase64');
|
96
|
+
const { parseDataUri } = await import('@/libs/model-runtime/utils/uriParser');
|
97
|
+
|
98
|
+
vi.mocked(parseDataUri).mockReturnValue({ type: 'url', base64: null, mimeType: null });
|
99
|
+
vi.mocked(isLocalUrl).mockReturnValue(false);
|
100
|
+
vi.mocked(imageUrlToBase64).mockResolvedValue({
|
101
|
+
base64: 'mock-base64',
|
102
|
+
mimeType: 'image/jpeg',
|
103
|
+
});
|
70
104
|
});
|
71
105
|
|
72
106
|
// mock auth
|
@@ -142,6 +176,164 @@ describe('ChatService', () => {
|
|
142
176
|
);
|
143
177
|
});
|
144
178
|
|
179
|
+
describe('extendParams functionality', () => {
|
180
|
+
it('should add reasoning parameters when model supports enableReasoning and user enables it', async () => {
|
181
|
+
const getChatCompletionSpy = vi.spyOn(chatService, 'getChatCompletion');
|
182
|
+
const messages = [{ content: 'Test reasoning', role: 'user' }] as ChatMessage[];
|
183
|
+
|
184
|
+
// Mock aiModelSelectors for extend params support
|
185
|
+
vi.spyOn(aiModelSelectors, 'isModelHasExtendParams').mockReturnValue(() => true);
|
186
|
+
vi.spyOn(aiModelSelectors, 'modelExtendParams').mockReturnValue(() => ['enableReasoning']);
|
187
|
+
|
188
|
+
// Mock agent chat config with reasoning enabled
|
189
|
+
vi.spyOn(agentChatConfigSelectors, 'currentChatConfig').mockReturnValue({
|
190
|
+
enableReasoning: true,
|
191
|
+
reasoningBudgetToken: 2048,
|
192
|
+
searchMode: 'off',
|
193
|
+
} as any);
|
194
|
+
|
195
|
+
await chatService.createAssistantMessage({
|
196
|
+
messages,
|
197
|
+
model: 'deepseek-reasoner',
|
198
|
+
provider: 'deepseek',
|
199
|
+
plugins: [],
|
200
|
+
});
|
201
|
+
|
202
|
+
expect(getChatCompletionSpy).toHaveBeenCalledWith(
|
203
|
+
expect.objectContaining({
|
204
|
+
thinking: {
|
205
|
+
budget_tokens: 2048,
|
206
|
+
type: 'enabled',
|
207
|
+
},
|
208
|
+
}),
|
209
|
+
undefined,
|
210
|
+
);
|
211
|
+
});
|
212
|
+
|
213
|
+
it('should disable reasoning when model supports enableReasoning but user disables it', async () => {
|
214
|
+
const getChatCompletionSpy = vi.spyOn(chatService, 'getChatCompletion');
|
215
|
+
const messages = [{ content: 'Test no reasoning', role: 'user' }] as ChatMessage[];
|
216
|
+
|
217
|
+
// Mock aiModelSelectors for extend params support
|
218
|
+
vi.spyOn(aiModelSelectors, 'isModelHasExtendParams').mockReturnValue(() => true);
|
219
|
+
vi.spyOn(aiModelSelectors, 'modelExtendParams').mockReturnValue(() => ['enableReasoning']);
|
220
|
+
|
221
|
+
// Mock agent chat config with reasoning disabled
|
222
|
+
vi.spyOn(agentChatConfigSelectors, 'currentChatConfig').mockReturnValue({
|
223
|
+
enableReasoning: false,
|
224
|
+
searchMode: 'off',
|
225
|
+
} as any);
|
226
|
+
|
227
|
+
await chatService.createAssistantMessage({
|
228
|
+
messages,
|
229
|
+
model: 'deepseek-reasoner',
|
230
|
+
provider: 'deepseek',
|
231
|
+
plugins: [],
|
232
|
+
});
|
233
|
+
|
234
|
+
expect(getChatCompletionSpy).toHaveBeenCalledWith(
|
235
|
+
expect.objectContaining({
|
236
|
+
thinking: {
|
237
|
+
budget_tokens: 0,
|
238
|
+
type: 'disabled',
|
239
|
+
},
|
240
|
+
}),
|
241
|
+
undefined,
|
242
|
+
);
|
243
|
+
});
|
244
|
+
|
245
|
+
it('should use default budget when reasoningBudgetToken is not set', async () => {
|
246
|
+
const getChatCompletionSpy = vi.spyOn(chatService, 'getChatCompletion');
|
247
|
+
const messages = [{ content: 'Test default budget', role: 'user' }] as ChatMessage[];
|
248
|
+
|
249
|
+
// Mock aiModelSelectors for extend params support
|
250
|
+
vi.spyOn(aiModelSelectors, 'isModelHasExtendParams').mockReturnValue(() => true);
|
251
|
+
vi.spyOn(aiModelSelectors, 'modelExtendParams').mockReturnValue(() => ['enableReasoning']);
|
252
|
+
|
253
|
+
// Mock agent chat config with reasoning enabled but no custom budget
|
254
|
+
vi.spyOn(agentChatConfigSelectors, 'currentChatConfig').mockReturnValue({
|
255
|
+
enableReasoning: true,
|
256
|
+
// reasoningBudgetToken is undefined
|
257
|
+
searchMode: 'off',
|
258
|
+
} as any);
|
259
|
+
|
260
|
+
await chatService.createAssistantMessage({
|
261
|
+
messages,
|
262
|
+
model: 'deepseek-reasoner',
|
263
|
+
provider: 'deepseek',
|
264
|
+
plugins: [],
|
265
|
+
});
|
266
|
+
|
267
|
+
expect(getChatCompletionSpy).toHaveBeenCalledWith(
|
268
|
+
expect.objectContaining({
|
269
|
+
thinking: {
|
270
|
+
budget_tokens: 1024, // default value
|
271
|
+
type: 'enabled',
|
272
|
+
},
|
273
|
+
}),
|
274
|
+
undefined,
|
275
|
+
);
|
276
|
+
});
|
277
|
+
|
278
|
+
it('should set reasoning_effort when model supports reasoningEffort and user configures it', async () => {
|
279
|
+
const getChatCompletionSpy = vi.spyOn(chatService, 'getChatCompletion');
|
280
|
+
const messages = [{ content: 'Test reasoning effort', role: 'user' }] as ChatMessage[];
|
281
|
+
|
282
|
+
// Mock aiModelSelectors for extend params support
|
283
|
+
vi.spyOn(aiModelSelectors, 'isModelHasExtendParams').mockReturnValue(() => true);
|
284
|
+
vi.spyOn(aiModelSelectors, 'modelExtendParams').mockReturnValue(() => ['reasoningEffort']);
|
285
|
+
|
286
|
+
// Mock agent chat config with reasoning effort set
|
287
|
+
vi.spyOn(agentChatConfigSelectors, 'currentChatConfig').mockReturnValue({
|
288
|
+
reasoningEffort: 'high',
|
289
|
+
searchMode: 'off',
|
290
|
+
} as any);
|
291
|
+
|
292
|
+
await chatService.createAssistantMessage({
|
293
|
+
messages,
|
294
|
+
model: 'test-model',
|
295
|
+
provider: 'test-provider',
|
296
|
+
plugins: [],
|
297
|
+
});
|
298
|
+
|
299
|
+
expect(getChatCompletionSpy).toHaveBeenCalledWith(
|
300
|
+
expect.objectContaining({
|
301
|
+
reasoning_effort: 'high',
|
302
|
+
}),
|
303
|
+
undefined,
|
304
|
+
);
|
305
|
+
});
|
306
|
+
|
307
|
+
it('should set thinkingBudget when model supports thinkingBudget and user configures it', async () => {
|
308
|
+
const getChatCompletionSpy = vi.spyOn(chatService, 'getChatCompletion');
|
309
|
+
const messages = [{ content: 'Test thinking budget', role: 'user' }] as ChatMessage[];
|
310
|
+
|
311
|
+
// Mock aiModelSelectors for extend params support
|
312
|
+
vi.spyOn(aiModelSelectors, 'isModelHasExtendParams').mockReturnValue(() => true);
|
313
|
+
vi.spyOn(aiModelSelectors, 'modelExtendParams').mockReturnValue(() => ['thinkingBudget']);
|
314
|
+
|
315
|
+
// Mock agent chat config with thinking budget set
|
316
|
+
vi.spyOn(agentChatConfigSelectors, 'currentChatConfig').mockReturnValue({
|
317
|
+
thinkingBudget: 5000,
|
318
|
+
searchMode: 'off',
|
319
|
+
} as any);
|
320
|
+
|
321
|
+
await chatService.createAssistantMessage({
|
322
|
+
messages,
|
323
|
+
model: 'test-model',
|
324
|
+
provider: 'test-provider',
|
325
|
+
plugins: [],
|
326
|
+
});
|
327
|
+
|
328
|
+
expect(getChatCompletionSpy).toHaveBeenCalledWith(
|
329
|
+
expect.objectContaining({
|
330
|
+
thinkingBudget: 5000,
|
331
|
+
}),
|
332
|
+
undefined,
|
333
|
+
);
|
334
|
+
});
|
335
|
+
});
|
336
|
+
|
145
337
|
describe('should handle content correctly for vision models', () => {
|
146
338
|
it('should include image content when with vision model', async () => {
|
147
339
|
const messages = [
|
@@ -209,6 +401,263 @@ describe('ChatService', () => {
|
|
209
401
|
});
|
210
402
|
});
|
211
403
|
|
404
|
+
describe('local image URL conversion', () => {
|
405
|
+
it('should convert local image URLs to base64 and call processImageList', async () => {
|
406
|
+
const { isLocalUrl } = await import('@/utils/url');
|
407
|
+
const { imageUrlToBase64 } = await import('@/utils/imageToBase64');
|
408
|
+
const { parseDataUri } = await import('@/libs/model-runtime/utils/uriParser');
|
409
|
+
|
410
|
+
// Mock for local URL
|
411
|
+
vi.mocked(parseDataUri).mockReturnValue({ type: 'url', base64: null, mimeType: null });
|
412
|
+
vi.mocked(isLocalUrl).mockReturnValue(true); // This is a local URL
|
413
|
+
vi.mocked(imageUrlToBase64).mockResolvedValue({
|
414
|
+
base64: 'converted-base64-content',
|
415
|
+
mimeType: 'image/png',
|
416
|
+
});
|
417
|
+
|
418
|
+
const messages = [
|
419
|
+
{
|
420
|
+
content: 'Hello',
|
421
|
+
role: 'user',
|
422
|
+
imageList: [
|
423
|
+
{
|
424
|
+
id: 'file1',
|
425
|
+
url: 'http://127.0.0.1:3000/uploads/image.png', // Real local URL
|
426
|
+
alt: 'local-image.png',
|
427
|
+
},
|
428
|
+
],
|
429
|
+
createdAt: Date.now(),
|
430
|
+
id: 'test-id',
|
431
|
+
meta: {},
|
432
|
+
updatedAt: Date.now(),
|
433
|
+
},
|
434
|
+
] as ChatMessage[];
|
435
|
+
|
436
|
+
// Spy on processImageList method
|
437
|
+
const processImageListSpy = vi.spyOn(chatService as any, 'processImageList');
|
438
|
+
const getChatCompletionSpy = vi.spyOn(chatService, 'getChatCompletion');
|
439
|
+
|
440
|
+
await chatService.createAssistantMessage({
|
441
|
+
messages,
|
442
|
+
plugins: [],
|
443
|
+
model: 'gpt-4-vision-preview',
|
444
|
+
});
|
445
|
+
|
446
|
+
// Verify processImageList was called with correct arguments
|
447
|
+
expect(processImageListSpy).toHaveBeenCalledWith({
|
448
|
+
imageList: [
|
449
|
+
{
|
450
|
+
id: 'file1',
|
451
|
+
url: 'http://127.0.0.1:3000/uploads/image.png',
|
452
|
+
alt: 'local-image.png',
|
453
|
+
},
|
454
|
+
],
|
455
|
+
model: 'gpt-4-vision-preview',
|
456
|
+
provider: undefined,
|
457
|
+
});
|
458
|
+
|
459
|
+
// Verify the utility functions were called
|
460
|
+
expect(parseDataUri).toHaveBeenCalledWith('http://127.0.0.1:3000/uploads/image.png');
|
461
|
+
expect(isLocalUrl).toHaveBeenCalledWith('http://127.0.0.1:3000/uploads/image.png');
|
462
|
+
expect(imageUrlToBase64).toHaveBeenCalledWith('http://127.0.0.1:3000/uploads/image.png');
|
463
|
+
|
464
|
+
// Verify the final result contains base64 converted URL
|
465
|
+
expect(getChatCompletionSpy).toHaveBeenCalledWith(
|
466
|
+
{
|
467
|
+
messages: [
|
468
|
+
{
|
469
|
+
content: [
|
470
|
+
{
|
471
|
+
text: 'Hello',
|
472
|
+
type: 'text',
|
473
|
+
},
|
474
|
+
{
|
475
|
+
image_url: {
|
476
|
+
detail: 'auto',
|
477
|
+
url: '-base64-content',
|
478
|
+
},
|
479
|
+
type: 'image_url',
|
480
|
+
},
|
481
|
+
],
|
482
|
+
role: 'user',
|
483
|
+
},
|
484
|
+
],
|
485
|
+
model: 'gpt-4-vision-preview',
|
486
|
+
},
|
487
|
+
undefined,
|
488
|
+
);
|
489
|
+
});
|
490
|
+
|
491
|
+
it('should not convert remote URLs to base64 and call processImageList', async () => {
|
492
|
+
const { isLocalUrl } = await import('@/utils/url');
|
493
|
+
const { imageUrlToBase64 } = await import('@/utils/imageToBase64');
|
494
|
+
const { parseDataUri } = await import('@/libs/model-runtime/utils/uriParser');
|
495
|
+
|
496
|
+
// Mock for remote URL
|
497
|
+
vi.mocked(parseDataUri).mockReturnValue({ type: 'url', base64: null, mimeType: null });
|
498
|
+
vi.mocked(isLocalUrl).mockReturnValue(false); // This is NOT a local URL
|
499
|
+
vi.mocked(imageUrlToBase64).mockClear(); // Clear to ensure it's not called
|
500
|
+
|
501
|
+
const messages = [
|
502
|
+
{
|
503
|
+
content: 'Hello',
|
504
|
+
role: 'user',
|
505
|
+
imageList: [
|
506
|
+
{
|
507
|
+
id: 'file1',
|
508
|
+
url: 'https://example.com/remote-image.jpg', // Remote URL
|
509
|
+
alt: 'remote-image.jpg',
|
510
|
+
},
|
511
|
+
],
|
512
|
+
createdAt: Date.now(),
|
513
|
+
id: 'test-id-2',
|
514
|
+
meta: {},
|
515
|
+
updatedAt: Date.now(),
|
516
|
+
},
|
517
|
+
] as ChatMessage[];
|
518
|
+
|
519
|
+
// Spy on processImageList method
|
520
|
+
const processImageListSpy = vi.spyOn(chatService as any, 'processImageList');
|
521
|
+
const getChatCompletionSpy = vi.spyOn(chatService, 'getChatCompletion');
|
522
|
+
|
523
|
+
await chatService.createAssistantMessage({
|
524
|
+
messages,
|
525
|
+
plugins: [],
|
526
|
+
model: 'gpt-4-vision-preview',
|
527
|
+
});
|
528
|
+
|
529
|
+
// Verify processImageList was called
|
530
|
+
expect(processImageListSpy).toHaveBeenCalledWith({
|
531
|
+
imageList: [
|
532
|
+
{
|
533
|
+
id: 'file1',
|
534
|
+
url: 'https://example.com/remote-image.jpg',
|
535
|
+
alt: 'remote-image.jpg',
|
536
|
+
},
|
537
|
+
],
|
538
|
+
model: 'gpt-4-vision-preview',
|
539
|
+
provider: undefined,
|
540
|
+
});
|
541
|
+
|
542
|
+
// Verify the utility functions were called
|
543
|
+
expect(parseDataUri).toHaveBeenCalledWith('https://example.com/remote-image.jpg');
|
544
|
+
expect(isLocalUrl).toHaveBeenCalledWith('https://example.com/remote-image.jpg');
|
545
|
+
expect(imageUrlToBase64).not.toHaveBeenCalled(); // Should NOT be called for remote URLs
|
546
|
+
|
547
|
+
// Verify the final result preserves original URL
|
548
|
+
expect(getChatCompletionSpy).toHaveBeenCalledWith(
|
549
|
+
{
|
550
|
+
messages: [
|
551
|
+
{
|
552
|
+
content: [
|
553
|
+
{
|
554
|
+
text: 'Hello',
|
555
|
+
type: 'text',
|
556
|
+
},
|
557
|
+
{
|
558
|
+
image_url: { detail: 'auto', url: 'https://example.com/remote-image.jpg' },
|
559
|
+
type: 'image_url',
|
560
|
+
},
|
561
|
+
],
|
562
|
+
role: 'user',
|
563
|
+
},
|
564
|
+
],
|
565
|
+
model: 'gpt-4-vision-preview',
|
566
|
+
},
|
567
|
+
undefined,
|
568
|
+
);
|
569
|
+
});
|
570
|
+
|
571
|
+
it('should handle mixed local and remote URLs correctly', async () => {
|
572
|
+
const { isLocalUrl } = await import('@/utils/url');
|
573
|
+
const { imageUrlToBase64 } = await import('@/utils/imageToBase64');
|
574
|
+
const { parseDataUri } = await import('@/libs/model-runtime/utils/uriParser');
|
575
|
+
|
576
|
+
// Mock parseDataUri to always return url type
|
577
|
+
vi.mocked(parseDataUri).mockReturnValue({ type: 'url', base64: null, mimeType: null });
|
578
|
+
|
579
|
+
// Mock isLocalUrl to return true only for 127.0.0.1 URLs
|
580
|
+
vi.mocked(isLocalUrl).mockImplementation((url: string) => {
|
581
|
+
return new URL(url).hostname === '127.0.0.1';
|
582
|
+
});
|
583
|
+
|
584
|
+
// Mock imageUrlToBase64 for conversion
|
585
|
+
vi.mocked(imageUrlToBase64).mockResolvedValue({
|
586
|
+
base64: 'local-file-base64',
|
587
|
+
mimeType: 'image/jpeg',
|
588
|
+
});
|
589
|
+
|
590
|
+
const messages = [
|
591
|
+
{
|
592
|
+
content: 'Multiple images',
|
593
|
+
role: 'user',
|
594
|
+
imageList: [
|
595
|
+
{
|
596
|
+
id: 'local1',
|
597
|
+
url: 'http://127.0.0.1:3000/local1.jpg', // Local URL
|
598
|
+
alt: 'local1.jpg',
|
599
|
+
},
|
600
|
+
{
|
601
|
+
id: 'remote1',
|
602
|
+
url: 'https://example.com/remote1.png', // Remote URL
|
603
|
+
alt: 'remote1.png',
|
604
|
+
},
|
605
|
+
{
|
606
|
+
id: 'local2',
|
607
|
+
url: 'http://127.0.0.1:8080/local2.gif', // Another local URL
|
608
|
+
alt: 'local2.gif',
|
609
|
+
},
|
610
|
+
],
|
611
|
+
createdAt: Date.now(),
|
612
|
+
id: 'test-id-3',
|
613
|
+
meta: {},
|
614
|
+
updatedAt: Date.now(),
|
615
|
+
},
|
616
|
+
] as ChatMessage[];
|
617
|
+
|
618
|
+
const processImageListSpy = vi.spyOn(chatService as any, 'processImageList');
|
619
|
+
const getChatCompletionSpy = vi.spyOn(chatService, 'getChatCompletion');
|
620
|
+
|
621
|
+
await chatService.createAssistantMessage({
|
622
|
+
messages,
|
623
|
+
plugins: [],
|
624
|
+
model: 'gpt-4-vision-preview',
|
625
|
+
});
|
626
|
+
|
627
|
+
// Verify processImageList was called
|
628
|
+
expect(processImageListSpy).toHaveBeenCalledWith({
|
629
|
+
imageList: [
|
630
|
+
{ id: 'local1', url: 'http://127.0.0.1:3000/local1.jpg', alt: 'local1.jpg' },
|
631
|
+
{ id: 'remote1', url: 'https://example.com/remote1.png', alt: 'remote1.png' },
|
632
|
+
{ id: 'local2', url: 'http://127.0.0.1:8080/local2.gif', alt: 'local2.gif' },
|
633
|
+
],
|
634
|
+
model: 'gpt-4-vision-preview',
|
635
|
+
provider: undefined,
|
636
|
+
});
|
637
|
+
|
638
|
+
// Verify isLocalUrl was called for each image
|
639
|
+
expect(isLocalUrl).toHaveBeenCalledWith('http://127.0.0.1:3000/local1.jpg');
|
640
|
+
expect(isLocalUrl).toHaveBeenCalledWith('https://example.com/remote1.png');
|
641
|
+
expect(isLocalUrl).toHaveBeenCalledWith('http://127.0.0.1:8080/local2.gif');
|
642
|
+
|
643
|
+
// Verify imageUrlToBase64 was called only for local URLs
|
644
|
+
expect(imageUrlToBase64).toHaveBeenCalledWith('http://127.0.0.1:3000/local1.jpg');
|
645
|
+
expect(imageUrlToBase64).toHaveBeenCalledWith('http://127.0.0.1:8080/local2.gif');
|
646
|
+
expect(imageUrlToBase64).toHaveBeenCalledTimes(2); // Only for local URLs
|
647
|
+
|
648
|
+
// Verify the final result has correct URLs
|
649
|
+
const callArgs = getChatCompletionSpy.mock.calls[0][0];
|
650
|
+
const imageContent = (callArgs.messages?.[0].content as any[])?.filter(
|
651
|
+
(c) => c.type === 'image_url',
|
652
|
+
);
|
653
|
+
|
654
|
+
expect(imageContent).toHaveLength(3);
|
655
|
+
expect(imageContent[0].image_url.url).toBe('-file-base64'); // Local converted
|
656
|
+
expect(imageContent[1].image_url.url).toBe('https://example.com/remote1.png'); // Remote preserved
|
657
|
+
expect(imageContent[2].image_url.url).toBe('-file-base64'); // Local converted
|
658
|
+
});
|
659
|
+
});
|
660
|
+
|
212
661
|
describe('with tools messages', () => {
|
213
662
|
it('should inject a tool system role for models with tools', async () => {
|
214
663
|
const getChatCompletionSpy = vi.spyOn(chatService, 'getChatCompletion');
|
@@ -607,6 +1056,15 @@ describe('ChatService', () => {
|
|
607
1056
|
});
|
608
1057
|
|
609
1058
|
describe('getChatCompletion', () => {
|
1059
|
+
let mockFetchSSE: any;
|
1060
|
+
|
1061
|
+
beforeEach(async () => {
|
1062
|
+
// Setup common fetchSSE mock for getChatCompletion tests
|
1063
|
+
const { fetchSSE } = await import('@/utils/fetch');
|
1064
|
+
mockFetchSSE = vi.fn().mockResolvedValue(new Response('mock response'));
|
1065
|
+
vi.mocked(fetchSSE).mockImplementation(mockFetchSSE);
|
1066
|
+
});
|
1067
|
+
|
610
1068
|
it('should make a POST request with the correct payload', async () => {
|
611
1069
|
const params: Partial<ChatStreamPayload> = {
|
612
1070
|
model: 'test-model',
|
@@ -622,12 +1080,16 @@ describe('ChatService', () => {
|
|
622
1080
|
|
623
1081
|
await chatService.getChatCompletion(params, options);
|
624
1082
|
|
625
|
-
expect(
|
626
|
-
|
627
|
-
|
628
|
-
|
629
|
-
|
1083
|
+
expect(mockFetchSSE).toHaveBeenCalledWith(
|
1084
|
+
expect.any(String),
|
1085
|
+
expect.objectContaining({
|
1086
|
+
body: JSON.stringify(expectedPayload),
|
1087
|
+
headers: expect.any(Object),
|
1088
|
+
method: 'POST',
|
1089
|
+
}),
|
1090
|
+
);
|
630
1091
|
});
|
1092
|
+
|
631
1093
|
it('should make a POST request without response in non-openai provider payload', async () => {
|
632
1094
|
const params: Partial<ChatStreamPayload> = {
|
633
1095
|
model: 'deepseek-reasoner',
|
@@ -647,52 +1109,52 @@ describe('ChatService', () => {
|
|
647
1109
|
|
648
1110
|
await chatService.getChatCompletion(params, options);
|
649
1111
|
|
650
|
-
expect(
|
651
|
-
|
652
|
-
|
653
|
-
|
654
|
-
|
1112
|
+
expect(mockFetchSSE).toHaveBeenCalledWith(
|
1113
|
+
expect.any(String),
|
1114
|
+
expect.objectContaining({
|
1115
|
+
body: JSON.stringify(expectedPayload),
|
1116
|
+
headers: expect.any(Object),
|
1117
|
+
method: 'POST',
|
1118
|
+
}),
|
1119
|
+
);
|
655
1120
|
});
|
656
1121
|
|
657
|
-
it('should
|
658
|
-
// Mock
|
659
|
-
const
|
660
|
-
|
661
|
-
|
662
|
-
|
663
|
-
|
664
|
-
|
665
|
-
|
666
|
-
|
667
|
-
|
1122
|
+
it('should return InvalidAccessCode error when enableFetchOnClient is true and auth is enabled but user is not signed in', async () => {
|
1123
|
+
// Mock fetchSSE to call onErrorHandle with the error
|
1124
|
+
const { fetchSSE } = await import('@/utils/fetch');
|
1125
|
+
|
1126
|
+
const mockFetchSSEWithError = vi.fn().mockImplementation((url, options) => {
|
1127
|
+
// Simulate the error being caught and passed to onErrorHandle
|
1128
|
+
if (options.onErrorHandle) {
|
1129
|
+
const error = {
|
1130
|
+
errorType: ChatErrorType.InvalidAccessCode,
|
1131
|
+
error: new Error('InvalidAccessCode'),
|
1132
|
+
};
|
1133
|
+
options.onErrorHandle(error, { errorType: ChatErrorType.InvalidAccessCode });
|
1134
|
+
}
|
1135
|
+
return Promise.resolve(new Response(''));
|
1136
|
+
});
|
668
1137
|
|
669
|
-
vi.
|
670
|
-
vi.spyOn(modelConfigSelectors, 'isProviderFetchOnClient').mockImplementationOnce(
|
671
|
-
mockModelConfigSelectors.isProviderFetchOnClient,
|
672
|
-
);
|
1138
|
+
vi.mocked(fetchSSE).mockImplementation(mockFetchSSEWithError);
|
673
1139
|
|
674
1140
|
const params: Partial<ChatStreamPayload> = {
|
675
1141
|
model: 'test-model',
|
676
1142
|
messages: [],
|
1143
|
+
provider: 'openai',
|
677
1144
|
};
|
678
|
-
const options = {};
|
679
|
-
const expectedPayload = {
|
680
|
-
model: DEFAULT_AGENT_CONFIG.model,
|
681
|
-
stream: true,
|
682
|
-
...DEFAULT_AGENT_CONFIG.params,
|
683
|
-
...params,
|
684
|
-
};
|
685
|
-
|
686
|
-
const result = await chatService.getChatCompletion(params, options);
|
687
1145
|
|
688
|
-
|
689
|
-
|
690
|
-
|
691
|
-
|
692
|
-
}),
|
693
|
-
method: 'POST',
|
1146
|
+
let errorHandled = false;
|
1147
|
+
const onErrorHandle = vi.fn((error: any) => {
|
1148
|
+
errorHandled = true;
|
1149
|
+
expect(error.errorType).toBe(ChatErrorType.InvalidAccessCode);
|
694
1150
|
});
|
695
|
-
|
1151
|
+
|
1152
|
+
// Call getChatCompletion with onErrorHandle to catch the error
|
1153
|
+
await chatService.getChatCompletion(params, { onErrorHandle });
|
1154
|
+
|
1155
|
+
// Verify that the error was handled
|
1156
|
+
expect(errorHandled).toBe(true);
|
1157
|
+
expect(onErrorHandle).toHaveBeenCalled();
|
696
1158
|
});
|
697
1159
|
|
698
1160
|
// Add more test cases to cover different scenarios and edge cases
|
@@ -717,10 +1179,29 @@ describe('ChatService', () => {
|
|
717
1179
|
|
718
1180
|
describe('fetchPresetTaskResult', () => {
|
719
1181
|
it('should handle successful chat completion response', async () => {
|
720
|
-
//
|
721
|
-
|
1182
|
+
// Mock getChatCompletion to simulate successful completion
|
1183
|
+
const getChatCompletionSpy = vi
|
1184
|
+
.spyOn(chatService, 'getChatCompletion')
|
1185
|
+
.mockImplementation(async (params, options) => {
|
1186
|
+
// Simulate successful response
|
1187
|
+
if (options?.onFinish) {
|
1188
|
+
options.onFinish('AI response', {
|
1189
|
+
type: 'done',
|
1190
|
+
observationId: null,
|
1191
|
+
toolCalls: undefined,
|
1192
|
+
traceId: null,
|
1193
|
+
});
|
1194
|
+
}
|
1195
|
+
if (options?.onMessageHandle) {
|
1196
|
+
options.onMessageHandle({ type: 'text', text: 'AI response' });
|
1197
|
+
}
|
1198
|
+
return Promise.resolve(new Response(''));
|
1199
|
+
});
|
1200
|
+
|
722
1201
|
const params = {
|
723
|
-
|
1202
|
+
messages: [{ content: 'Hello', role: 'user' as const }],
|
1203
|
+
model: 'gpt-4',
|
1204
|
+
provider: 'openai',
|
724
1205
|
};
|
725
1206
|
|
726
1207
|
const onMessageHandle = vi.fn();
|
@@ -748,25 +1229,31 @@ describe('ChatService', () => {
|
|
748
1229
|
});
|
749
1230
|
expect(onError).not.toHaveBeenCalled();
|
750
1231
|
expect(onMessageHandle).toHaveBeenCalled();
|
751
|
-
expect(onLoadingChange).toHaveBeenCalledWith(false); //
|
1232
|
+
expect(onLoadingChange).toHaveBeenCalledWith(false); // Confirm loading state is set to false
|
752
1233
|
expect(onLoadingChange).toHaveBeenCalledTimes(2);
|
753
1234
|
});
|
754
1235
|
|
755
1236
|
it('should handle error in chat completion', async () => {
|
756
|
-
//
|
757
|
-
vi
|
758
|
-
|
759
|
-
|
1237
|
+
// Mock getChatCompletion to simulate error
|
1238
|
+
const getChatCompletionSpy = vi
|
1239
|
+
.spyOn(chatService, 'getChatCompletion')
|
1240
|
+
.mockImplementation(async (params, options) => {
|
1241
|
+
// Simulate error response
|
1242
|
+
if (options?.onErrorHandle) {
|
1243
|
+
options.onErrorHandle({ message: 'translated_response.404', type: 404 });
|
1244
|
+
}
|
1245
|
+
return Promise.resolve(new Response(''));
|
1246
|
+
});
|
760
1247
|
|
761
1248
|
const params = {
|
762
|
-
|
1249
|
+
messages: [{ content: 'Hello', role: 'user' as const }],
|
1250
|
+
model: 'gpt-4',
|
1251
|
+
provider: 'openai',
|
763
1252
|
};
|
764
1253
|
const onError = vi.fn();
|
765
1254
|
const onLoadingChange = vi.fn();
|
766
1255
|
const abortController = new AbortController();
|
767
|
-
const trace = {
|
768
|
-
/* 填充跟踪信息 */
|
769
|
-
};
|
1256
|
+
const trace = {};
|
770
1257
|
|
771
1258
|
await chatService.fetchPresetTaskResult({
|
772
1259
|
params,
|
@@ -780,7 +1267,7 @@ describe('ChatService', () => {
|
|
780
1267
|
message: 'translated_response.404',
|
781
1268
|
type: 404,
|
782
1269
|
});
|
783
|
-
expect(onLoadingChange).toHaveBeenCalledWith(false); //
|
1270
|
+
expect(onLoadingChange).toHaveBeenCalledWith(false); // Confirm loading state is set to false
|
784
1271
|
});
|
785
1272
|
});
|
786
1273
|
|
@@ -910,6 +1397,18 @@ describe('ChatService', () => {
|
|
910
1397
|
// 需要在修改模拟后重新导入相关模块
|
911
1398
|
const { chatService } = await import('../chat');
|
912
1399
|
|
1400
|
+
// Mock processImageList to return expected image content
|
1401
|
+
const processImageListSpy = vi.spyOn(chatService as any, 'processImageList');
|
1402
|
+
processImageListSpy.mockImplementation(async () => {
|
1403
|
+
// Mock the expected return value for an image
|
1404
|
+
return [
|
1405
|
+
{
|
1406
|
+
image_url: { detail: 'auto', url: 'http://example.com/xxx0asd-dsd.png' },
|
1407
|
+
type: 'image_url',
|
1408
|
+
},
|
1409
|
+
];
|
1410
|
+
});
|
1411
|
+
|
913
1412
|
const messages = [
|
914
1413
|
{
|
915
1414
|
content: 'Hello',
|
@@ -941,7 +1440,7 @@ describe('ChatService', () => {
|
|
941
1440
|
{ content: 'Hey', role: 'assistant' }, // Regular user message
|
942
1441
|
] as ChatMessage[];
|
943
1442
|
|
944
|
-
const output = chatService['processMessages']({
|
1443
|
+
const output = await chatService['processMessages']({
|
945
1444
|
messages,
|
946
1445
|
model: 'gpt-4o',
|
947
1446
|
provider: 'openai',
|
@@ -1062,7 +1561,7 @@ describe('ChatService', () => {
|
|
1062
1561
|
});
|
1063
1562
|
});
|
1064
1563
|
|
1065
|
-
it('should handle empty tool calls messages correctly', () => {
|
1564
|
+
it('should handle empty tool calls messages correctly', async () => {
|
1066
1565
|
const messages = [
|
1067
1566
|
{
|
1068
1567
|
content: '## Tools\n\nYou can use these tools',
|
@@ -1075,7 +1574,7 @@ describe('ChatService', () => {
|
|
1075
1574
|
},
|
1076
1575
|
] as ChatMessage[];
|
1077
1576
|
|
1078
|
-
const result = chatService['processMessages']({
|
1577
|
+
const result = await chatService['processMessages']({
|
1079
1578
|
messages,
|
1080
1579
|
model: 'gpt-4',
|
1081
1580
|
provider: 'openai',
|
@@ -1093,7 +1592,7 @@ describe('ChatService', () => {
|
|
1093
1592
|
]);
|
1094
1593
|
});
|
1095
1594
|
|
1096
|
-
it('should handle assistant messages with reasoning correctly', () => {
|
1595
|
+
it('should handle assistant messages with reasoning correctly', async () => {
|
1097
1596
|
const messages = [
|
1098
1597
|
{
|
1099
1598
|
role: 'assistant',
|
@@ -1105,7 +1604,7 @@ describe('ChatService', () => {
|
|
1105
1604
|
},
|
1106
1605
|
] as ChatMessage[];
|
1107
1606
|
|
1108
|
-
const result = chatService['processMessages']({
|
1607
|
+
const result = await chatService['processMessages']({
|
1109
1608
|
messages,
|
1110
1609
|
model: 'gpt-4',
|
1111
1610
|
provider: 'openai',
|
@@ -1128,6 +1627,70 @@ describe('ChatService', () => {
|
|
1128
1627
|
},
|
1129
1628
|
]);
|
1130
1629
|
});
|
1630
|
+
|
1631
|
+
it('should inject INBOX_GUIDE_SYSTEMROLE for welcome questions in inbox session', async () => {
|
1632
|
+
// Don't mock INBOX_GUIDE_SYSTEMROLE, use the real one
|
1633
|
+
const messages: ChatMessage[] = [
|
1634
|
+
{
|
1635
|
+
role: 'user',
|
1636
|
+
content: 'Hello, this is my first question',
|
1637
|
+
createdAt: Date.now(),
|
1638
|
+
id: 'test-welcome',
|
1639
|
+
meta: {},
|
1640
|
+
updatedAt: Date.now(),
|
1641
|
+
},
|
1642
|
+
];
|
1643
|
+
|
1644
|
+
const result = await chatService['processMessages'](
|
1645
|
+
{
|
1646
|
+
messages,
|
1647
|
+
model: 'gpt-4',
|
1648
|
+
provider: 'openai',
|
1649
|
+
},
|
1650
|
+
{
|
1651
|
+
isWelcomeQuestion: true,
|
1652
|
+
trace: { sessionId: 'inbox' },
|
1653
|
+
},
|
1654
|
+
);
|
1655
|
+
|
1656
|
+
// Should have system message with inbox guide content
|
1657
|
+
const systemMessage = result.find((msg) => msg.role === 'system');
|
1658
|
+
expect(systemMessage).toBeDefined();
|
1659
|
+
// Check for characteristic content of the actual INBOX_GUIDE_SYSTEMROLE
|
1660
|
+
expect(systemMessage!.content).toContain('LobeChat Support Assistant');
|
1661
|
+
expect(systemMessage!.content).toContain('LobeHub');
|
1662
|
+
});
|
1663
|
+
|
1664
|
+
it('should inject historySummary into system message when provided', async () => {
|
1665
|
+
const historySummary = 'Previous conversation summary: User discussed AI topics.';
|
1666
|
+
|
1667
|
+
const messages: ChatMessage[] = [
|
1668
|
+
{
|
1669
|
+
role: 'user',
|
1670
|
+
content: 'Continue our discussion',
|
1671
|
+
createdAt: Date.now(),
|
1672
|
+
id: 'test-history',
|
1673
|
+
meta: {},
|
1674
|
+
updatedAt: Date.now(),
|
1675
|
+
},
|
1676
|
+
];
|
1677
|
+
|
1678
|
+
const result = await chatService['processMessages'](
|
1679
|
+
{
|
1680
|
+
messages,
|
1681
|
+
model: 'gpt-4',
|
1682
|
+
provider: 'openai',
|
1683
|
+
},
|
1684
|
+
{
|
1685
|
+
historySummary,
|
1686
|
+
},
|
1687
|
+
);
|
1688
|
+
|
1689
|
+
// Should have system message with history summary
|
1690
|
+
const systemMessage = result.find((msg) => msg.role === 'system');
|
1691
|
+
expect(systemMessage).toBeDefined();
|
1692
|
+
expect(systemMessage!.content).toContain(historySummary);
|
1693
|
+
});
|
1131
1694
|
});
|
1132
1695
|
});
|
1133
1696
|
|
@@ -1139,6 +1702,379 @@ vi.mock('../_auth', async (importOriginal) => {
|
|
1139
1702
|
return importOriginal();
|
1140
1703
|
});
|
1141
1704
|
|
1705
|
+
describe('ChatService private methods', () => {
|
1706
|
+
describe('processImageList', () => {
|
1707
|
+
beforeEach(() => {
|
1708
|
+
vi.resetModules();
|
1709
|
+
});
|
1710
|
+
|
1711
|
+
it('should return empty array if model cannot use vision (non-deprecated)', async () => {
|
1712
|
+
vi.doMock('@/const/version', () => ({
|
1713
|
+
isServerMode: false,
|
1714
|
+
isDeprecatedEdition: false,
|
1715
|
+
isDesktop: false,
|
1716
|
+
}));
|
1717
|
+
const { aiModelSelectors } = await import('@/store/aiInfra');
|
1718
|
+
vi.spyOn(aiModelSelectors, 'isModelSupportVision').mockReturnValue(() => false);
|
1719
|
+
|
1720
|
+
const { chatService } = await import('../chat');
|
1721
|
+
const result = await chatService['processImageList']({
|
1722
|
+
imageList: [{ url: 'image_url', alt: '', id: 'test' } as ChatImageItem],
|
1723
|
+
model: 'any-model',
|
1724
|
+
provider: 'any-provider',
|
1725
|
+
});
|
1726
|
+
expect(result).toEqual([]);
|
1727
|
+
});
|
1728
|
+
|
1729
|
+
it('should process images if model can use vision (non-deprecated)', async () => {
|
1730
|
+
vi.doMock('@/const/version', () => ({
|
1731
|
+
isServerMode: false,
|
1732
|
+
isDeprecatedEdition: false,
|
1733
|
+
isDesktop: false,
|
1734
|
+
}));
|
1735
|
+
const { aiModelSelectors } = await import('@/store/aiInfra');
|
1736
|
+
vi.spyOn(aiModelSelectors, 'isModelSupportVision').mockReturnValue(() => true);
|
1737
|
+
|
1738
|
+
const { chatService } = await import('../chat');
|
1739
|
+
const result = await chatService['processImageList']({
|
1740
|
+
imageList: [{ url: 'image_url', alt: '', id: 'test' } as ChatImageItem],
|
1741
|
+
model: 'any-model',
|
1742
|
+
provider: 'any-provider',
|
1743
|
+
});
|
1744
|
+
expect(result.length).toBe(1);
|
1745
|
+
expect(result[0].type).toBe('image_url');
|
1746
|
+
});
|
1747
|
+
|
1748
|
+
it('should return empty array when vision disabled in deprecated edition', async () => {
|
1749
|
+
vi.doMock('@/const/version', () => ({
|
1750
|
+
isServerMode: false,
|
1751
|
+
isDeprecatedEdition: true,
|
1752
|
+
isDesktop: false,
|
1753
|
+
}));
|
1754
|
+
|
1755
|
+
const { modelProviderSelectors } = await import('@/store/user/selectors');
|
1756
|
+
const spy = vi
|
1757
|
+
.spyOn(modelProviderSelectors, 'isModelEnabledVision')
|
1758
|
+
.mockReturnValue(() => false);
|
1759
|
+
|
1760
|
+
const { chatService } = await import('../chat');
|
1761
|
+
const result = await chatService['processImageList']({
|
1762
|
+
imageList: [{ url: 'image_url', alt: '', id: 'test' } as ChatImageItem],
|
1763
|
+
model: 'any-model',
|
1764
|
+
provider: 'any-provider',
|
1765
|
+
});
|
1766
|
+
|
1767
|
+
expect(spy).toHaveBeenCalled();
|
1768
|
+
expect(result).toEqual([]);
|
1769
|
+
});
|
1770
|
+
|
1771
|
+
it('should process images when vision enabled in deprecated edition', async () => {
|
1772
|
+
vi.doMock('@/const/version', () => ({
|
1773
|
+
isServerMode: false,
|
1774
|
+
isDeprecatedEdition: true,
|
1775
|
+
isDesktop: false,
|
1776
|
+
}));
|
1777
|
+
|
1778
|
+
const { modelProviderSelectors } = await import('@/store/user/selectors');
|
1779
|
+
const spy = vi
|
1780
|
+
.spyOn(modelProviderSelectors, 'isModelEnabledVision')
|
1781
|
+
.mockReturnValue(() => true);
|
1782
|
+
|
1783
|
+
const { chatService } = await import('../chat');
|
1784
|
+
const result = await chatService['processImageList']({
|
1785
|
+
imageList: [{ url: 'image_url' } as ChatImageItem],
|
1786
|
+
model: 'any-model',
|
1787
|
+
provider: 'any-provider',
|
1788
|
+
});
|
1789
|
+
|
1790
|
+
expect(spy).toHaveBeenCalled();
|
1791
|
+
expect(result.length).toBe(1);
|
1792
|
+
expect(result[0].type).toBe('image_url');
|
1793
|
+
});
|
1794
|
+
});
|
1795
|
+
|
1796
|
+
describe('processMessages', () => {
|
1797
|
+
describe('getAssistantContent', () => {
|
1798
|
+
it('should handle assistant message with imageList and content', async () => {
|
1799
|
+
const messages: ChatMessage[] = [
|
1800
|
+
{
|
1801
|
+
role: 'assistant',
|
1802
|
+
content: 'Here is an image.',
|
1803
|
+
imageList: [{ id: 'img1', url: 'http://example.com/image.png', alt: 'test.png' }],
|
1804
|
+
createdAt: Date.now(),
|
1805
|
+
id: 'test-id',
|
1806
|
+
meta: {},
|
1807
|
+
updatedAt: Date.now(),
|
1808
|
+
},
|
1809
|
+
];
|
1810
|
+
const result = await chatService['processMessages']({
|
1811
|
+
messages,
|
1812
|
+
model: 'gpt-4-vision-preview',
|
1813
|
+
provider: 'openai',
|
1814
|
+
});
|
1815
|
+
|
1816
|
+
expect(result[0].content).toEqual([
|
1817
|
+
{ text: 'Here is an image.', type: 'text' },
|
1818
|
+
{ image_url: { detail: 'auto', url: 'http://example.com/image.png' }, type: 'image_url' },
|
1819
|
+
]);
|
1820
|
+
});
|
1821
|
+
|
1822
|
+
it('should handle assistant message with imageList but no content', async () => {
|
1823
|
+
const messages: ChatMessage[] = [
|
1824
|
+
{
|
1825
|
+
role: 'assistant',
|
1826
|
+
content: '',
|
1827
|
+
imageList: [{ id: 'img1', url: 'http://example.com/image.png', alt: 'test.png' }],
|
1828
|
+
createdAt: Date.now(),
|
1829
|
+
id: 'test-id-2',
|
1830
|
+
meta: {},
|
1831
|
+
updatedAt: Date.now(),
|
1832
|
+
},
|
1833
|
+
];
|
1834
|
+
const result = await chatService['processMessages']({
|
1835
|
+
messages,
|
1836
|
+
model: 'gpt-4-vision-preview',
|
1837
|
+
provider: 'openai',
|
1838
|
+
});
|
1839
|
+
|
1840
|
+
expect(result[0].content).toEqual([
|
1841
|
+
{ image_url: { detail: 'auto', url: 'http://example.com/image.png' }, type: 'image_url' },
|
1842
|
+
]);
|
1843
|
+
});
|
1844
|
+
});
|
1845
|
+
|
1846
|
+
it('should not include tool_calls for assistant message if model does not support tools', async () => {
|
1847
|
+
// Mock isCanUseFC to return false
|
1848
|
+
vi.spyOn(
|
1849
|
+
(await import('@/store/aiInfra')).aiModelSelectors,
|
1850
|
+
'isModelSupportToolUse',
|
1851
|
+
).mockReturnValue(() => false);
|
1852
|
+
|
1853
|
+
const messages: ChatMessage[] = [
|
1854
|
+
{
|
1855
|
+
role: 'assistant',
|
1856
|
+
content: 'I have a tool call.',
|
1857
|
+
tools: [
|
1858
|
+
{
|
1859
|
+
id: 'tool_123',
|
1860
|
+
type: 'default',
|
1861
|
+
apiName: 'testApi',
|
1862
|
+
arguments: '{}',
|
1863
|
+
identifier: 'test-plugin',
|
1864
|
+
},
|
1865
|
+
],
|
1866
|
+
createdAt: Date.now(),
|
1867
|
+
id: 'test-id-3',
|
1868
|
+
meta: {},
|
1869
|
+
updatedAt: Date.now(),
|
1870
|
+
},
|
1871
|
+
];
|
1872
|
+
|
1873
|
+
const result = await chatService['processMessages']({
|
1874
|
+
messages,
|
1875
|
+
model: 'some-model-without-fc',
|
1876
|
+
provider: 'openai',
|
1877
|
+
});
|
1878
|
+
|
1879
|
+
expect(result[0].tool_calls).toBeUndefined();
|
1880
|
+
expect(result[0].content).toBe('I have a tool call.');
|
1881
|
+
});
|
1882
|
+
});
|
1883
|
+
|
1884
|
+
describe('reorderToolMessages', () => {
|
1885
|
+
it('should correctly reorder when a tool message appears before the assistant message', () => {
|
1886
|
+
const input: OpenAIChatMessage[] = [
|
1887
|
+
{
|
1888
|
+
role: 'system',
|
1889
|
+
content: 'System message',
|
1890
|
+
},
|
1891
|
+
{
|
1892
|
+
role: 'tool',
|
1893
|
+
tool_call_id: 'tool_call_1',
|
1894
|
+
name: 'test-plugin____testApi',
|
1895
|
+
content: 'Tool result',
|
1896
|
+
},
|
1897
|
+
{
|
1898
|
+
role: 'assistant',
|
1899
|
+
content: '',
|
1900
|
+
tool_calls: [
|
1901
|
+
{ id: 'tool_call_1', type: 'function', function: { name: 'testApi', arguments: '{}' } },
|
1902
|
+
],
|
1903
|
+
},
|
1904
|
+
];
|
1905
|
+
|
1906
|
+
const output = chatService['reorderToolMessages'](input);
|
1907
|
+
|
1908
|
+
// Verify reordering logic works and covers line 688 hasPushed check
|
1909
|
+
// In this test, tool messages are duplicated but the second occurrence is skipped
|
1910
|
+
expect(output.length).toBe(4); // Original has 3, assistant will add corresponding tool message again
|
1911
|
+
expect(output[0].role).toBe('system');
|
1912
|
+
expect(output[1].role).toBe('tool');
|
1913
|
+
expect(output[2].role).toBe('assistant');
|
1914
|
+
expect(output[3].role).toBe('tool'); // Tool message added by assistant's tool_calls
|
1915
|
+
});
|
1916
|
+
});
|
1917
|
+
|
1918
|
+
describe('getChatCompletion', () => {
|
1919
|
+
it('should merge responseAnimation styles correctly', async () => {
|
1920
|
+
const { fetchSSE } = await import('@/utils/fetch');
|
1921
|
+
vi.mock('@/utils/fetch', async (importOriginal) => {
|
1922
|
+
const module = await importOriginal();
|
1923
|
+
return {
|
1924
|
+
...(module as any),
|
1925
|
+
fetchSSE: vi.fn(),
|
1926
|
+
};
|
1927
|
+
});
|
1928
|
+
|
1929
|
+
// Mock provider config
|
1930
|
+
const { aiProviderSelectors } = await import('@/store/aiInfra');
|
1931
|
+
vi.spyOn(aiProviderSelectors, 'providerConfigById').mockReturnValue({
|
1932
|
+
id: 'openai',
|
1933
|
+
settings: {
|
1934
|
+
responseAnimation: 'slow',
|
1935
|
+
},
|
1936
|
+
} as any);
|
1937
|
+
|
1938
|
+
// Mock user preference
|
1939
|
+
const { userGeneralSettingsSelectors } = await import('@/store/user/selectors');
|
1940
|
+
vi.spyOn(userGeneralSettingsSelectors, 'transitionMode').mockReturnValue('smooth');
|
1941
|
+
|
1942
|
+
await chatService.getChatCompletion(
|
1943
|
+
{ provider: 'openai', messages: [] },
|
1944
|
+
{ responseAnimation: { speed: 20 } },
|
1945
|
+
);
|
1946
|
+
|
1947
|
+
expect(fetchSSE).toHaveBeenCalled();
|
1948
|
+
const fetchSSEOptions = (fetchSSE as any).mock.calls[0][1];
|
1949
|
+
|
1950
|
+
expect(fetchSSEOptions.responseAnimation).toEqual({
|
1951
|
+
speed: 20,
|
1952
|
+
text: 'fadeIn',
|
1953
|
+
toolsCalling: 'fadeIn',
|
1954
|
+
});
|
1955
|
+
});
|
1956
|
+
});
|
1957
|
+
|
1958
|
+
describe('extendParams', () => {
|
1959
|
+
it('should set enabledContextCaching to false when model supports disableContextCaching and user enables it', async () => {
|
1960
|
+
const getChatCompletionSpy = vi.spyOn(chatService, 'getChatCompletion');
|
1961
|
+
const messages = [{ content: 'Test context caching', role: 'user' }] as ChatMessage[];
|
1962
|
+
|
1963
|
+
// Mock aiModelSelectors for extend params support
|
1964
|
+
vi.spyOn(aiModelSelectors, 'isModelHasExtendParams').mockReturnValue(() => true);
|
1965
|
+
vi.spyOn(aiModelSelectors, 'modelExtendParams').mockReturnValue(() => [
|
1966
|
+
'disableContextCaching',
|
1967
|
+
]);
|
1968
|
+
|
1969
|
+
// Mock agent chat config with context caching disabled
|
1970
|
+
vi.spyOn(agentChatConfigSelectors, 'currentChatConfig').mockReturnValue({
|
1971
|
+
disableContextCaching: true,
|
1972
|
+
searchMode: 'off',
|
1973
|
+
} as any);
|
1974
|
+
|
1975
|
+
await chatService.createAssistantMessage({
|
1976
|
+
messages,
|
1977
|
+
model: 'test-model',
|
1978
|
+
provider: 'test-provider',
|
1979
|
+
plugins: [],
|
1980
|
+
});
|
1981
|
+
|
1982
|
+
expect(getChatCompletionSpy).toHaveBeenCalledWith(
|
1983
|
+
expect.objectContaining({
|
1984
|
+
enabledContextCaching: false,
|
1985
|
+
}),
|
1986
|
+
undefined,
|
1987
|
+
);
|
1988
|
+
});
|
1989
|
+
|
1990
|
+
it('should not set enabledContextCaching when disableContextCaching is false', async () => {
|
1991
|
+
const getChatCompletionSpy = vi.spyOn(chatService, 'getChatCompletion');
|
1992
|
+
const messages = [{ content: 'Test context caching enabled', role: 'user' }] as ChatMessage[];
|
1993
|
+
|
1994
|
+
// Mock aiModelSelectors for extend params support
|
1995
|
+
vi.spyOn(aiModelSelectors, 'isModelHasExtendParams').mockReturnValue(() => true);
|
1996
|
+
vi.spyOn(aiModelSelectors, 'modelExtendParams').mockReturnValue(() => [
|
1997
|
+
'disableContextCaching',
|
1998
|
+
]);
|
1999
|
+
|
2000
|
+
// Mock agent chat config with context caching enabled (default)
|
2001
|
+
vi.spyOn(agentChatConfigSelectors, 'currentChatConfig').mockReturnValue({
|
2002
|
+
disableContextCaching: false,
|
2003
|
+
searchMode: 'off',
|
2004
|
+
} as any);
|
2005
|
+
|
2006
|
+
await chatService.createAssistantMessage({
|
2007
|
+
messages,
|
2008
|
+
model: 'test-model',
|
2009
|
+
provider: 'test-provider',
|
2010
|
+
plugins: [],
|
2011
|
+
});
|
2012
|
+
|
2013
|
+
// enabledContextCaching should not be present in the call
|
2014
|
+
const callArgs = getChatCompletionSpy.mock.calls[0][0];
|
2015
|
+
expect(callArgs).not.toHaveProperty('enabledContextCaching');
|
2016
|
+
});
|
2017
|
+
|
2018
|
+
it('should set reasoning_effort when model supports reasoningEffort and user configures it', async () => {
|
2019
|
+
const getChatCompletionSpy = vi.spyOn(chatService, 'getChatCompletion');
|
2020
|
+
const messages = [{ content: 'Test reasoning effort', role: 'user' }] as ChatMessage[];
|
2021
|
+
|
2022
|
+
// Mock aiModelSelectors for extend params support
|
2023
|
+
vi.spyOn(aiModelSelectors, 'isModelHasExtendParams').mockReturnValue(() => true);
|
2024
|
+
vi.spyOn(aiModelSelectors, 'modelExtendParams').mockReturnValue(() => ['reasoningEffort']);
|
2025
|
+
|
2026
|
+
// Mock agent chat config with reasoning effort set
|
2027
|
+
vi.spyOn(agentChatConfigSelectors, 'currentChatConfig').mockReturnValue({
|
2028
|
+
reasoningEffort: 'high',
|
2029
|
+
searchMode: 'off',
|
2030
|
+
} as any);
|
2031
|
+
|
2032
|
+
await chatService.createAssistantMessage({
|
2033
|
+
messages,
|
2034
|
+
model: 'test-model',
|
2035
|
+
provider: 'test-provider',
|
2036
|
+
plugins: [],
|
2037
|
+
});
|
2038
|
+
|
2039
|
+
expect(getChatCompletionSpy).toHaveBeenCalledWith(
|
2040
|
+
expect.objectContaining({
|
2041
|
+
reasoning_effort: 'high',
|
2042
|
+
}),
|
2043
|
+
undefined,
|
2044
|
+
);
|
2045
|
+
});
|
2046
|
+
|
2047
|
+
it('should set thinkingBudget when model supports thinkingBudget and user configures it', async () => {
|
2048
|
+
const getChatCompletionSpy = vi.spyOn(chatService, 'getChatCompletion');
|
2049
|
+
const messages = [{ content: 'Test thinking budget', role: 'user' }] as ChatMessage[];
|
2050
|
+
|
2051
|
+
// Mock aiModelSelectors for extend params support
|
2052
|
+
vi.spyOn(aiModelSelectors, 'isModelHasExtendParams').mockReturnValue(() => true);
|
2053
|
+
vi.spyOn(aiModelSelectors, 'modelExtendParams').mockReturnValue(() => ['thinkingBudget']);
|
2054
|
+
|
2055
|
+
// Mock agent chat config with thinking budget set
|
2056
|
+
vi.spyOn(agentChatConfigSelectors, 'currentChatConfig').mockReturnValue({
|
2057
|
+
thinkingBudget: 5000,
|
2058
|
+
searchMode: 'off',
|
2059
|
+
} as any);
|
2060
|
+
|
2061
|
+
await chatService.createAssistantMessage({
|
2062
|
+
messages,
|
2063
|
+
model: 'test-model',
|
2064
|
+
provider: 'test-provider',
|
2065
|
+
plugins: [],
|
2066
|
+
});
|
2067
|
+
|
2068
|
+
expect(getChatCompletionSpy).toHaveBeenCalledWith(
|
2069
|
+
expect.objectContaining({
|
2070
|
+
thinkingBudget: 5000,
|
2071
|
+
}),
|
2072
|
+
undefined,
|
2073
|
+
);
|
2074
|
+
});
|
2075
|
+
});
|
2076
|
+
});
|
2077
|
+
|
1142
2078
|
describe('AgentRuntimeOnClient', () => {
|
1143
2079
|
describe('initializeWithClientStore', () => {
|
1144
2080
|
describe('should initialize with options correctly', () => {
|