Skip to content

Commit

Permalink
🐛 fix: fix Google Gemini pro 1.5 and system role not take effect (lob…
Browse files Browse the repository at this point in the history
…ehub#1801)

* 🐛 fix: fix google gemini pro 1.5

* Update index.ts

* 1.5 support vision

* Update index.ts

* Update index.ts

* 🐛 fix: improve google model

* 🐛 fix: 修正 google 对话没有配对的问题

* 🐛 fix: 修正 vision 判断

* 🐛 fix: improve model card and region

* 🐛 fix: fix vision

* hide ultra model
  • Loading branch information
arvinxx authored Mar 29, 2024
1 parent 11013a7 commit 0a3e3f7
Show file tree
Hide file tree
Showing 6 changed files with 246 additions and 81 deletions.
2 changes: 1 addition & 1 deletion package.json
Original file line number Diff line number Diff line change
Expand Up @@ -84,7 +84,7 @@
"@aws-sdk/client-bedrock-runtime": "^3.525.0",
"@azure/openai": "^1.0.0-beta.11",
"@cfworker/json-schema": "^1",
"@google/generative-ai": "^0.2.0",
"@google/generative-ai": "^0.3.1",
"@icons-pack/react-simple-icons": "^9",
"@lobehub/chat-plugin-sdk": "latest",
"@lobehub/chat-plugins-gateway": "latest",
Expand Down
16 changes: 2 additions & 14 deletions src/app/api/chat/google/route.ts
Original file line number Diff line number Diff line change
Expand Up @@ -13,19 +13,7 @@ import { POST as UniverseRoute } from '../[provider]/route';
// so if you want to use with proxy, you need comment the code below
export const runtime = 'edge';

export const preferredRegion = [
'bom1',
'cle1',
'cpt1',
'gru1',
'hnd1',
'iad1',
'icn1',
'kix1',
'pdx1',
'sfo1',
'sin1',
'syd1',
];
// due to gemini-1.5-pro only can be used in us, so we need to set the preferred region only in US
export const preferredRegion = ['cle1', 'iad1', 'pdx1', 'sfo1'];

export const POST = async (req: Request) => UniverseRoute(req, { params: { provider: 'google' } });
75 changes: 68 additions & 7 deletions src/config/modelProviders/google.ts
Original file line number Diff line number Diff line change
Expand Up @@ -3,25 +3,86 @@ import { ModelProviderCard } from '@/types/llm';
const Google: ModelProviderCard = {
chatModels: [
{
displayName: 'Gemini Pro',
description: 'A legacy text-only model optimized for chat conversations',
displayName: 'PaLM 2 Chat (Legacy)',
hidden: true,
id: 'chat-bison-001',
maxOutput: 1024,
tokens: 5120,
},
{
description: 'A legacy model that understands text and generates text as an output',
displayName: 'PaLM 2 (Legacy)',
hidden: true,
id: 'text-bison-001',
maxOutput: 1024,
tokens: 9220,
},
{
description: 'The best model for scaling across a wide range of tasks',
displayName: 'Gemini 1.0 Pro',
id: 'gemini-pro',
tokens: 30_720,
maxOutput: 2048,
tokens: 32_768,
},
{
description: 'The best image understanding model to handle a broad range of applications',
displayName: 'Gemini 1.0 Pro Vision',
id: 'gemini-1.0-pro-vision-latest',
maxOutput: 4096,
tokens: 16_384,
vision: true,
},
{
displayName: 'Gemini Pro Vision',
description: 'The best image understanding model to handle a broad range of applications',
displayName: 'Gemini 1.0 Pro Vision',
hidden: true,
id: 'gemini-pro-vision',
tokens: 12_288,
maxOutput: 4096,
tokens: 16_384,
vision: true,
},
{
description: 'The best model for scaling across a wide range of tasks',
displayName: 'Gemini 1.0 Pro',
hidden: true,
id: '1.0-pro',
maxOutput: 2048,
tokens: 32_768,
},
{
description:
'The best model for scaling across a wide range of tasks. This is a stable model that supports tuning.',
displayName: 'Gemini 1.0 Pro 001 (Tuning)',
hidden: true,
id: 'gemini-1.0-pro-001',
maxOutput: 2048,
tokens: 32_768,
},
{
description:
'The best model for scaling across a wide range of tasks. This is the latest model.',
displayName: 'Gemini 1.0 Pro Latest',
hidden: true,
id: 'gemini-1.0-pro-latest',
maxOutput: 2048,
tokens: 32_768,
},
{
description: 'Mid-size multimodal model that supports up to 1 million tokens',
displayName: 'Gemini 1.5 Pro',
id: 'gemini-1.5-pro-latest',
tokens: 1_048_576,
maxOutput: 8192,
tokens: 1_056_768,
vision: true,
},
{
displayName: 'Gemini Ultra',
description: 'The most capable model for highly complex tasks',
displayName: 'Gemini 1.0 Ultra',
hidden: true,
id: 'gemini-ultra-latest',
tokens: 30_720,
maxOutput: 2048,
tokens: 32_768,
},
],
id: 'google',
Expand Down
82 changes: 74 additions & 8 deletions src/libs/agent-runtime/google/index.test.ts
Original file line number Diff line number Diff line change
@@ -1,5 +1,4 @@
// @vitest-environment edge-runtime
import { GenerateContentRequest, GenerateContentStreamResult, Part } from '@google/generative-ai';
import OpenAI from 'openai';
import { Mock, afterEach, beforeEach, describe, expect, it, vi } from 'vitest';

Expand Down Expand Up @@ -317,17 +316,55 @@ describe('LobeGoogleAI', () => {
});

describe('buildGoogleMessages', () => {
it('should use default text model when no images are included in messages', () => {
it('get default result with gemini-pro', () => {
const messages: OpenAIChatMessage[] = [{ content: 'Hello', role: 'user' }];

const contents = instance['buildGoogleMessages'](messages, 'gemini-pro');

expect(contents).toHaveLength(1);
expect(contents).toEqual([{ parts: [{ text: 'Hello' }], role: 'user' }]);
});

it('messages should end with user if using gemini-pro', () => {
const messages: OpenAIChatMessage[] = [
{ content: 'Hello', role: 'user' },
{ content: 'Hi', role: 'assistant' },
];
const model = 'text-davinci-003';

// 调用 buildGoogleMessages 方法
const { contents, model: usedModel } = instance['buildGoogleMessages'](messages, model);
const contents = instance['buildGoogleMessages'](messages, 'gemini-pro');

expect(contents).toHaveLength(3);
expect(contents).toEqual([
{ parts: [{ text: 'Hello' }], role: 'user' },
{ parts: [{ text: 'Hi' }], role: 'model' },
{ parts: [{ text: '' }], role: 'user' },
]);
});

it('should include system role if there is a system role prompt', () => {
const messages: OpenAIChatMessage[] = [
{ content: 'you are ChatGPT', role: 'system' },
{ content: 'Who are you', role: 'user' },
];

const contents = instance['buildGoogleMessages'](messages, 'gemini-pro');

expect(contents).toHaveLength(3);
expect(contents).toEqual([
{ parts: [{ text: 'you are ChatGPT' }], role: 'user' },
{ parts: [{ text: '' }], role: 'model' },
{ parts: [{ text: 'Who are you' }], role: 'user' },
]);
});

it('should not modify the length if model is gemini-1.5-pro', () => {
const messages: OpenAIChatMessage[] = [
{ content: 'Hello', role: 'user' },
{ content: 'Hi', role: 'assistant' },
];

const contents = instance['buildGoogleMessages'](messages, 'gemini-1.5-pro-latest');

expect(usedModel).toEqual('gemini-pro'); // 假设 'gemini-pro' 是默认文本模型
expect(contents).toHaveLength(2);
expect(contents).toEqual([
{ parts: [{ text: 'Hello' }], role: 'user' },
Expand All @@ -348,9 +385,8 @@ describe('LobeGoogleAI', () => {
const model = 'gemini-pro-vision';

// 调用 buildGoogleMessages 方法
const { contents, model: usedModel } = instance['buildGoogleMessages'](messages, model);
const contents = instance['buildGoogleMessages'](messages, model);

expect(usedModel).toEqual(model);
expect(contents).toHaveLength(1);
expect(contents).toEqual([
{
Expand All @@ -360,5 +396,35 @@ describe('LobeGoogleAI', () => {
]);
});
});

describe('convertModel', () => {
it('should use default text model when no images are included in messages', () => {
const messages: OpenAIChatMessage[] = [
{ content: 'Hello', role: 'user' },
{ content: 'Hi', role: 'assistant' },
];

// 调用 buildGoogleMessages 方法
const model = instance['convertModel']('gemini-pro-vision', messages);

expect(model).toEqual('gemini-pro'); // 假设 'gemini-pro' 是默认文本模型
});

it('should use specified model when images are included in messages', () => {
const messages: OpenAIChatMessage[] = [
{
content: [
{ type: 'text', text: 'Hello' },
{ type: 'image_url', image_url: { url: 'data:image/png;base64,...' } },
],
role: 'user',
},
];

const model = instance['convertModel']('gemini-pro-vision', messages);

expect(model).toEqual('gemini-pro-vision');
});
});
});
});
Loading

0 comments on commit 0a3e3f7

Please sign in to comment.