mirror of
https://github.com/QuantumNous/new-api.git
synced 2026-03-30 04:40:59 +00:00
Merge pull request #2663 from seefs001/feature/retry-status-code
feat: customizable automatic retry status codes
This commit is contained in:
71
web/src/components/settings/HttpStatusCodeRulesInput.jsx
Normal file
71
web/src/components/settings/HttpStatusCodeRulesInput.jsx
Normal file
@@ -0,0 +1,71 @@
|
||||
/*
|
||||
Copyright (C) 2025 QuantumNous
|
||||
|
||||
This program is free software: you can redistribute it and/or modify
|
||||
it under the terms of the GNU Affero General Public License as
|
||||
published by the Free Software Foundation, either version 3 of the
|
||||
License, or (at your option) any later version.
|
||||
|
||||
This program is distributed in the hope that it will be useful,
|
||||
but WITHOUT ANY WARRANTY; without even the implied warranty of
|
||||
MERCHANTABILITY or FITNESS FOR A PARTICULAR PURPOSE. See the
|
||||
GNU Affero General Public License for more details.
|
||||
|
||||
You should have received a copy of the GNU Affero General Public License
|
||||
along with this program. If not, see <https://www.gnu.org/licenses/>.
|
||||
|
||||
For commercial licensing, please contact support@quantumnous.com
|
||||
*/
|
||||
|
||||
import React from 'react';
|
||||
import { Form, Tag, Typography } from '@douyinfe/semi-ui';
|
||||
|
||||
export default function HttpStatusCodeRulesInput(props) {
|
||||
const { Text } = Typography;
|
||||
const {
|
||||
label,
|
||||
field,
|
||||
placeholder,
|
||||
extraText,
|
||||
onChange,
|
||||
parsed,
|
||||
invalidText,
|
||||
} = props;
|
||||
|
||||
return (
|
||||
<>
|
||||
<Form.Input
|
||||
label={label}
|
||||
placeholder={placeholder}
|
||||
extraText={extraText}
|
||||
field={field}
|
||||
onChange={onChange}
|
||||
/>
|
||||
{parsed?.ok && parsed.tokens?.length > 0 && (
|
||||
<div
|
||||
style={{
|
||||
display: 'flex',
|
||||
flexWrap: 'wrap',
|
||||
gap: 8,
|
||||
marginTop: 8,
|
||||
}}
|
||||
>
|
||||
{parsed.tokens.map((token) => (
|
||||
<Tag key={token} size='small'>
|
||||
{token}
|
||||
</Tag>
|
||||
))}
|
||||
</div>
|
||||
)}
|
||||
{!parsed?.ok && (
|
||||
<Text type='danger' style={{ display: 'block', marginTop: 8 }}>
|
||||
{invalidText}
|
||||
{parsed?.invalidTokens && parsed.invalidTokens.length > 0
|
||||
? `: ${parsed.invalidTokens.join(', ')}`
|
||||
: ''}
|
||||
</Text>
|
||||
)}
|
||||
</>
|
||||
);
|
||||
}
|
||||
|
||||
@@ -71,6 +71,7 @@ const OperationSetting = () => {
|
||||
AutomaticEnableChannelEnabled: false,
|
||||
AutomaticDisableKeywords: '',
|
||||
AutomaticDisableStatusCodes: '401',
|
||||
AutomaticRetryStatusCodes: '100-199,300-399,401-407,409-499,500-503,505-523,525-599',
|
||||
'monitor_setting.auto_test_channel_enabled': false,
|
||||
'monitor_setting.auto_test_channel_minutes': 10 /* 签到设置 */,
|
||||
'checkin_setting.enabled': false,
|
||||
|
||||
@@ -1925,6 +1925,8 @@
|
||||
"自动禁用关键词": "Automatic disable keywords",
|
||||
"自动禁用状态码": "Auto-disable status codes",
|
||||
"自动禁用状态码格式不正确": "Invalid auto-disable status code format",
|
||||
"自动重试状态码": "Auto-retry status codes",
|
||||
"自动重试状态码格式不正确": "Invalid auto-retry status code format",
|
||||
"支持填写单个状态码或范围(含首尾),使用逗号分隔": "Supports single status codes or inclusive ranges; separate with commas",
|
||||
"例如:401, 403, 429, 500-599": "e.g. 401,403,429,500-599",
|
||||
"自动选择": "Auto Select",
|
||||
|
||||
@@ -1911,6 +1911,8 @@
|
||||
"自动禁用关键词": "自动禁用关键词",
|
||||
"自动禁用状态码": "自动禁用状态码",
|
||||
"自动禁用状态码格式不正确": "自动禁用状态码格式不正确",
|
||||
"自动重试状态码": "自动重试状态码",
|
||||
"自动重试状态码格式不正确": "自动重试状态码格式不正确",
|
||||
"支持填写单个状态码或范围(含首尾),使用逗号分隔": "支持填写单个状态码或范围(含首尾),使用逗号分隔",
|
||||
"例如:401, 403, 429, 500-599": "例如:401,403,429,500-599",
|
||||
"自动选择": "自动选择",
|
||||
|
||||
@@ -24,8 +24,6 @@ import {
|
||||
Form,
|
||||
Row,
|
||||
Spin,
|
||||
Tag,
|
||||
Typography,
|
||||
} from '@douyinfe/semi-ui';
|
||||
import {
|
||||
compareObjects,
|
||||
@@ -34,13 +32,12 @@ import {
|
||||
showSuccess,
|
||||
showWarning,
|
||||
parseHttpStatusCodeRules,
|
||||
verifyJSON,
|
||||
} from '../../../helpers';
|
||||
import { useTranslation } from 'react-i18next';
|
||||
import HttpStatusCodeRulesInput from '../../../components/settings/HttpStatusCodeRulesInput';
|
||||
|
||||
export default function SettingsMonitoring(props) {
|
||||
const { t } = useTranslation();
|
||||
const { Text } = Typography;
|
||||
const [loading, setLoading] = useState(false);
|
||||
const [inputs, setInputs] = useState({
|
||||
ChannelDisableThreshold: '',
|
||||
@@ -49,6 +46,7 @@ export default function SettingsMonitoring(props) {
|
||||
AutomaticEnableChannelEnabled: false,
|
||||
AutomaticDisableKeywords: '',
|
||||
AutomaticDisableStatusCodes: '401',
|
||||
AutomaticRetryStatusCodes: '100-199,300-399,401-407,409-499,500-503,505-523,525-599',
|
||||
'monitor_setting.auto_test_channel_enabled': false,
|
||||
'monitor_setting.auto_test_channel_minutes': 10,
|
||||
});
|
||||
@@ -57,6 +55,9 @@ export default function SettingsMonitoring(props) {
|
||||
const parsedAutoDisableStatusCodes = parseHttpStatusCodeRules(
|
||||
inputs.AutomaticDisableStatusCodes || '',
|
||||
);
|
||||
const parsedAutoRetryStatusCodes = parseHttpStatusCodeRules(
|
||||
inputs.AutomaticRetryStatusCodes || '',
|
||||
);
|
||||
|
||||
function onSubmit() {
|
||||
const updateArray = compareObjects(inputs, inputsRow);
|
||||
@@ -69,16 +70,24 @@ export default function SettingsMonitoring(props) {
|
||||
: '';
|
||||
return showError(`${t('自动禁用状态码格式不正确')}${details}`);
|
||||
}
|
||||
if (!parsedAutoRetryStatusCodes.ok) {
|
||||
const details =
|
||||
parsedAutoRetryStatusCodes.invalidTokens &&
|
||||
parsedAutoRetryStatusCodes.invalidTokens.length > 0
|
||||
? `: ${parsedAutoRetryStatusCodes.invalidTokens.join(', ')}`
|
||||
: '';
|
||||
return showError(`${t('自动重试状态码格式不正确')}${details}`);
|
||||
}
|
||||
const requestQueue = updateArray.map((item) => {
|
||||
let value = '';
|
||||
if (typeof inputs[item.key] === 'boolean') {
|
||||
value = String(inputs[item.key]);
|
||||
} else {
|
||||
if (item.key === 'AutomaticDisableStatusCodes') {
|
||||
value = parsedAutoDisableStatusCodes.normalized;
|
||||
} else {
|
||||
value = inputs[item.key];
|
||||
}
|
||||
const normalizedMap = {
|
||||
AutomaticDisableStatusCodes: parsedAutoDisableStatusCodes.normalized,
|
||||
AutomaticRetryStatusCodes: parsedAutoRetryStatusCodes.normalized,
|
||||
};
|
||||
value = normalizedMap[item.key] ?? inputs[item.key];
|
||||
}
|
||||
return API.put('/api/option/', {
|
||||
key: item.key,
|
||||
@@ -233,7 +242,7 @@ export default function SettingsMonitoring(props) {
|
||||
</Row>
|
||||
<Row gutter={16}>
|
||||
<Col xs={24} sm={16}>
|
||||
<Form.Input
|
||||
<HttpStatusCodeRulesInput
|
||||
label={t('自动禁用状态码')}
|
||||
placeholder={t('例如:401, 403, 429, 500-599')}
|
||||
extraText={t(
|
||||
@@ -243,35 +252,22 @@ export default function SettingsMonitoring(props) {
|
||||
onChange={(value) =>
|
||||
setInputs({ ...inputs, AutomaticDisableStatusCodes: value })
|
||||
}
|
||||
parsed={parsedAutoDisableStatusCodes}
|
||||
invalidText={t('自动禁用状态码格式不正确')}
|
||||
/>
|
||||
{parsedAutoDisableStatusCodes.ok &&
|
||||
parsedAutoDisableStatusCodes.tokens.length > 0 && (
|
||||
<div
|
||||
style={{
|
||||
display: 'flex',
|
||||
flexWrap: 'wrap',
|
||||
gap: 8,
|
||||
marginTop: 8,
|
||||
}}
|
||||
>
|
||||
{parsedAutoDisableStatusCodes.tokens.map((token) => (
|
||||
<Tag key={token} size='small'>
|
||||
{token}
|
||||
</Tag>
|
||||
))}
|
||||
</div>
|
||||
<HttpStatusCodeRulesInput
|
||||
label={t('自动重试状态码')}
|
||||
placeholder={t('例如:401, 403, 429, 500-599')}
|
||||
extraText={t(
|
||||
'支持填写单个状态码或范围(含首尾),使用逗号分隔',
|
||||
)}
|
||||
{!parsedAutoDisableStatusCodes.ok && (
|
||||
<Text type='danger' style={{ display: 'block', marginTop: 8 }}>
|
||||
{t('自动禁用状态码格式不正确')}
|
||||
{parsedAutoDisableStatusCodes.invalidTokens &&
|
||||
parsedAutoDisableStatusCodes.invalidTokens.length > 0
|
||||
? `: ${parsedAutoDisableStatusCodes.invalidTokens.join(
|
||||
', ',
|
||||
)}`
|
||||
: ''}
|
||||
</Text>
|
||||
)}
|
||||
field={'AutomaticRetryStatusCodes'}
|
||||
onChange={(value) =>
|
||||
setInputs({ ...inputs, AutomaticRetryStatusCodes: value })
|
||||
}
|
||||
parsed={parsedAutoRetryStatusCodes}
|
||||
invalidText={t('自动重试状态码格式不正确')}
|
||||
/>
|
||||
<Form.TextArea
|
||||
label={t('自动禁用关键词')}
|
||||
placeholder={t('一行一个,不区分大小写')}
|
||||
|
||||
Reference in New Issue
Block a user