@lobehub/chat 1.53.4 → 1.53.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.
@@ -0,0 +1,200 @@
1
+ import { beforeEach, describe, expect, it, vi } from 'vitest';
2
+
3
+ import { AiInfraRepos } from '@/database/repositories/aiInfra';
4
+ import { serverDB } from '@/database/server';
5
+ import { AiProviderModel } from '@/database/server/models/aiProvider';
6
+ import { UserModel } from '@/database/server/models/user';
7
+ import { getServerGlobalConfig } from '@/server/globalConfig';
8
+ import { KeyVaultsGateKeeper } from '@/server/modules/KeyVaultsEncrypt';
9
+ import { AiProviderDetailItem, AiProviderRuntimeState } from '@/types/aiProvider';
10
+
11
+ import { aiProviderRouter } from './aiProvider';
12
+
13
+ vi.mock('@/server/globalConfig');
14
+ vi.mock('@/server/modules/KeyVaultsEncrypt');
15
+ vi.mock('@/database/repositories/aiInfra');
16
+ vi.mock('@/database/server/models/aiProvider');
17
+ vi.mock('@/database/server/models/user');
18
+
19
+ describe('aiProviderRouter', () => {
20
+ const mockUserId = 'test-user-id';
21
+ const mockProviderId = 'test-provider-id';
22
+ const mockEncrypt = vi.fn();
23
+ const mockDecrypt = vi.fn();
24
+
25
+ const mockGateKeeper = {
26
+ encrypt: mockEncrypt,
27
+ decrypt: mockDecrypt,
28
+ };
29
+
30
+ const mockProviderDetail: AiProviderDetailItem = {
31
+ id: mockProviderId,
32
+ name: 'Test Provider',
33
+ enabled: true,
34
+ description: 'Test Description',
35
+ source: 'custom',
36
+ settings: {},
37
+ };
38
+
39
+ const mockRuntimeState: AiProviderRuntimeState = {
40
+ enabledAiModels: [],
41
+ enabledAiProviders: [],
42
+ runtimeConfig: {},
43
+ };
44
+
45
+ beforeEach(() => {
46
+ vi.clearAllMocks();
47
+
48
+ vi.mocked(getServerGlobalConfig).mockReturnValue({
49
+ aiProvider: {},
50
+ } as any);
51
+
52
+ vi.mocked(KeyVaultsGateKeeper.initWithEnvKey).mockResolvedValue(mockGateKeeper as any);
53
+ });
54
+
55
+ const createMockContext = () => ({
56
+ userId: mockUserId,
57
+ });
58
+
59
+ describe('createAiProvider', () => {
60
+ it('should create a new AI provider', async () => {
61
+ const mockCreate = vi.fn().mockResolvedValue({ id: mockProviderId });
62
+ vi.mocked(AiProviderModel).prototype.create = mockCreate;
63
+
64
+ const caller = aiProviderRouter.createCaller(createMockContext());
65
+ const result = await caller.createAiProvider({
66
+ id: mockProviderId,
67
+ name: 'Test Provider',
68
+ source: 'custom',
69
+ });
70
+
71
+ expect(result).toBe(mockProviderId);
72
+ expect(mockCreate).toHaveBeenCalledWith(
73
+ expect.objectContaining({
74
+ id: mockProviderId,
75
+ name: 'Test Provider',
76
+ }),
77
+ mockGateKeeper.encrypt,
78
+ );
79
+ });
80
+ });
81
+
82
+ describe('getAiProviderById', () => {
83
+ it('should get AI provider by id', async () => {
84
+ const mockGetDetail = vi.fn().mockResolvedValue(mockProviderDetail);
85
+ vi.mocked(AiInfraRepos).prototype.getAiProviderDetail = mockGetDetail;
86
+
87
+ const caller = aiProviderRouter.createCaller(createMockContext());
88
+ const result = await caller.getAiProviderById({ id: mockProviderId });
89
+
90
+ expect(result).toEqual(mockProviderDetail);
91
+ expect(mockGetDetail).toHaveBeenCalledWith(
92
+ mockProviderId,
93
+ KeyVaultsGateKeeper.getUserKeyVaults,
94
+ );
95
+ });
96
+ });
97
+
98
+ describe('getAiProviderList', () => {
99
+ it('should get AI provider list', async () => {
100
+ const mockList = [mockProviderDetail];
101
+ const mockGetList = vi.fn().mockResolvedValue(mockList);
102
+ vi.mocked(AiInfraRepos).prototype.getAiProviderList = mockGetList;
103
+
104
+ const caller = aiProviderRouter.createCaller(createMockContext());
105
+ const result = await caller.getAiProviderList();
106
+
107
+ expect(result).toEqual(mockList);
108
+ expect(mockGetList).toHaveBeenCalled();
109
+ });
110
+ });
111
+
112
+ describe('getAiProviderRuntimeState', () => {
113
+ it('should get AI provider runtime state', async () => {
114
+ const mockGetState = vi.fn().mockResolvedValue(mockRuntimeState);
115
+ vi.mocked(AiInfraRepos).prototype.getAiProviderRuntimeState = mockGetState;
116
+
117
+ const caller = aiProviderRouter.createCaller(createMockContext());
118
+ const result = await caller.getAiProviderRuntimeState({});
119
+
120
+ expect(result).toEqual(mockRuntimeState);
121
+ expect(mockGetState).toHaveBeenCalledWith(KeyVaultsGateKeeper.getUserKeyVaults);
122
+ });
123
+ });
124
+
125
+ describe('removeAiProvider', () => {
126
+ it('should remove AI provider', async () => {
127
+ const mockDelete = vi.fn();
128
+ vi.mocked(AiProviderModel).prototype.delete = mockDelete;
129
+
130
+ const caller = aiProviderRouter.createCaller(createMockContext());
131
+ await caller.removeAiProvider({ id: mockProviderId });
132
+
133
+ expect(mockDelete).toHaveBeenCalledWith(mockProviderId);
134
+ });
135
+ });
136
+
137
+ describe('toggleProviderEnabled', () => {
138
+ it('should toggle provider enabled state', async () => {
139
+ const mockToggle = vi.fn();
140
+ vi.mocked(AiProviderModel).prototype.toggleProviderEnabled = mockToggle;
141
+
142
+ const caller = aiProviderRouter.createCaller(createMockContext());
143
+ await caller.toggleProviderEnabled({
144
+ id: mockProviderId,
145
+ enabled: true,
146
+ });
147
+
148
+ expect(mockToggle).toHaveBeenCalledWith(mockProviderId, true);
149
+ });
150
+ });
151
+
152
+ describe('updateAiProvider', () => {
153
+ it('should update AI provider', async () => {
154
+ const mockUpdate = vi.fn();
155
+ vi.mocked(AiProviderModel).prototype.update = mockUpdate;
156
+
157
+ const caller = aiProviderRouter.createCaller(createMockContext());
158
+ await caller.updateAiProvider({
159
+ id: mockProviderId,
160
+ value: { name: 'Updated Provider' },
161
+ });
162
+
163
+ expect(mockUpdate).toHaveBeenCalledWith(mockProviderId, {
164
+ name: 'Updated Provider',
165
+ });
166
+ });
167
+ });
168
+
169
+ describe('updateAiProviderConfig', () => {
170
+ it('should update AI provider config', async () => {
171
+ const mockUpdateConfig = vi.fn();
172
+ vi.mocked(AiProviderModel).prototype.updateConfig = mockUpdateConfig;
173
+
174
+ const caller = aiProviderRouter.createCaller(createMockContext());
175
+ await caller.updateAiProviderConfig({
176
+ id: mockProviderId,
177
+ value: { checkModel: 'gpt-4' },
178
+ });
179
+
180
+ expect(mockUpdateConfig).toHaveBeenCalledWith(
181
+ mockProviderId,
182
+ { checkModel: 'gpt-4' },
183
+ mockGateKeeper.encrypt,
184
+ );
185
+ });
186
+ });
187
+
188
+ describe('updateAiProviderOrder', () => {
189
+ it('should update AI provider order', async () => {
190
+ const mockUpdateOrder = vi.fn();
191
+ vi.mocked(AiProviderModel).prototype.updateOrder = mockUpdateOrder;
192
+
193
+ const sortMap = [{ id: mockProviderId, sort: 1 }];
194
+ const caller = aiProviderRouter.createCaller(createMockContext());
195
+ await caller.updateAiProviderOrder({ sortMap });
196
+
197
+ expect(mockUpdateOrder).toHaveBeenCalledWith(sortMap);
198
+ });
199
+ });
200
+ });