@lobehub/chat
Version:
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.
227 lines (203 loc) • 8.15 kB
text/typescript
import { uniqBy } from 'lodash-es';
import { SWRResponse, mutate } from 'swr';
import { StateCreator } from 'zustand/vanilla';
import { DEFAULT_MODEL_PROVIDER_LIST } from '@/config/modelProviders';
import { isDeprecatedEdition, isDesktop, isUsePgliteDB } from '@/const/version';
import { useClientDataSWR } from '@/libs/swr';
import { aiProviderService } from '@/services/aiProvider';
import { AiInfraStore } from '@/store/aiInfra/store';
import { ModelAbilities } from '@/types/aiModel';
import {
AiProviderDetailItem,
AiProviderListItem,
AiProviderRuntimeState,
AiProviderSortMap,
AiProviderSourceEnum,
CreateAiProviderParams,
UpdateAiProviderConfigParams,
UpdateAiProviderParams,
} from '@/types/aiProvider';
enum AiProviderSwrKey {
fetchAiProviderItem = 'FETCH_AI_PROVIDER_ITEM',
fetchAiProviderList = 'FETCH_AI_PROVIDER',
fetchAiProviderRuntimeState = 'FETCH_AI_PROVIDER_RUNTIME_STATE',
}
export interface AiProviderAction {
createNewAiProvider: (params: CreateAiProviderParams) => Promise<void>;
deleteAiProvider: (id: string) => Promise<void>;
internal_toggleAiProviderConfigUpdating: (id: string, loading: boolean) => void;
internal_toggleAiProviderLoading: (id: string, loading: boolean) => void;
refreshAiProviderDetail: () => Promise<void>;
refreshAiProviderList: () => Promise<void>;
refreshAiProviderRuntimeState: () => Promise<void>;
removeAiProvider: (id: string) => Promise<void>;
toggleProviderEnabled: (id: string, enabled: boolean) => Promise<void>;
updateAiProvider: (id: string, value: UpdateAiProviderParams) => Promise<void>;
updateAiProviderConfig: (id: string, value: UpdateAiProviderConfigParams) => Promise<void>;
updateAiProviderSort: (items: AiProviderSortMap[]) => Promise<void>;
useFetchAiProviderItem: (id: string) => SWRResponse<AiProviderDetailItem | undefined>;
useFetchAiProviderList: (params?: { suspense?: boolean }) => SWRResponse<AiProviderListItem[]>;
/**
* fetch provider keyVaults and user enabled model list
* @param isLoginOnInit
*/
useFetchAiProviderRuntimeState: (
isLoginOnInit: boolean | undefined,
) => SWRResponse<AiProviderRuntimeState | undefined>;
}
export const createAiProviderSlice: StateCreator<
AiInfraStore,
[['zustand/devtools', never]],
[],
AiProviderAction
> = (set, get) => ({
createNewAiProvider: async (params) => {
await aiProviderService.createAiProvider({ ...params, source: AiProviderSourceEnum.Custom });
await get().refreshAiProviderList();
},
deleteAiProvider: async (id: string) => {
await aiProviderService.deleteAiProvider(id);
await get().refreshAiProviderList();
},
internal_toggleAiProviderConfigUpdating: (id, loading) => {
set(
(state) => {
if (loading)
return { aiProviderConfigUpdatingIds: [...state.aiProviderConfigUpdatingIds, id] };
return {
aiProviderConfigUpdatingIds: state.aiProviderConfigUpdatingIds.filter((i) => i !== id),
};
},
false,
'toggleAiProviderLoading',
);
},
internal_toggleAiProviderLoading: (id, loading) => {
set(
(state) => {
if (loading) return { aiProviderLoadingIds: [...state.aiProviderLoadingIds, id] };
return { aiProviderLoadingIds: state.aiProviderLoadingIds.filter((i) => i !== id) };
},
false,
'toggleAiProviderLoading',
);
},
refreshAiProviderDetail: async () => {
await mutate([AiProviderSwrKey.fetchAiProviderItem, get().activeAiProvider]);
await get().refreshAiProviderRuntimeState();
},
refreshAiProviderList: async () => {
await mutate(AiProviderSwrKey.fetchAiProviderList);
await get().refreshAiProviderRuntimeState();
},
refreshAiProviderRuntimeState: async () => {
await mutate([AiProviderSwrKey.fetchAiProviderRuntimeState, true]);
},
removeAiProvider: async (id) => {
await aiProviderService.deleteAiProvider(id);
await get().refreshAiProviderList();
},
toggleProviderEnabled: async (id: string, enabled: boolean) => {
get().internal_toggleAiProviderLoading(id, true);
await aiProviderService.toggleProviderEnabled(id, enabled);
await get().refreshAiProviderList();
get().internal_toggleAiProviderLoading(id, false);
},
updateAiProvider: async (id, value) => {
get().internal_toggleAiProviderLoading(id, true);
await aiProviderService.updateAiProvider(id, value);
await get().refreshAiProviderList();
await get().refreshAiProviderDetail();
get().internal_toggleAiProviderLoading(id, false);
},
updateAiProviderConfig: async (id, value) => {
get().internal_toggleAiProviderConfigUpdating(id, true);
await aiProviderService.updateAiProviderConfig(id, value);
await get().refreshAiProviderDetail();
get().internal_toggleAiProviderConfigUpdating(id, false);
},
updateAiProviderSort: async (items) => {
await aiProviderService.updateAiProviderOrder(items);
await get().refreshAiProviderList();
},
useFetchAiProviderItem: (id) =>
useClientDataSWR<AiProviderDetailItem | undefined>(
[AiProviderSwrKey.fetchAiProviderItem, id],
() => aiProviderService.getAiProviderById(id),
{
onSuccess: (data) => {
if (!data) return;
set({ activeAiProvider: id, aiProviderDetail: data }, false, 'useFetchAiProviderItem');
},
},
),
useFetchAiProviderList: () =>
useClientDataSWR<AiProviderListItem[]>(
AiProviderSwrKey.fetchAiProviderList,
() => aiProviderService.getAiProviderList(),
{
fallbackData: [],
onSuccess: (data) => {
if (!get().initAiProviderList) {
set(
{ aiProviderList: data, initAiProviderList: true },
false,
'useFetchAiProviderList/init',
);
return;
}
set({ aiProviderList: data }, false, 'useFetchAiProviderList/refresh');
},
},
),
useFetchAiProviderRuntimeState: (isLogin) =>
useClientDataSWR<AiProviderRuntimeState | undefined>(
!isDeprecatedEdition ? [AiProviderSwrKey.fetchAiProviderRuntimeState, isLogin] : null,
async ([, isLogin]) => {
if (isLogin) return aiProviderService.getAiProviderRuntimeState();
const { LOBE_DEFAULT_MODEL_LIST } = await import('@/config/aiModels');
return {
enabledAiModels: LOBE_DEFAULT_MODEL_LIST.filter((m) => m.enabled),
enabledAiProviders: DEFAULT_MODEL_PROVIDER_LIST.filter(
(provider) => provider.enabled,
).map((item) => ({ id: item.id, name: item.name, source: 'builtin' })),
runtimeConfig: {},
};
},
{
focusThrottleInterval: isDesktop || isUsePgliteDB ? 100 : undefined,
onSuccess: async (data) => {
if (!data) return;
const getModelListByType = (providerId: string, type: string) => {
const models = data.enabledAiModels
.filter((model) => model.providerId === providerId && model.type === type)
.map((model) => ({
abilities: (model.abilities || {}) as ModelAbilities,
contextWindowTokens: model.contextWindowTokens,
displayName: model.displayName ?? '',
id: model.id,
}));
return uniqBy(models, 'id');
};
// 3. 组装最终数据结构
const enabledChatModelList = data.enabledAiProviders.map((provider) => ({
...provider,
children: getModelListByType(provider.id, 'chat'),
name: provider.name || provider.id,
}));
const { LOBE_DEFAULT_MODEL_LIST } = await import('@/config/aiModels');
set(
{
aiProviderRuntimeConfig: data.runtimeConfig,
builtinAiModelList: LOBE_DEFAULT_MODEL_LIST,
enabledAiModels: data.enabledAiModels,
enabledAiProviders: data.enabledAiProviders,
enabledChatModelList,
},
false,
'useFetchAiProviderRuntimeState',
);
},
},
),
});