| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
| |
|
|
|
|
| import React, { useEffect, useState, useRef } from 'react';
|
| import { Button, Col, Form, Row, Spin } from '@douyinfe/semi-ui';
|
| import {
|
| compareObjects,
|
| API,
|
| showError,
|
| showSuccess,
|
| showWarning,
|
| verifyJSON,
|
| } from '../../../helpers';
|
| import { useTranslation } from 'react-i18next';
|
| import Text from '@douyinfe/semi-ui/lib/es/typography/text';
|
|
|
| const GEMINI_SETTING_EXAMPLE = {
|
| default: 'OFF',
|
| HARM_CATEGORY_CIVIC_INTEGRITY: 'BLOCK_NONE',
|
| };
|
|
|
| const GEMINI_VERSION_EXAMPLE = {
|
| default: 'v1beta',
|
| };
|
|
|
| const DEFAULT_GEMINI_INPUTS = {
|
| 'gemini.safety_settings': '',
|
| 'gemini.version_settings': '',
|
| 'gemini.supported_imagine_models': '',
|
| 'gemini.thinking_adapter_enabled': false,
|
| 'gemini.thinking_adapter_budget_tokens_percentage': 0.6,
|
| 'gemini.function_call_thought_signature_enabled': true,
|
| 'gemini.remove_function_response_id_enabled': true,
|
| };
|
|
|
| export default function SettingGeminiModel(props) {
|
| const { t } = useTranslation();
|
|
|
| const [loading, setLoading] = useState(false);
|
| const [inputs, setInputs] = useState(DEFAULT_GEMINI_INPUTS);
|
| const refForm = useRef();
|
| const [inputsRow, setInputsRow] = useState(DEFAULT_GEMINI_INPUTS);
|
|
|
| async function onSubmit() {
|
| await refForm.current
|
| .validate()
|
| .then(() => {
|
| const updateArray = compareObjects(inputs, inputsRow);
|
| if (!updateArray.length) return showWarning(t('你似乎并没有修改什么'));
|
| const requestQueue = updateArray.map((item) => {
|
| let value = String(inputs[item.key]);
|
| return API.put('/api/option/', {
|
| key: item.key,
|
| value,
|
| });
|
| });
|
| setLoading(true);
|
| Promise.all(requestQueue)
|
| .then((res) => {
|
| if (requestQueue.length === 1) {
|
| if (res.includes(undefined)) return;
|
| } else if (requestQueue.length > 1) {
|
| if (res.includes(undefined))
|
| return showError(t('部分保存失败,请重试'));
|
| }
|
| showSuccess(t('保存成功'));
|
| props.refresh();
|
| })
|
| .catch(() => {
|
| showError(t('保存失败,请重试'));
|
| })
|
| .finally(() => {
|
| setLoading(false);
|
| });
|
| })
|
| .catch((error) => {
|
| console.error('Validation failed:', error);
|
| showError(t('请检查输入'));
|
| });
|
| }
|
|
|
| useEffect(() => {
|
| const currentInputs = { ...DEFAULT_GEMINI_INPUTS };
|
| for (let key in props.options) {
|
| if (Object.prototype.hasOwnProperty.call(DEFAULT_GEMINI_INPUTS, key)) {
|
| currentInputs[key] = props.options[key];
|
| }
|
| }
|
| setInputs(currentInputs);
|
| setInputsRow(structuredClone(currentInputs));
|
| refForm.current.setValues(currentInputs);
|
| }, [props.options]);
|
|
|
| return (
|
| <>
|
| <Spin spinning={loading}>
|
| <Form
|
| values={inputs}
|
| getFormApi={(formAPI) => (refForm.current = formAPI)}
|
| style={{ marginBottom: 15 }}
|
| >
|
| <Form.Section text={t('Gemini设置')}>
|
| <Row>
|
| <Col xs={24} sm={12} md={8} lg={8} xl={8}>
|
| <Form.TextArea
|
| label={t('Gemini安全设置')}
|
| placeholder={
|
| t('为一个 JSON 文本,例如:') +
|
| '\n' +
|
| JSON.stringify(GEMINI_SETTING_EXAMPLE, null, 2)
|
| }
|
| field={'gemini.safety_settings'}
|
| extraText={t(
|
| 'default为默认设置,可单独设置每个分类的安全等级',
|
| )}
|
| autosize={{ minRows: 6, maxRows: 12 }}
|
| trigger='blur'
|
| stopValidateWithError
|
| rules={[
|
| {
|
| validator: (rule, value) => verifyJSON(value),
|
| message: t('不是合法的 JSON 字符串'),
|
| },
|
| ]}
|
| onChange={(value) =>
|
| setInputs({ ...inputs, 'gemini.safety_settings': value })
|
| }
|
| />
|
| </Col>
|
| </Row>
|
| <Row>
|
| <Col xs={24} sm={12} md={8} lg={8} xl={8}>
|
| <Form.TextArea
|
| label={t('Gemini版本设置')}
|
| placeholder={
|
| t('为一个 JSON 文本,例如:') +
|
| '\n' +
|
| JSON.stringify(GEMINI_VERSION_EXAMPLE, null, 2)
|
| }
|
| field={'gemini.version_settings'}
|
| extraText={t('default为默认设置,可单独设置每个模型的版本')}
|
| autosize={{ minRows: 6, maxRows: 12 }}
|
| trigger='blur'
|
| stopValidateWithError
|
| rules={[
|
| {
|
| validator: (rule, value) => verifyJSON(value),
|
| message: t('不是合法的 JSON 字符串'),
|
| },
|
| ]}
|
| onChange={(value) =>
|
| setInputs({ ...inputs, 'gemini.version_settings': value })
|
| }
|
| />
|
| </Col>
|
| </Row>
|
| <Row>
|
| <Col span={16}>
|
| <Form.Switch
|
| label={t('启用FunctionCall思维签名填充')}
|
| field={'gemini.function_call_thought_signature_enabled'}
|
| extraText={t(
|
| '仅为使用OpenAI格式的Gemini/Vertex渠道填充thoughtSignature',
|
| )}
|
| onChange={(value) =>
|
| setInputs({
|
| ...inputs,
|
| 'gemini.function_call_thought_signature_enabled': value,
|
| })
|
| }
|
| />
|
| </Col>
|
| </Row>
|
| <Row>
|
| <Col span={16}>
|
| <Form.Switch
|
| label={t('移除 functionResponse.id 字段')}
|
| field={'gemini.remove_function_response_id_enabled'}
|
| extraText={t(
|
| 'Vertex AI 不支持 functionResponse.id 字段,开启后将自动移除该字段',
|
| )}
|
| onChange={(value) =>
|
| setInputs({
|
| ...inputs,
|
| 'gemini.remove_function_response_id_enabled': value,
|
| })
|
| }
|
| />
|
| </Col>
|
| </Row>
|
| <Row>
|
| <Col xs={24} sm={12} md={8} lg={8} xl={8}>
|
| <Form.TextArea
|
| field={'gemini.supported_imagine_models'}
|
| label={t('支持的图像模型')}
|
| placeholder={
|
| t('例如:') +
|
| '\n' +
|
| JSON.stringify(
|
| ['gemini-2.0-flash-exp-image-generation'],
|
| null,
|
| 2,
|
| )
|
| }
|
| onChange={(value) =>
|
| setInputs({
|
| ...inputs,
|
| 'gemini.supported_imagine_models': value,
|
| })
|
| }
|
| trigger='blur'
|
| stopValidateWithError
|
| rules={[
|
| {
|
| validator: (rule, value) => verifyJSON(value),
|
| message: t('不是合法的 JSON 字符串'),
|
| },
|
| ]}
|
| />
|
| </Col>
|
| </Row>
|
| </Form.Section>
|
|
|
| <Form.Section text={t('Gemini思考适配设置')}>
|
| <Row>
|
| <Col span={16}>
|
| <Text>
|
| {t(
|
| '和Claude不同,默认情况下Gemini的思考模型会自动决定要不要思考,就算不开启适配模型也可以正常使用,' +
|
| '如果您需要计费,推荐设置无后缀模型价格按思考价格设置。' +
|
| '支持使用 gemini-2.5-pro-preview-06-05-thinking-128 格式来精确传递思考预算。',
|
| )}
|
| </Text>
|
| </Col>
|
| </Row>
|
| <Row>
|
| <Col span={16}>
|
| <Form.Switch
|
| label={t('启用Gemini思考后缀适配')}
|
| field={'gemini.thinking_adapter_enabled'}
|
| extraText={t(
|
| '适配 -thinking、-thinking-预算数字 和 -nothinking 后缀',
|
| )}
|
| onChange={(value) =>
|
| setInputs({
|
| ...inputs,
|
| 'gemini.thinking_adapter_enabled': value,
|
| })
|
| }
|
| />
|
| </Col>
|
| </Row>
|
| <Row>
|
| <Col span={16}>
|
| <Text>
|
| {t(
|
| 'Gemini思考适配 BudgetTokens = MaxTokens * BudgetTokens 百分比',
|
| )}
|
| </Text>
|
| </Col>
|
| </Row>
|
| <Row>
|
| <Col xs={24} sm={12} md={8} lg={8} xl={8}>
|
| <Form.InputNumber
|
| label={t('思考预算占比')}
|
| field={'gemini.thinking_adapter_budget_tokens_percentage'}
|
| initValue={''}
|
| extraText={t('0.002-1之间的小数')}
|
| min={0.002}
|
| max={1}
|
| onChange={(value) =>
|
| setInputs({
|
| ...inputs,
|
| 'gemini.thinking_adapter_budget_tokens_percentage': value,
|
| })
|
| }
|
| />
|
| </Col>
|
| </Row>
|
| </Form.Section>
|
|
|
| <Row>
|
| <Button size='default' onClick={onSubmit}>
|
| {t('保存')}
|
| </Button>
|
| </Row>
|
| </Form>
|
| </Spin>
|
| </>
|
| );
|
| }
|
|
|