2023-05-16 05:22:57 +00:00
|
|
|
import {Dropdown, Input, Label, Option, Select, Switch} from '@fluentui/react-components';
|
2023-05-06 12:17:39 +00:00
|
|
|
import {AddCircle20Regular, DataUsageSettings20Regular, Delete20Regular, Save20Regular} from '@fluentui/react-icons';
|
2023-05-05 15:23:34 +00:00
|
|
|
import React, {FC} from 'react';
|
2023-05-07 13:27:13 +00:00
|
|
|
import {Section} from '../components/Section';
|
|
|
|
import {Labeled} from '../components/Labeled';
|
|
|
|
import {ToolTipButton} from '../components/ToolTipButton';
|
2023-05-16 05:22:57 +00:00
|
|
|
import commonStore, {ApiParameters, Device, ModelParameters, Precision} from '../stores/commonStore';
|
2023-05-15 13:55:57 +00:00
|
|
|
import {observer} from 'mobx-react-lite';
|
|
|
|
import {toast} from 'react-toastify';
|
|
|
|
import {ValuedSlider} from '../components/ValuedSlider';
|
|
|
|
import {NumberInput} from '../components/NumberInput';
|
|
|
|
import {Page} from '../components/Page';
|
2023-05-15 14:51:52 +00:00
|
|
|
import {useNavigate} from 'react-router';
|
2023-05-16 05:22:57 +00:00
|
|
|
import {RunButton} from '../components/RunButton';
|
2023-05-17 03:39:00 +00:00
|
|
|
import {updateConfig} from '../apis';
|
2023-05-18 14:48:37 +00:00
|
|
|
import {ConvertModel, FileExists} from '../../wailsjs/go/backend_golang/App';
|
2023-05-17 13:20:41 +00:00
|
|
|
import manifest from '../../../manifest.json';
|
|
|
|
import {getStrategy, refreshLocalModels} from '../utils';
|
2023-05-18 12:48:53 +00:00
|
|
|
import {useTranslation} from 'react-i18next';
|
2023-05-15 13:55:57 +00:00
|
|
|
|
|
|
|
export const Configs: FC = observer(() => {
|
2023-05-18 12:48:53 +00:00
|
|
|
const {t} = useTranslation();
|
2023-05-15 13:55:57 +00:00
|
|
|
const [selectedIndex, setSelectedIndex] = React.useState(commonStore.currentModelConfigIndex);
|
|
|
|
const [selectedConfig, setSelectedConfig] = React.useState(commonStore.modelConfigs[selectedIndex]);
|
|
|
|
|
2023-05-15 14:51:52 +00:00
|
|
|
const navigate = useNavigate();
|
|
|
|
|
2023-05-15 13:55:57 +00:00
|
|
|
const updateSelectedIndex = (newIndex: number) => {
|
|
|
|
setSelectedIndex(newIndex);
|
|
|
|
setSelectedConfig(commonStore.modelConfigs[newIndex]);
|
|
|
|
|
|
|
|
// if you don't want to update the config used by the current startup in real time, comment out this line
|
|
|
|
commonStore.setCurrentConfigIndex(newIndex);
|
|
|
|
};
|
|
|
|
|
|
|
|
const setSelectedConfigName = (newName: string) => {
|
|
|
|
setSelectedConfig({...selectedConfig, name: newName});
|
|
|
|
};
|
|
|
|
|
|
|
|
const setSelectedConfigApiParams = (newParams: Partial<ApiParameters>) => {
|
|
|
|
setSelectedConfig({
|
|
|
|
...selectedConfig, apiParameters: {
|
|
|
|
...selectedConfig.apiParameters,
|
|
|
|
...newParams
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
|
|
|
|
|
|
|
const setSelectedConfigModelParams = (newParams: Partial<ModelParameters>) => {
|
|
|
|
setSelectedConfig({
|
|
|
|
...selectedConfig, modelParameters: {
|
|
|
|
...selectedConfig.modelParameters,
|
|
|
|
...newParams
|
|
|
|
}
|
|
|
|
});
|
|
|
|
};
|
2023-05-04 15:55:24 +00:00
|
|
|
|
2023-05-17 03:39:00 +00:00
|
|
|
const onClickSave = () => {
|
|
|
|
commonStore.setModelConfig(selectedIndex, selectedConfig);
|
|
|
|
updateConfig({
|
|
|
|
max_tokens: selectedConfig.apiParameters.maxResponseToken,
|
|
|
|
temperature: selectedConfig.apiParameters.temperature,
|
|
|
|
top_p: selectedConfig.apiParameters.topP,
|
|
|
|
presence_penalty: selectedConfig.apiParameters.presencePenalty,
|
|
|
|
frequency_penalty: selectedConfig.apiParameters.frequencyPenalty
|
|
|
|
});
|
2023-05-18 13:19:13 +00:00
|
|
|
toast(t('Config Saved'), {autoClose: 300, type: 'success'});
|
2023-05-17 03:39:00 +00:00
|
|
|
};
|
|
|
|
|
2023-05-05 15:23:34 +00:00
|
|
|
return (
|
2023-05-18 12:48:53 +00:00
|
|
|
<Page title={t('Configs')} content={
|
2023-05-15 13:55:57 +00:00
|
|
|
<div className="flex flex-col gap-2 overflow-hidden">
|
|
|
|
<div className="flex gap-2 items-center">
|
|
|
|
<Dropdown style={{minWidth: 0}} className="grow" value={commonStore.modelConfigs[selectedIndex].name}
|
|
|
|
selectedOptions={[selectedIndex.toString()]}
|
|
|
|
onOptionSelect={(_, data) => {
|
|
|
|
if (data.optionValue) {
|
|
|
|
updateSelectedIndex(Number(data.optionValue));
|
|
|
|
}
|
|
|
|
}}>
|
|
|
|
{commonStore.modelConfigs.map((config, index) =>
|
|
|
|
<Option key={index} value={index.toString()}>{config.name}</Option>
|
|
|
|
)}
|
|
|
|
</Dropdown>
|
2023-05-18 12:48:53 +00:00
|
|
|
<ToolTipButton desc={t('New Config')} icon={<AddCircle20Regular/>} onClick={() => {
|
2023-05-15 13:55:57 +00:00
|
|
|
commonStore.createModelConfig();
|
|
|
|
updateSelectedIndex(commonStore.modelConfigs.length - 1);
|
|
|
|
}}/>
|
2023-05-18 12:48:53 +00:00
|
|
|
<ToolTipButton desc={t('Delete Config')} icon={<Delete20Regular/>} onClick={() => {
|
2023-05-15 13:55:57 +00:00
|
|
|
commonStore.deleteModelConfig(selectedIndex);
|
|
|
|
updateSelectedIndex(Math.min(selectedIndex, commonStore.modelConfigs.length - 1));
|
|
|
|
}}/>
|
2023-05-18 12:48:53 +00:00
|
|
|
<ToolTipButton desc={t('Save Config')} icon={<Save20Regular/>} onClick={onClickSave}/>
|
2023-05-15 13:55:57 +00:00
|
|
|
</div>
|
|
|
|
<div className="flex items-center gap-4">
|
2023-05-18 12:48:53 +00:00
|
|
|
<Label>{t('Config Name')}</Label>
|
2023-05-15 13:55:57 +00:00
|
|
|
<Input className="grow" value={selectedConfig.name} onChange={(e, data) => {
|
|
|
|
setSelectedConfigName(data.value);
|
|
|
|
}}/>
|
|
|
|
</div>
|
|
|
|
<div className="flex flex-col gap-2 overflow-y-hidden">
|
|
|
|
<Section
|
2023-05-18 12:48:53 +00:00
|
|
|
title={t('Default API Parameters')}
|
|
|
|
desc={t('Hover your mouse over the text to view a detailed description. Settings marked with * will take effect immediately after being saved.')}
|
2023-05-15 13:55:57 +00:00
|
|
|
content={
|
|
|
|
<div className="grid grid-cols-1 sm:grid-cols-2 gap-2">
|
2023-05-18 12:48:53 +00:00
|
|
|
<Labeled label={t('API Port')} desc={`127.0.0.1:${selectedConfig.apiParameters.apiPort}`} content={
|
2023-05-15 13:55:57 +00:00
|
|
|
<NumberInput value={selectedConfig.apiParameters.apiPort} min={1} max={65535} step={1}
|
|
|
|
onChange={(e, data) => {
|
|
|
|
setSelectedConfigApiParams({
|
|
|
|
apiPort: data.value
|
|
|
|
});
|
|
|
|
}}/>
|
|
|
|
}/>
|
2023-05-18 12:48:53 +00:00
|
|
|
<Labeled label={t('Max Response Token *')} content={
|
2023-05-15 13:55:57 +00:00
|
|
|
<ValuedSlider value={selectedConfig.apiParameters.maxResponseToken} min={100} max={8100} step={400}
|
|
|
|
input
|
|
|
|
onChange={(e, data) => {
|
|
|
|
setSelectedConfigApiParams({
|
|
|
|
maxResponseToken: data.value
|
|
|
|
});
|
|
|
|
}}/>
|
|
|
|
}/>
|
2023-05-18 12:48:53 +00:00
|
|
|
<Labeled label={t('Temperature *')} content={
|
2023-05-15 13:55:57 +00:00
|
|
|
<ValuedSlider value={selectedConfig.apiParameters.temperature} min={0} max={2} step={0.1} input
|
|
|
|
onChange={(e, data) => {
|
|
|
|
setSelectedConfigApiParams({
|
|
|
|
temperature: data.value
|
|
|
|
});
|
|
|
|
}}/>
|
|
|
|
}/>
|
2023-05-18 12:48:53 +00:00
|
|
|
<Labeled label={t('Top_P *')} content={
|
2023-05-15 13:55:57 +00:00
|
|
|
<ValuedSlider value={selectedConfig.apiParameters.topP} min={0} max={1} step={0.1} input
|
|
|
|
onChange={(e, data) => {
|
|
|
|
setSelectedConfigApiParams({
|
|
|
|
topP: data.value
|
|
|
|
});
|
|
|
|
}}/>
|
|
|
|
}/>
|
2023-05-18 12:48:53 +00:00
|
|
|
<Labeled label={t('Presence Penalty *')} content={
|
2023-05-15 13:55:57 +00:00
|
|
|
<ValuedSlider value={selectedConfig.apiParameters.presencePenalty} min={-2} max={2} step={0.1} input
|
|
|
|
onChange={(e, data) => {
|
|
|
|
setSelectedConfigApiParams({
|
|
|
|
presencePenalty: data.value
|
|
|
|
});
|
|
|
|
}}/>
|
|
|
|
}/>
|
2023-05-18 12:48:53 +00:00
|
|
|
<Labeled label={t('Frequency Penalty *')} content={
|
2023-05-17 03:39:00 +00:00
|
|
|
<ValuedSlider value={selectedConfig.apiParameters.frequencyPenalty} min={-2} max={2} step={0.1} input
|
2023-05-15 13:55:57 +00:00
|
|
|
onChange={(e, data) => {
|
|
|
|
setSelectedConfigApiParams({
|
2023-05-17 03:39:00 +00:00
|
|
|
frequencyPenalty: data.value
|
2023-05-15 13:55:57 +00:00
|
|
|
});
|
|
|
|
}}/>
|
|
|
|
}/>
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
<Section
|
2023-05-18 12:48:53 +00:00
|
|
|
title={t('Model Parameters')}
|
2023-05-15 13:55:57 +00:00
|
|
|
content={
|
|
|
|
<div className="grid grid-cols-1 sm:grid-cols-2 gap-2">
|
2023-05-18 12:48:53 +00:00
|
|
|
<Labeled label={t('Model')} content={
|
2023-05-15 13:55:57 +00:00
|
|
|
<div className="flex gap-2 grow">
|
2023-05-15 14:51:52 +00:00
|
|
|
<Select style={{minWidth: 0}} className="grow"
|
|
|
|
value={selectedConfig.modelParameters.modelName}
|
|
|
|
onChange={(e, data) => {
|
|
|
|
setSelectedConfigModelParams({
|
|
|
|
modelName: data.value
|
|
|
|
});
|
|
|
|
}}>
|
2023-05-15 13:55:57 +00:00
|
|
|
{commonStore.modelSourceList.map((modelItem, index) =>
|
2023-05-15 14:51:52 +00:00
|
|
|
modelItem.isLocal && <option key={index} value={modelItem.name}>{modelItem.name}</option>
|
2023-05-15 13:55:57 +00:00
|
|
|
)}
|
2023-05-15 14:51:52 +00:00
|
|
|
</Select>
|
2023-05-18 12:48:53 +00:00
|
|
|
<ToolTipButton desc={t('Manage Models')} icon={<DataUsageSettings20Regular/>} onClick={() => {
|
2023-05-15 14:51:52 +00:00
|
|
|
navigate({pathname: '/models'});
|
|
|
|
}}/>
|
2023-05-15 13:55:57 +00:00
|
|
|
</div>
|
|
|
|
}/>
|
2023-05-18 14:48:37 +00:00
|
|
|
<ToolTipButton text={t('Convert')} desc={t('Convert model with these configs')} onClick={async () => {
|
2023-05-17 13:20:41 +00:00
|
|
|
const modelPath = `${manifest.localModelDir}/${selectedConfig.modelParameters.modelName}`;
|
2023-05-18 14:48:37 +00:00
|
|
|
if (await FileExists(modelPath)) {
|
|
|
|
const strategy = getStrategy(selectedConfig);
|
|
|
|
const newModelPath = modelPath + '-' + strategy.replace(/[> *+]/g, '-');
|
|
|
|
toast(t('Start Converting'), {autoClose: 1000, type: 'info'});
|
|
|
|
ConvertModel(modelPath, strategy, newModelPath).then(() => {
|
|
|
|
toast(`${t('Convert Success')} - ${newModelPath}`, {type: 'success'});
|
|
|
|
refreshLocalModels({models: commonStore.modelSourceList});
|
|
|
|
}).catch(e => {
|
|
|
|
toast(`${t('Convert Failed')} - ${e}`, {type: 'error'});
|
|
|
|
});
|
|
|
|
} else {
|
|
|
|
toast(`${t('Model Not Found')} - ${modelPath}`, {type: 'error'});
|
|
|
|
}
|
2023-05-17 13:20:41 +00:00
|
|
|
}}/>
|
2023-05-18 12:48:53 +00:00
|
|
|
<Labeled label={t('Device')} content={
|
2023-05-16 05:22:57 +00:00
|
|
|
<Dropdown style={{minWidth: 0}} className="grow" value={selectedConfig.modelParameters.device}
|
|
|
|
selectedOptions={[selectedConfig.modelParameters.device]}
|
|
|
|
onOptionSelect={(_, data) => {
|
|
|
|
if (data.optionText) {
|
|
|
|
setSelectedConfigModelParams({
|
|
|
|
device: data.optionText as Device
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}}>
|
2023-05-15 13:55:57 +00:00
|
|
|
<Option>CPU</Option>
|
2023-05-15 14:51:52 +00:00
|
|
|
<Option>CUDA</Option>
|
2023-05-15 13:55:57 +00:00
|
|
|
</Dropdown>
|
|
|
|
}/>
|
2023-05-18 12:48:53 +00:00
|
|
|
<Labeled label={t('Precision')} content={
|
2023-05-16 05:22:57 +00:00
|
|
|
<Dropdown style={{minWidth: 0}} className="grow" value={selectedConfig.modelParameters.precision}
|
|
|
|
selectedOptions={[selectedConfig.modelParameters.precision]}
|
|
|
|
onOptionSelect={(_, data) => {
|
|
|
|
if (data.optionText) {
|
|
|
|
setSelectedConfigModelParams({
|
|
|
|
precision: data.optionText as Precision
|
|
|
|
});
|
|
|
|
}
|
|
|
|
}}>
|
2023-05-15 13:55:57 +00:00
|
|
|
<Option>fp16</Option>
|
|
|
|
<Option>int8</Option>
|
|
|
|
<Option>fp32</Option>
|
|
|
|
</Dropdown>
|
|
|
|
}/>
|
2023-05-18 12:48:53 +00:00
|
|
|
<Labeled label={t('Stored Layers')} content={
|
2023-05-17 03:39:00 +00:00
|
|
|
<ValuedSlider value={selectedConfig.modelParameters.storedLayers} min={0}
|
|
|
|
max={selectedConfig.modelParameters.maxStoredLayers} step={1} input
|
2023-05-16 05:22:57 +00:00
|
|
|
onChange={(e, data) => {
|
|
|
|
setSelectedConfigModelParams({
|
2023-05-17 03:39:00 +00:00
|
|
|
storedLayers: data.value
|
2023-05-16 05:22:57 +00:00
|
|
|
});
|
|
|
|
}}/>
|
2023-05-15 13:55:57 +00:00
|
|
|
}/>
|
2023-05-18 12:48:53 +00:00
|
|
|
<Labeled label={t('Enable High Precision For Last Layer')} content={
|
2023-05-16 05:22:57 +00:00
|
|
|
<Switch checked={selectedConfig.modelParameters.enableHighPrecisionForLastLayer}
|
|
|
|
onChange={(e, data) => {
|
|
|
|
setSelectedConfigModelParams({
|
|
|
|
enableHighPrecisionForLastLayer: data.checked
|
|
|
|
});
|
|
|
|
}}/>
|
2023-05-15 13:55:57 +00:00
|
|
|
}/>
|
|
|
|
</div>
|
|
|
|
}
|
|
|
|
/>
|
|
|
|
</div>
|
|
|
|
<div className="flex flex-row-reverse sm:fixed bottom-2 right-2">
|
2023-05-17 03:39:00 +00:00
|
|
|
<RunButton onClickRun={onClickSave}/>
|
2023-05-15 13:55:57 +00:00
|
|
|
</div>
|
2023-05-05 15:23:34 +00:00
|
|
|
</div>
|
2023-05-15 13:55:57 +00:00
|
|
|
}/>
|
2023-05-05 15:23:34 +00:00
|
|
|
);
|
2023-05-15 13:55:57 +00:00
|
|
|
});
|