@lobehub/chat 1.5.2 โ†’ 1.5.4

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/CHANGELOG.md CHANGED
@@ -2,6 +2,57 @@
2
2
 
3
3
  # Changelog
4
4
 
5
+ ### [Version 1.5.4](https://github.com/lobehub/lobe-chat/compare/v1.5.3...v1.5.4)
6
+
7
+ <sup>Released on **2024-07-17**</sup>
8
+
9
+ #### ๐Ÿ› Bug Fixes
10
+
11
+ - **misc**: Fix delete session group.
12
+
13
+ <br/>
14
+
15
+ <details>
16
+ <summary><kbd>Improvements and Fixes</kbd></summary>
17
+
18
+ #### What's fixed
19
+
20
+ - **misc**: Fix delete session group, closes [#3245](https://github.com/lobehub/lobe-chat/issues/3245) ([8f7167d](https://github.com/lobehub/lobe-chat/commit/8f7167d))
21
+
22
+ </details>
23
+
24
+ <div align="right">
25
+
26
+ [![](https://img.shields.io/badge/-BACK_TO_TOP-151515?style=flat-square)](#readme-top)
27
+
28
+ </div>
29
+
30
+ ### [Version 1.5.3](https://github.com/lobehub/lobe-chat/compare/v1.5.2...v1.5.3)
31
+
32
+ <sup>Released on **2024-07-17**</sup>
33
+
34
+ #### ๐Ÿ› Bug Fixes
35
+
36
+ - **misc**: Fix `OpenAI` deployment restrictions, fix cant duplicate assistant.
37
+
38
+ <br/>
39
+
40
+ <details>
41
+ <summary><kbd>Improvements and Fixes</kbd></summary>
42
+
43
+ #### What's fixed
44
+
45
+ - **misc**: Fix `OpenAI` deployment restrictions, closes [#3206](https://github.com/lobehub/lobe-chat/issues/3206) ([3d7a35d](https://github.com/lobehub/lobe-chat/commit/3d7a35d))
46
+ - **misc**: Fix cant duplicate assistant, closes [#3242](https://github.com/lobehub/lobe-chat/issues/3242) ([0edc851](https://github.com/lobehub/lobe-chat/commit/0edc851))
47
+
48
+ </details>
49
+
50
+ <div align="right">
51
+
52
+ [![](https://img.shields.io/badge/-BACK_TO_TOP-151515?style=flat-square)](#readme-top)
53
+
54
+ </div>
55
+
5
56
  ### [Version 1.5.2](https://github.com/lobehub/lobe-chat/compare/v1.5.1...v1.5.2)
6
57
 
7
58
  <sup>Released on **2024-07-17**</sup>
package/package.json CHANGED
@@ -1,6 +1,6 @@
1
1
  {
2
2
  "name": "@lobehub/chat",
3
- "version": "1.5.2",
3
+ "version": "1.5.4",
4
4
  "description": "Lobe Chat - an open-source, high-performance chatbot framework that supports speech synthesis, multimodal, and extensible Function Call plugin system. Supports one-click free deployment of your private ChatGPT/LLM web application.",
5
5
  "keywords": [
6
6
  "framework",
@@ -1,4 +1,3 @@
1
- import { getPreferredRegion } from '@/app/api/config';
2
1
  import { createErrorResponse } from '@/app/api/errorResponse';
3
2
  import { AgentRuntime, ChatCompletionErrorPayload } from '@/libs/agent-runtime';
4
3
  import { ChatErrorType } from '@/types/fetch';
@@ -10,8 +9,6 @@ import { createTraceOptions, initAgentRuntimeWithUserPayload } from '../agentRun
10
9
 
11
10
  export const runtime = 'edge';
12
11
 
13
- export const preferredRegion = getPreferredRegion();
14
-
15
12
  export const POST = checkAuth(async (req: Request, { params, jwtPayload, createRuntime }) => {
16
13
  const { provider } = params;
17
14
 
@@ -1,6 +1,5 @@
1
1
  import { NextResponse } from 'next/server';
2
2
 
3
- import { getPreferredRegion } from '@/app/api/config';
4
3
  import { createErrorResponse } from '@/app/api/errorResponse';
5
4
  import { ChatCompletionErrorPayload, ModelProvider } from '@/libs/agent-runtime';
6
5
  import { ChatErrorType } from '@/types/fetch';
@@ -10,8 +9,6 @@ import { initAgentRuntimeWithUserPayload } from '../../agentRuntime';
10
9
 
11
10
  export const runtime = 'edge';
12
11
 
13
- export const preferredRegion = getPreferredRegion();
14
-
15
12
  const noNeedAPIKey = (provider: string) =>
16
13
  [ModelProvider.OpenRouter, ModelProvider.TogetherAI].includes(provider as any);
17
14
 
@@ -0,0 +1,28 @@
1
+ // @vitest-environment edge-runtime
2
+ import { describe, expect, it, vi } from 'vitest';
3
+
4
+ import { POST as UniverseRoute } from '../[provider]/route';
5
+ import { POST, preferredRegion, runtime } from './route';
6
+
7
+ // ๆจกๆ‹Ÿ '../[provider]/route'
8
+ vi.mock('../[provider]/route', () => ({
9
+ POST: vi.fn().mockResolvedValue('mocked response'),
10
+ }));
11
+
12
+ describe('Configuration tests', () => {
13
+ it('should have runtime set to "edge"', () => {
14
+ expect(runtime).toBe('edge');
15
+ });
16
+
17
+ it('should contain specific regions in preferredRegion', () => {
18
+ expect(preferredRegion).not.contain(['hkg1']);
19
+ });
20
+ });
21
+
22
+ describe('OpenAI POST function tests', () => {
23
+ it('should call UniverseRoute with correct parameters', async () => {
24
+ const mockRequest = new Request('https://example.com', { method: 'POST' });
25
+ await POST(mockRequest);
26
+ expect(UniverseRoute).toHaveBeenCalledWith(mockRequest, { params: { provider: 'openai' } });
27
+ });
28
+ });
@@ -0,0 +1,25 @@
1
+ import { POST as UniverseRoute } from '../[provider]/route';
2
+
3
+ export const runtime = 'edge';
4
+
5
+ export const preferredRegion = [
6
+ 'arn1',
7
+ 'bom1',
8
+ 'cdg1',
9
+ 'cle1',
10
+ 'cpt1',
11
+ 'dub1',
12
+ 'fra1',
13
+ 'gru1',
14
+ 'hnd1',
15
+ 'iad1',
16
+ 'icn1',
17
+ 'kix1',
18
+ 'lhr1',
19
+ 'pdx1',
20
+ 'sfo1',
21
+ 'sin1',
22
+ 'syd1',
23
+ ];
24
+
25
+ export const POST = async (req: Request) => UniverseRoute(req, { params: { provider: 'openai' } });
@@ -1,11 +1,29 @@
1
1
  import { OpenAISTTPayload } from '@lobehub/tts';
2
2
  import { createOpenaiAudioTranscriptions } from '@lobehub/tts/server';
3
3
 
4
- import { getPreferredRegion } from '../../config';
5
- import { createBizOpenAI } from '../createBizOpenAI';
4
+ import { createBizOpenAI } from '@/app/api/openai/createBizOpenAI';
6
5
 
7
6
  export const runtime = 'edge';
8
- export const preferredRegion = getPreferredRegion();
7
+
8
+ export const preferredRegion = [
9
+ 'arn1',
10
+ 'bom1',
11
+ 'cdg1',
12
+ 'cle1',
13
+ 'cpt1',
14
+ 'dub1',
15
+ 'fra1',
16
+ 'gru1',
17
+ 'hnd1',
18
+ 'iad1',
19
+ 'icn1',
20
+ 'kix1',
21
+ 'lhr1',
22
+ 'pdx1',
23
+ 'sfo1',
24
+ 'sin1',
25
+ 'syd1',
26
+ ];
9
27
 
10
28
  export const POST = async (req: Request) => {
11
29
  const formData = await req.formData();
@@ -1,11 +1,29 @@
1
1
  import { OpenAITTSPayload } from '@lobehub/tts';
2
2
  import { createOpenaiAudioSpeech } from '@lobehub/tts/server';
3
3
 
4
- import { getPreferredRegion } from '../../config';
5
- import { createBizOpenAI } from '../createBizOpenAI';
4
+ import { createBizOpenAI } from '@/app/api/openai/createBizOpenAI';
6
5
 
7
6
  export const runtime = 'edge';
8
- export const preferredRegion = getPreferredRegion();
7
+
8
+ export const preferredRegion = [
9
+ 'arn1',
10
+ 'bom1',
11
+ 'cdg1',
12
+ 'cle1',
13
+ 'cpt1',
14
+ 'dub1',
15
+ 'fra1',
16
+ 'gru1',
17
+ 'hnd1',
18
+ 'iad1',
19
+ 'icn1',
20
+ 'kix1',
21
+ 'lhr1',
22
+ 'pdx1',
23
+ 'sfo1',
24
+ 'sin1',
25
+ 'syd1',
26
+ ];
9
27
 
10
28
  export const POST = async (req: Request) => {
11
29
  const payload = (await req.json()) as OpenAITTSPayload;
@@ -1,17 +1,33 @@
1
1
  import { NextResponse } from 'next/server';
2
2
 
3
- import { getPreferredRegion } from '@/app/api/config';
3
+ import { initAgentRuntimeWithUserPayload } from '@/app/api/chat/agentRuntime';
4
4
  import { createErrorResponse } from '@/app/api/errorResponse';
5
+ import { checkAuth } from '@/app/api/middleware/auth';
5
6
  import { ChatCompletionErrorPayload } from '@/libs/agent-runtime';
6
7
  import { TextToImagePayload } from '@/libs/agent-runtime/types';
7
8
  import { ChatErrorType } from '@/types/fetch';
8
9
 
9
- import { initAgentRuntimeWithUserPayload } from '../../chat/agentRuntime';
10
- import { checkAuth } from '../../middleware/auth';
11
-
12
10
  export const runtime = 'edge';
13
11
 
14
- export const preferredRegion = getPreferredRegion();
12
+ export const preferredRegion = [
13
+ 'arn1',
14
+ 'bom1',
15
+ 'cdg1',
16
+ 'cle1',
17
+ 'cpt1',
18
+ 'dub1',
19
+ 'fra1',
20
+ 'gru1',
21
+ 'hnd1',
22
+ 'iad1',
23
+ 'icn1',
24
+ 'kix1',
25
+ 'lhr1',
26
+ 'pdx1',
27
+ 'sfo1',
28
+ 'sin1',
29
+ 'syd1',
30
+ ];
15
31
 
16
32
  // return NextResponse.json(
17
33
  // {
package/src/config/llm.ts CHANGED
@@ -3,12 +3,6 @@ import { createEnv } from '@t3-oss/env-nextjs';
3
3
  import { z } from 'zod';
4
4
 
5
5
  export const getLLMConfig = () => {
6
- // region format: iad1,sfo1
7
- let regions: string[] = [];
8
- if (process.env.OPENAI_FUNCTION_REGIONS) {
9
- regions = process.env.OPENAI_FUNCTION_REGIONS.split(',');
10
- }
11
-
12
6
  return createEnv({
13
7
  server: {
14
8
  API_KEY_SELECT_MODE: z.string().optional(),
@@ -17,7 +11,6 @@ export const getLLMConfig = () => {
17
11
  OPENAI_API_KEY: z.string().optional(),
18
12
  OPENAI_PROXY_URL: z.string().optional(),
19
13
  OPENAI_MODEL_LIST: z.string().optional(),
20
- OPENAI_FUNCTION_REGIONS: z.array(z.string()),
21
14
 
22
15
  ENABLED_AZURE_OPENAI: z.boolean(),
23
16
  AZURE_API_KEY: z.string().optional(),
@@ -99,7 +92,6 @@ export const getLLMConfig = () => {
99
92
  OPENAI_API_KEY: process.env.OPENAI_API_KEY,
100
93
  OPENAI_PROXY_URL: process.env.OPENAI_PROXY_URL,
101
94
  OPENAI_MODEL_LIST: process.env.OPENAI_MODEL_LIST,
102
- OPENAI_FUNCTION_REGIONS: regions as any,
103
95
 
104
96
  ENABLED_AZURE_OPENAI: !!process.env.AZURE_API_KEY,
105
97
  AZURE_API_KEY: process.env.AZURE_API_KEY,
@@ -0,0 +1,156 @@
1
+ // @vitest-environment node
2
+ import { eq } from 'drizzle-orm';
3
+ import { desc } from 'drizzle-orm/expressions';
4
+ import { afterEach, beforeEach, describe, expect, it, vi } from 'vitest';
5
+
6
+ import { getTestDBInstance } from '@/database/server/core/dbForTest';
7
+
8
+ import { sessionGroups, users } from '../../schemas/lobechat';
9
+ import { SessionGroupModel } from '../sessionGroup';
10
+
11
+ let serverDB = await getTestDBInstance();
12
+
13
+ vi.mock('@/database/server/core/db', async () => ({
14
+ get serverDB() {
15
+ return serverDB;
16
+ },
17
+ }));
18
+
19
+ const userId = 'session-group-model-test-user-id';
20
+ const sessionGroupModel = new SessionGroupModel(userId);
21
+
22
+ beforeEach(async () => {
23
+ await serverDB.delete(users);
24
+ await serverDB.insert(users).values([{ id: userId }, { id: 'user2' }]);
25
+ });
26
+
27
+ afterEach(async () => {
28
+ await serverDB.delete(users).where(eq(users.id, userId));
29
+ await serverDB.delete(sessionGroups).where(eq(sessionGroups.userId, userId));
30
+ });
31
+
32
+ describe('SessionGroupModel', () => {
33
+ describe('create', () => {
34
+ it('should create a new session group', async () => {
35
+ const params = {
36
+ name: 'Test Group',
37
+ sort: 1,
38
+ };
39
+
40
+ const result = await sessionGroupModel.create(params);
41
+ expect(result.id).toBeDefined();
42
+ expect(result).toMatchObject({ ...params, userId });
43
+
44
+ const group = await serverDB.query.sessionGroups.findFirst({
45
+ where: eq(sessionGroups.id, result.id),
46
+ });
47
+ expect(group).toMatchObject({ ...params, userId });
48
+ });
49
+ });
50
+ describe('delete', () => {
51
+ it('should delete a session group by id', async () => {
52
+ const { id } = await sessionGroupModel.create({ name: 'Test Group' });
53
+
54
+ await sessionGroupModel.delete(id);
55
+
56
+ const group = await serverDB.query.sessionGroups.findFirst({
57
+ where: eq(sessionGroups.id, id),
58
+ });
59
+ expect(group).toBeUndefined();
60
+ });
61
+ });
62
+ describe('deleteAll', () => {
63
+ it('should delete all session groups for the user', async () => {
64
+ await sessionGroupModel.create({ name: 'Test Group 1' });
65
+ await sessionGroupModel.create({ name: 'Test Group 2' });
66
+
67
+ await sessionGroupModel.deleteAll();
68
+
69
+ const userGroups = await serverDB.query.sessionGroups.findMany({
70
+ where: eq(sessionGroups.userId, userId),
71
+ });
72
+ expect(userGroups).toHaveLength(0);
73
+ });
74
+ it('should only delete session groups for the user, not others', async () => {
75
+ await sessionGroupModel.create({ name: 'Test Group 1' });
76
+ await sessionGroupModel.create({ name: 'Test Group 333' });
77
+
78
+ const anotherSessionGroupModel = new SessionGroupModel('user2');
79
+ await anotherSessionGroupModel.create({ name: 'Test Group 2' });
80
+
81
+ await sessionGroupModel.deleteAll();
82
+
83
+ const userGroups = await serverDB.query.sessionGroups.findMany({
84
+ where: eq(sessionGroups.userId, userId),
85
+ });
86
+ const total = await serverDB.query.sessionGroups.findMany();
87
+ expect(userGroups).toHaveLength(0);
88
+ expect(total).toHaveLength(1);
89
+ });
90
+ });
91
+
92
+ describe('query', () => {
93
+ it('should query session groups for the user', async () => {
94
+ await sessionGroupModel.create({ name: 'Test Group 1', sort: 2 });
95
+ await sessionGroupModel.create({ name: 'Test Group 2', sort: 1 });
96
+
97
+ const userGroups = await sessionGroupModel.query();
98
+ expect(userGroups).toHaveLength(2);
99
+ expect(userGroups[0].name).toBe('Test Group 2');
100
+ expect(userGroups[1].name).toBe('Test Group 1');
101
+ });
102
+ });
103
+
104
+ describe('findById', () => {
105
+ it('should find a session group by id', async () => {
106
+ const { id } = await sessionGroupModel.create({ name: 'Test Group' });
107
+
108
+ const group = await sessionGroupModel.findById(id);
109
+ expect(group).toMatchObject({
110
+ id,
111
+ name: 'Test Group',
112
+ userId,
113
+ });
114
+ });
115
+ });
116
+
117
+ describe('update', () => {
118
+ it('should update a session group', async () => {
119
+ const { id } = await sessionGroupModel.create({ name: 'Test Group' });
120
+
121
+ await sessionGroupModel.update(id, { name: 'Updated Test Group', sort: 3 });
122
+
123
+ const updatedGroup = await serverDB.query.sessionGroups.findFirst({
124
+ where: eq(sessionGroups.id, id),
125
+ });
126
+ expect(updatedGroup).toMatchObject({
127
+ id,
128
+ name: 'Updated Test Group',
129
+ sort: 3,
130
+ userId,
131
+ });
132
+ });
133
+ });
134
+
135
+ describe('updateOrder', () => {
136
+ it('should update order of session groups', async () => {
137
+ const group1 = await sessionGroupModel.create({ name: 'Test Group 1', sort: 1 });
138
+ const group2 = await sessionGroupModel.create({ name: 'Test Group 2', sort: 2 });
139
+
140
+ await sessionGroupModel.updateOrder([
141
+ { id: group1.id, sort: 3 },
142
+ { id: group2.id, sort: 4 },
143
+ ]);
144
+
145
+ const updatedGroup1 = await serverDB.query.sessionGroups.findFirst({
146
+ where: eq(sessionGroups.id, group1.id),
147
+ });
148
+ const updatedGroup2 = await serverDB.query.sessionGroups.findFirst({
149
+ where: eq(sessionGroups.id, group2.id),
150
+ });
151
+
152
+ expect(updatedGroup1?.sort).toBe(3);
153
+ expect(updatedGroup2?.sort).toBe(4);
154
+ });
155
+ });
156
+ });
@@ -173,8 +173,11 @@ export class SessionModel {
173
173
  const { agent, ...session } = result;
174
174
  const sessionId = this.genId();
175
175
 
176
+ // eslint-disable-next-line @typescript-eslint/no-unused-vars
177
+ const { id: _, slug: __, ...config } = agent;
178
+
176
179
  return this.create({
177
- config: agent,
180
+ config: config,
178
181
  id: sessionId,
179
182
  session: {
180
183
  ...session,
@@ -29,7 +29,7 @@ export class SessionGroupModel {
29
29
  };
30
30
 
31
31
  deleteAll = async () => {
32
- return serverDB.delete(sessionGroups);
32
+ return serverDB.delete(sessionGroups).where(eq(sessionGroups.userId, this.userId));
33
33
  };
34
34
 
35
35
  query = async () => {
@@ -1,43 +0,0 @@
1
- // @vitest-environment node
2
- import { describe, expect, it, vi } from 'vitest';
3
-
4
- import { getPreferredRegion } from './config';
5
-
6
- // Stub the global process object to safely mock environment variables
7
- vi.stubGlobal('process', {
8
- ...process, // Preserve the original process object
9
- env: { ...process.env }, // Clone the environment variables object for modification
10
- });
11
-
12
- describe('getPreferredRegion', () => {
13
- beforeEach(() => {
14
- // Reset environment variables before each test case
15
- vi.restoreAllMocks();
16
- });
17
-
18
- it('returns default value when get config error', () => {
19
- const originalProcess = global.process;
20
- const originalError = console.error;
21
- // @ts-ignore
22
- global.process = undefined;
23
- console.error = () => {};
24
-
25
- const preferredRegion = getPreferredRegion();
26
- expect(preferredRegion).toBe('auto');
27
-
28
- global.process = originalProcess;
29
- console.error = originalError;
30
- });
31
-
32
- it('return default value when preferredRegion is empty', () => {
33
- process.env.OPENAI_FUNCTION_REGIONS = '';
34
- const preferredRegion = getPreferredRegion();
35
- expect(preferredRegion).toBe('auto');
36
- });
37
-
38
- it('return correct list values when preferredRegion is correctly passed', () => {
39
- process.env.OPENAI_FUNCTION_REGIONS = 'ida1,sfo1';
40
- const preferredRegion = getPreferredRegion();
41
- expect(preferredRegion).toStrictEqual(['ida1', 'sfo1']);
42
- });
43
- });
@@ -1,14 +0,0 @@
1
- import { getLLMConfig } from '@/config/llm';
2
-
3
- export const getPreferredRegion = (region: string | string[] = 'auto') => {
4
- try {
5
- if (getLLMConfig().OPENAI_FUNCTION_REGIONS.length <= 0) {
6
- return region;
7
- }
8
-
9
- return getLLMConfig().OPENAI_FUNCTION_REGIONS;
10
- } catch (error) {
11
- console.error('get server config failed, error:', error);
12
- return region;
13
- }
14
- };