mirror of
https://github.com/ItzCrazyKns/Perplexica.git
synced 2025-12-24 20:48:14 +00:00
feat(providers): add lemonade
This commit is contained in:
@@ -5,6 +5,7 @@ import OllamaProvider from './ollama';
|
|||||||
import GeminiProvider from './gemini';
|
import GeminiProvider from './gemini';
|
||||||
import TransformersProvider from './transformers';
|
import TransformersProvider from './transformers';
|
||||||
import GroqProvider from './groq';
|
import GroqProvider from './groq';
|
||||||
|
import LemonadeProvider from './lemonade';
|
||||||
|
|
||||||
export const providers: Record<string, ProviderConstructor<any>> = {
|
export const providers: Record<string, ProviderConstructor<any>> = {
|
||||||
openai: OpenAIProvider,
|
openai: OpenAIProvider,
|
||||||
@@ -12,6 +13,7 @@ export const providers: Record<string, ProviderConstructor<any>> = {
|
|||||||
gemini: GeminiProvider,
|
gemini: GeminiProvider,
|
||||||
transformers: TransformersProvider,
|
transformers: TransformersProvider,
|
||||||
groq: GroqProvider,
|
groq: GroqProvider,
|
||||||
|
lemonade: LemonadeProvider
|
||||||
};
|
};
|
||||||
|
|
||||||
export const getModelProvidersUIConfigSection =
|
export const getModelProvidersUIConfigSection =
|
||||||
|
|||||||
152
src/lib/models/providers/lemonade/index.ts
Normal file
152
src/lib/models/providers/lemonade/index.ts
Normal file
@@ -0,0 +1,152 @@
|
|||||||
|
import { BaseChatModel } from '@langchain/core/language_models/chat_models';
|
||||||
|
import { ChatOpenAI, OpenAIEmbeddings } from '@langchain/openai';
|
||||||
|
import { Embeddings } from '@langchain/core/embeddings';
|
||||||
|
import { UIConfigField } from '@/lib/config/types';
|
||||||
|
import { getConfiguredModelProviderById } from '@/lib/config/serverRegistry';
|
||||||
|
import BaseModelProvider from '../../base/provider';
|
||||||
|
import { Model, ModelList, ProviderMetadata } from '../../types';
|
||||||
|
import BaseLLM from '../../base/llm';
|
||||||
|
import LemonadeLLM from './lemonadeLLM';
|
||||||
|
import BaseEmbedding from '../../base/embedding';
|
||||||
|
import LemonadeEmbedding from './lemonadeEmbedding';
|
||||||
|
|
||||||
|
interface LemonadeConfig {
|
||||||
|
baseURL: string;
|
||||||
|
apiKey?: string;
|
||||||
|
}
|
||||||
|
|
||||||
|
const providerConfigFields: UIConfigField[] = [
|
||||||
|
{
|
||||||
|
type: 'string',
|
||||||
|
name: 'Base URL',
|
||||||
|
key: 'baseURL',
|
||||||
|
description: 'The base URL for Lemonade API',
|
||||||
|
required: true,
|
||||||
|
placeholder: 'https://api.lemonade.ai/v1',
|
||||||
|
env: 'LEMONADE_BASE_URL',
|
||||||
|
scope: 'server',
|
||||||
|
},
|
||||||
|
{
|
||||||
|
type: 'password',
|
||||||
|
name: 'API Key',
|
||||||
|
key: 'apiKey',
|
||||||
|
description: 'Your Lemonade API key (optional)',
|
||||||
|
required: false,
|
||||||
|
placeholder: 'Lemonade API Key',
|
||||||
|
env: 'LEMONADE_API_KEY',
|
||||||
|
scope: 'server',
|
||||||
|
},
|
||||||
|
];
|
||||||
|
|
||||||
|
class LemonadeProvider extends BaseModelProvider<LemonadeConfig> {
|
||||||
|
constructor(id: string, name: string, config: LemonadeConfig) {
|
||||||
|
super(id, name, config);
|
||||||
|
}
|
||||||
|
|
||||||
|
async getDefaultModels(): Promise<ModelList> {
|
||||||
|
try {
|
||||||
|
const res = await fetch(`${this.config.baseURL}/models`, {
|
||||||
|
method: 'GET',
|
||||||
|
headers: {
|
||||||
|
'Content-Type': 'application/json',
|
||||||
|
...this.config.apiKey ? {'Authorization': `Bearer ${this.config.apiKey}`} : {}
|
||||||
|
},
|
||||||
|
});
|
||||||
|
|
||||||
|
const data = await res.json();
|
||||||
|
|
||||||
|
const models: Model[] = data.data.filter((m: any) => m.recipe === 'llamacpp').map((m: any) => {
|
||||||
|
return {
|
||||||
|
name: m.id,
|
||||||
|
key: m.id,
|
||||||
|
};
|
||||||
|
});
|
||||||
|
|
||||||
|
return {
|
||||||
|
embedding: models,
|
||||||
|
chat: models,
|
||||||
|
};
|
||||||
|
} catch (err) {
|
||||||
|
if (err instanceof TypeError) {
|
||||||
|
throw new Error(
|
||||||
|
'Error connecting to Lemonade API. Please ensure the base URL is correct and the service is available.',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
throw err;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
async getModelList(): Promise<ModelList> {
|
||||||
|
const defaultModels = await this.getDefaultModels();
|
||||||
|
const configProvider = getConfiguredModelProviderById(this.id)!;
|
||||||
|
|
||||||
|
return {
|
||||||
|
embedding: [
|
||||||
|
...defaultModels.embedding,
|
||||||
|
...configProvider.embeddingModels,
|
||||||
|
],
|
||||||
|
chat: [...defaultModels.chat, ...configProvider.chatModels],
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
async loadChatModel(key: string): Promise<BaseLLM<any>> {
|
||||||
|
const modelList = await this.getModelList();
|
||||||
|
|
||||||
|
const exists = modelList.chat.find((m) => m.key === key);
|
||||||
|
|
||||||
|
if (!exists) {
|
||||||
|
throw new Error(
|
||||||
|
'Error Loading Lemonade Chat Model. Invalid Model Selected',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new LemonadeLLM({
|
||||||
|
apiKey: this.config.apiKey || 'not-needed',
|
||||||
|
model: key,
|
||||||
|
baseURL: this.config.baseURL,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
async loadEmbeddingModel(key: string): Promise<BaseEmbedding<any>> {
|
||||||
|
const modelList = await this.getModelList();
|
||||||
|
const exists = modelList.embedding.find((m) => m.key === key);
|
||||||
|
|
||||||
|
if (!exists) {
|
||||||
|
throw new Error(
|
||||||
|
'Error Loading Lemonade Embedding Model. Invalid Model Selected.',
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
return new LemonadeEmbedding({
|
||||||
|
apiKey: this.config.apiKey || 'not-needed',
|
||||||
|
model: key,
|
||||||
|
baseURL: this.config.baseURL,
|
||||||
|
});
|
||||||
|
}
|
||||||
|
|
||||||
|
static parseAndValidate(raw: any): LemonadeConfig {
|
||||||
|
if (!raw || typeof raw !== 'object')
|
||||||
|
throw new Error('Invalid config provided. Expected object');
|
||||||
|
if (!raw.baseURL)
|
||||||
|
throw new Error('Invalid config provided. Base URL must be provided');
|
||||||
|
|
||||||
|
return {
|
||||||
|
baseURL: String(raw.baseURL),
|
||||||
|
apiKey: raw.apiKey ? String(raw.apiKey) : undefined,
|
||||||
|
};
|
||||||
|
}
|
||||||
|
|
||||||
|
static getProviderConfigFields(): UIConfigField[] {
|
||||||
|
return providerConfigFields;
|
||||||
|
}
|
||||||
|
|
||||||
|
static getProviderMetadata(): ProviderMetadata {
|
||||||
|
return {
|
||||||
|
key: 'lemonade',
|
||||||
|
name: 'Lemonade',
|
||||||
|
};
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
export default LemonadeProvider;
|
||||||
5
src/lib/models/providers/lemonade/lemonadeEmbedding.ts
Normal file
5
src/lib/models/providers/lemonade/lemonadeEmbedding.ts
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
import OpenAIEmbedding from "../openai/openaiEmbedding";
|
||||||
|
|
||||||
|
class LemonadeEmbedding extends OpenAIEmbedding {}
|
||||||
|
|
||||||
|
export default LemonadeEmbedding;
|
||||||
5
src/lib/models/providers/lemonade/lemonadeLLM.ts
Normal file
5
src/lib/models/providers/lemonade/lemonadeLLM.ts
Normal file
@@ -0,0 +1,5 @@
|
|||||||
|
import OpenAILLM from "../openai/openaiLLM";
|
||||||
|
|
||||||
|
class LemonadeLLM extends OpenAILLM {}
|
||||||
|
|
||||||
|
export default LemonadeLLM;
|
||||||
Reference in New Issue
Block a user