feat: allow user configure remote model from my model (#3348)
* feat: allow user configure remote model from my model * chore: fix linter * chore: fix linter * chore: add dedpendecies useCallback model item
This commit is contained in:
parent
fe8ed1f26d
commit
fdab8af057
@ -69,7 +69,7 @@ const TopPanel = () => {
|
|||||||
}, [activeThread?.id, setActiveThread, threads])
|
}, [activeThread?.id, setActiveThread, threads])
|
||||||
|
|
||||||
const onCreateThreadClicked = useCallback(async () => {
|
const onCreateThreadClicked = useCallback(async () => {
|
||||||
if (!assistants || assistants.length) {
|
if (!assistants || !assistants.length) {
|
||||||
toaster({
|
toaster({
|
||||||
title: 'No assistant available.',
|
title: 'No assistant available.',
|
||||||
description: `Could not create a new thread. Please add an assistant.`,
|
description: `Could not create a new thread. Please add an assistant.`,
|
||||||
|
|||||||
@ -129,7 +129,7 @@ const ModelSection: React.FC<Props> = ({
|
|||||||
{engineName}
|
{engineName}
|
||||||
</h6>
|
</h6>
|
||||||
</div>
|
</div>
|
||||||
<div className="flex items-center">
|
<div className="flex items-center gap-x-0.5">
|
||||||
{isRemoteEngine && (
|
{isRemoteEngine && (
|
||||||
<Button theme="icon" variant="outline" onClick={onSettingClick}>
|
<Button theme="icon" variant="outline" onClick={onSettingClick}>
|
||||||
{isEngineReady ? (
|
{isEngineReady ? (
|
||||||
@ -165,7 +165,6 @@ const ModelSection: React.FC<Props> = ({
|
|||||||
<ul>
|
<ul>
|
||||||
{models.map((model) => {
|
{models.map((model) => {
|
||||||
if (!showModel) return null
|
if (!showModel) return null
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<li
|
<li
|
||||||
key={model.model}
|
key={model.model}
|
||||||
|
|||||||
@ -18,7 +18,7 @@ const useGetModelsByEngine = () => {
|
|||||||
|
|
||||||
// TODO: this function needs to be clean up
|
// TODO: this function needs to be clean up
|
||||||
const getModelsByEngine = useCallback(
|
const getModelsByEngine = useCallback(
|
||||||
(engine: LlmEngine, searchText: string): Model[] => {
|
(engine: LlmEngine, searchText = ''): Model[] => {
|
||||||
if (LocalEngines.some((x) => x === engine)) {
|
if (LocalEngines.some((x) => x === engine)) {
|
||||||
return downloadedModels
|
return downloadedModels
|
||||||
.filter((m) => m.engine === engine)
|
.filter((m) => m.engine === engine)
|
||||||
|
|||||||
169
web/screens/Settings/MyModels/ModelGroup/index.tsx
Normal file
169
web/screens/Settings/MyModels/ModelGroup/index.tsx
Normal file
@ -0,0 +1,169 @@
|
|||||||
|
import React, { useCallback, useEffect, useState } from 'react'
|
||||||
|
|
||||||
|
import Image from 'next/image'
|
||||||
|
|
||||||
|
import {
|
||||||
|
EngineStatus,
|
||||||
|
LlmEngine,
|
||||||
|
LocalEngine,
|
||||||
|
Model,
|
||||||
|
RemoteEngine,
|
||||||
|
RemoteEngines,
|
||||||
|
} from '@janhq/core'
|
||||||
|
|
||||||
|
import { Button } from '@janhq/joi'
|
||||||
|
import { useAtom, useSetAtom } from 'jotai'
|
||||||
|
import {
|
||||||
|
SettingsIcon,
|
||||||
|
ChevronDownIcon,
|
||||||
|
ChevronUpIcon,
|
||||||
|
PlusIcon,
|
||||||
|
} from 'lucide-react'
|
||||||
|
|
||||||
|
import useEngineQuery from '@/hooks/useEngineQuery'
|
||||||
|
import useGetModelsByEngine from '@/hooks/useGetModelsByEngine'
|
||||||
|
|
||||||
|
import { getLogoByLocalEngine, getTitleByCategory } from '@/utils/model-engine'
|
||||||
|
|
||||||
|
import ModelItem from '../ModelItem'
|
||||||
|
|
||||||
|
import { showEngineListModelAtom } from '@/helpers/atoms/Model.atom'
|
||||||
|
import { setUpRemoteModelStageAtom } from '@/helpers/atoms/SetupRemoteModel.atom'
|
||||||
|
|
||||||
|
type Props = {
|
||||||
|
engine: LlmEngine
|
||||||
|
searchText: string
|
||||||
|
}
|
||||||
|
|
||||||
|
const ModelGroup: React.FC<Props> = ({ engine, searchText }) => {
|
||||||
|
const [models, setModels] = useState<Model[]>([])
|
||||||
|
const { getModelsByEngine } = useGetModelsByEngine()
|
||||||
|
const setUpRemoteModelStage = useSetAtom(setUpRemoteModelStageAtom)
|
||||||
|
const { data: engineData } = useEngineQuery()
|
||||||
|
|
||||||
|
const [showEngineListModel, setShowEngineListModel] = useAtom(
|
||||||
|
showEngineListModelAtom
|
||||||
|
)
|
||||||
|
|
||||||
|
const engineLogo: string | undefined = models.find(
|
||||||
|
(entry) => entry?.metadata?.logo != null
|
||||||
|
)?.metadata?.logo
|
||||||
|
|
||||||
|
const apiKeyUrl: string | undefined = models.find(
|
||||||
|
(entry) => entry?.metadata?.api_key_url != null
|
||||||
|
)?.metadata?.api_key_url
|
||||||
|
|
||||||
|
const onSettingClick = useCallback(() => {
|
||||||
|
setUpRemoteModelStage('SETUP_API_KEY', engine as unknown as RemoteEngine, {
|
||||||
|
logo: engineLogo,
|
||||||
|
api_key_url: apiKeyUrl,
|
||||||
|
})
|
||||||
|
}, [apiKeyUrl, engine, engineLogo, setUpRemoteModelStage])
|
||||||
|
|
||||||
|
const isEngineReady =
|
||||||
|
engineData?.find((e) => e.name === engine)?.status === EngineStatus.Ready
|
||||||
|
|
||||||
|
const getEngineStatusReady: LlmEngine[] | undefined = engineData
|
||||||
|
?.filter((e) => e.status === EngineStatus.Ready)
|
||||||
|
.map((x) => x.name as LlmEngine)
|
||||||
|
|
||||||
|
const showModel = showEngineListModel.includes(engine)
|
||||||
|
|
||||||
|
const onClickChevron = useCallback(() => {
|
||||||
|
if (showModel) {
|
||||||
|
setShowEngineListModel((prev) => prev.filter((item) => item !== engine))
|
||||||
|
} else {
|
||||||
|
setShowEngineListModel((prev) => [...prev, engine])
|
||||||
|
}
|
||||||
|
}, [engine, setShowEngineListModel, showModel])
|
||||||
|
|
||||||
|
useEffect(() => {
|
||||||
|
const matchedModels = getModelsByEngine(engine, searchText)
|
||||||
|
setModels(matchedModels)
|
||||||
|
setShowEngineListModel((prev) => [
|
||||||
|
...prev,
|
||||||
|
...(getEngineStatusReady as LlmEngine[]),
|
||||||
|
])
|
||||||
|
// eslint-disable-next-line react-hooks/exhaustive-deps
|
||||||
|
}, [getModelsByEngine, engine, searchText, setShowEngineListModel])
|
||||||
|
|
||||||
|
const engineName = getTitleByCategory(engine)
|
||||||
|
const localEngineLogo = getLogoByLocalEngine(engine as LocalEngine)
|
||||||
|
const isRemoteEngine = RemoteEngines.includes(engine as RemoteEngine)
|
||||||
|
|
||||||
|
if (models.length === 0) return null
|
||||||
|
|
||||||
|
return (
|
||||||
|
<div className="w-full py-3">
|
||||||
|
<div className="mb-2 flex justify-between pr-2">
|
||||||
|
<div
|
||||||
|
className="flex cursor-pointer items-center gap-2 pl-3"
|
||||||
|
onClick={onClickChevron}
|
||||||
|
>
|
||||||
|
{!isRemoteEngine && (
|
||||||
|
<Image
|
||||||
|
className="h-6 w-6 flex-shrink-0 rounded-full object-cover"
|
||||||
|
width={48}
|
||||||
|
height={48}
|
||||||
|
src={localEngineLogo as string}
|
||||||
|
alt="logo"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
{engineLogo && (
|
||||||
|
<Image
|
||||||
|
className="h-6 w-6 flex-shrink-0 rounded-full object-cover"
|
||||||
|
width={48}
|
||||||
|
height={48}
|
||||||
|
src={engineLogo}
|
||||||
|
alt="logo"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
<h6 className="pr-3 font-medium text-[hsla(var(--text-secondary))]">
|
||||||
|
{engineName}
|
||||||
|
</h6>
|
||||||
|
</div>
|
||||||
|
<div className="flex items-center gap-x-0.5">
|
||||||
|
{isRemoteEngine && (
|
||||||
|
<Button theme="icon" onClick={onSettingClick} variant="outline">
|
||||||
|
{isEngineReady ? (
|
||||||
|
<SettingsIcon
|
||||||
|
size={14}
|
||||||
|
className="text-[hsla(var(--text-secondary))]"
|
||||||
|
/>
|
||||||
|
) : (
|
||||||
|
<PlusIcon
|
||||||
|
size={14}
|
||||||
|
className="text-[hsla(var(--text-secondary))]"
|
||||||
|
/>
|
||||||
|
)}
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
{!showModel ? (
|
||||||
|
<Button theme="icon" onClick={onClickChevron}>
|
||||||
|
<ChevronDownIcon
|
||||||
|
size={14}
|
||||||
|
className="text-[hsla(var(--text-secondary))]"
|
||||||
|
/>
|
||||||
|
</Button>
|
||||||
|
) : (
|
||||||
|
<Button theme="icon" onClick={onClickChevron}>
|
||||||
|
<ChevronUpIcon
|
||||||
|
size={14}
|
||||||
|
className="text-[hsla(var(--text-secondary))]"
|
||||||
|
/>
|
||||||
|
</Button>
|
||||||
|
)}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
<div>
|
||||||
|
{models.map((model) => {
|
||||||
|
if (!showModel) return null
|
||||||
|
|
||||||
|
return <ModelItem model={model} key={model.id} />
|
||||||
|
})}
|
||||||
|
</div>
|
||||||
|
</div>
|
||||||
|
)
|
||||||
|
}
|
||||||
|
|
||||||
|
export default ModelGroup
|
||||||
@ -1,6 +1,12 @@
|
|||||||
import { memo, useCallback, useMemo, useState } from 'react'
|
import { memo, useCallback, useMemo, useState } from 'react'
|
||||||
|
|
||||||
import { LocalEngines, Model } from '@janhq/core'
|
import {
|
||||||
|
EngineStatus,
|
||||||
|
LocalEngines,
|
||||||
|
Model,
|
||||||
|
RemoteEngine,
|
||||||
|
RemoteEngines,
|
||||||
|
} from '@janhq/core'
|
||||||
import { Badge, Button, useClickOutside } from '@janhq/joi'
|
import { Badge, Button, useClickOutside } from '@janhq/joi'
|
||||||
|
|
||||||
import { useAtomValue, useSetAtom } from 'jotai'
|
import { useAtomValue, useSetAtom } from 'jotai'
|
||||||
@ -12,12 +18,19 @@ import {
|
|||||||
} from 'lucide-react'
|
} from 'lucide-react'
|
||||||
import { twMerge } from 'tailwind-merge'
|
import { twMerge } from 'tailwind-merge'
|
||||||
|
|
||||||
|
import { toaster } from '@/containers/Toast'
|
||||||
|
|
||||||
|
import useAssistantQuery from '@/hooks/useAssistantQuery'
|
||||||
|
import useEngineQuery from '@/hooks/useEngineQuery'
|
||||||
import useModelStart from '@/hooks/useModelStart'
|
import useModelStart from '@/hooks/useModelStart'
|
||||||
import useModelStop from '@/hooks/useModelStop'
|
import useModelStop from '@/hooks/useModelStop'
|
||||||
import useModels from '@/hooks/useModels'
|
import useModels from '@/hooks/useModels'
|
||||||
|
|
||||||
|
import useThreadCreateMutation from '@/hooks/useThreadCreateMutation'
|
||||||
|
|
||||||
import { showWarningMultipleModelModalAtom } from '@/screens/HubScreen2/components/WarningMultipleModelModal'
|
import { showWarningMultipleModelModalAtom } from '@/screens/HubScreen2/components/WarningMultipleModelModal'
|
||||||
|
|
||||||
|
import { MainViewState, mainViewStateAtom } from '@/helpers/atoms/App.atom'
|
||||||
import { activeModelsAtom } from '@/helpers/atoms/Model.atom'
|
import { activeModelsAtom } from '@/helpers/atoms/Model.atom'
|
||||||
|
|
||||||
type Props = {
|
type Props = {
|
||||||
@ -33,6 +46,14 @@ const ModelItem: React.FC<Props> = ({ model }) => {
|
|||||||
const stopModel = useModelStop()
|
const stopModel = useModelStop()
|
||||||
const [more, setMore] = useState(false)
|
const [more, setMore] = useState(false)
|
||||||
const { deleteModel } = useModels()
|
const { deleteModel } = useModels()
|
||||||
|
const { data: engineData } = useEngineQuery()
|
||||||
|
const createThreadMutation = useThreadCreateMutation()
|
||||||
|
const { data: assistants } = useAssistantQuery()
|
||||||
|
const setMainViewState = useSetAtom(mainViewStateAtom)
|
||||||
|
const isRemoteEngine = RemoteEngines.includes(model.engine as RemoteEngine)
|
||||||
|
const isEngineReady =
|
||||||
|
engineData?.find((e) => e.name === model.engine)?.status ===
|
||||||
|
EngineStatus.Ready
|
||||||
|
|
||||||
const [menu, setMenu] = useState<HTMLDivElement | null>(null)
|
const [menu, setMenu] = useState<HTMLDivElement | null>(null)
|
||||||
const [toggle, setToggle] = useState<HTMLDivElement | null>(null)
|
const [toggle, setToggle] = useState<HTMLDivElement | null>(null)
|
||||||
@ -82,6 +103,33 @@ const ModelItem: React.FC<Props> = ({ model }) => {
|
|||||||
(e) => model.engine != null && e === model.engine
|
(e) => model.engine != null && e === model.engine
|
||||||
)
|
)
|
||||||
|
|
||||||
|
const onClickCloudModel = useCallback(async () => {
|
||||||
|
if (!isRemoteEngine) return null
|
||||||
|
if (!model || !engineData) return
|
||||||
|
if (!assistants || !assistants.length) {
|
||||||
|
toaster({
|
||||||
|
title: 'No assistant available.',
|
||||||
|
description: `Could not create a new thread. Please add an assistant.`,
|
||||||
|
type: 'error',
|
||||||
|
})
|
||||||
|
return
|
||||||
|
}
|
||||||
|
|
||||||
|
await createThreadMutation.mutateAsync({
|
||||||
|
modelId: model.model,
|
||||||
|
assistant: assistants[0],
|
||||||
|
})
|
||||||
|
|
||||||
|
setMainViewState(MainViewState.Thread)
|
||||||
|
}, [
|
||||||
|
assistants,
|
||||||
|
createThreadMutation,
|
||||||
|
engineData,
|
||||||
|
isRemoteEngine,
|
||||||
|
model,
|
||||||
|
setMainViewState,
|
||||||
|
])
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="border border-b-0 border-[hsla(var(--app-border))] bg-[hsla(var(--tertiary-bg))] p-4 first:rounded-t-lg last:rounded-b-lg last:border-b">
|
<div className="border border-b-0 border-[hsla(var(--app-border))] bg-[hsla(var(--tertiary-bg))] p-4 first:rounded-t-lg last:rounded-b-lg last:border-b">
|
||||||
<div className="flex flex-col items-start justify-start gap-4 sm:flex-row sm:items-center sm:justify-between">
|
<div className="flex flex-col items-start justify-start gap-4 sm:flex-row sm:items-center sm:justify-between">
|
||||||
@ -89,11 +137,14 @@ const ModelItem: React.FC<Props> = ({ model }) => {
|
|||||||
<div className="flex w-full items-center justify-between">
|
<div className="flex w-full items-center justify-between">
|
||||||
<h6
|
<h6
|
||||||
className={twMerge(
|
className={twMerge(
|
||||||
'line-clamp-1 max-w-[200px] font-medium',
|
'line-clamp-1 font-medium text-[hsla(var(--text-secondary))]',
|
||||||
model.engine !== 'cortex.llamacpp' &&
|
isLocalModel && 'max-w-[200px]',
|
||||||
'max-w-none text-[hsla(var(--text-secondary))]'
|
isRemoteEngine && !isEngineReady
|
||||||
|
? 'cursor-not-allowed text-[hsla(var(--text-tertiary))]'
|
||||||
|
: 'cursor-pointer'
|
||||||
)}
|
)}
|
||||||
title={model.model}
|
title={model.model}
|
||||||
|
onClick={onClickCloudModel}
|
||||||
>
|
>
|
||||||
{model.model}
|
{model.model}
|
||||||
</h6>
|
</h6>
|
||||||
|
|||||||
@ -18,7 +18,7 @@ import useDropModelBinaries from '@/hooks/useDropModelBinaries'
|
|||||||
|
|
||||||
import { setImportModelStageAtom } from '@/hooks/useImportModel'
|
import { setImportModelStageAtom } from '@/hooks/useImportModel'
|
||||||
|
|
||||||
import ModelItem from './ModelItem'
|
import ModelGroup from './ModelGroup'
|
||||||
|
|
||||||
import { MainViewState, mainViewStateAtom } from '@/helpers/atoms/App.atom'
|
import { MainViewState, mainViewStateAtom } from '@/helpers/atoms/App.atom'
|
||||||
import { downloadedModelsAtom } from '@/helpers/atoms/Model.atom'
|
import { downloadedModelsAtom } from '@/helpers/atoms/Model.atom'
|
||||||
@ -123,27 +123,14 @@ const MyModels = () => {
|
|||||||
)}
|
)}
|
||||||
</>
|
</>
|
||||||
) : (
|
) : (
|
||||||
<div className="relative w-full">
|
<div className="relative mt-4 w-full">
|
||||||
{LlmEngines.map((engine) => {
|
{LlmEngines.map((engine) => {
|
||||||
const modelByEngine = filteredDownloadedModels.filter(
|
|
||||||
(x) => x.engine === engine
|
|
||||||
)
|
|
||||||
|
|
||||||
if (modelByEngine.length === 0) return null
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className="my-6" key={engine}>
|
<ModelGroup
|
||||||
<div className="flex flex-col items-start justify-start gap-2 sm:flex-row sm:items-center sm:justify-between">
|
engine={engine}
|
||||||
<h6 className="text-base font-semibold capitalize">
|
key={engine}
|
||||||
{engine}
|
searchText={searchText}
|
||||||
</h6>
|
/>
|
||||||
</div>
|
|
||||||
<div className="mt-2">
|
|
||||||
{modelByEngine.map((model) => (
|
|
||||||
<ModelItem key={model.model} model={model} />
|
|
||||||
))}
|
|
||||||
</div>
|
|
||||||
</div>
|
|
||||||
)
|
)
|
||||||
})}
|
})}
|
||||||
</div>
|
</div>
|
||||||
|
|||||||
@ -7,6 +7,12 @@ export const getTitleByCategory = (category: ModelHubCategory) => {
|
|||||||
switch (category) {
|
switch (category) {
|
||||||
case 'cortex.llamacpp':
|
case 'cortex.llamacpp':
|
||||||
return 'llama.cpp'
|
return 'llama.cpp'
|
||||||
|
case 'cortex.onnx':
|
||||||
|
return 'Onnx'
|
||||||
|
case 'cortex.tensorrt-llm':
|
||||||
|
return 'Tensorrt-llm'
|
||||||
|
case 'triton_trtllm':
|
||||||
|
return 'Triton-trtllm'
|
||||||
case 'BuiltInModels':
|
case 'BuiltInModels':
|
||||||
return 'Built-in Models'
|
return 'Built-in Models'
|
||||||
case 'HuggingFace':
|
case 'HuggingFace':
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user