mirror of
https://github.com/QuantumNous/new-api.git
synced 2026-03-30 08:36:22 +00:00
Replace the legacy boolean “DisplayInCurrencyEnabled” with an injected, type-safe
configuration `general_setting.quota_display_type`, and wire it through the
backend and frontend.
Backend
- Add `QuotaDisplayType` to `operation_setting.GeneralSetting` with injected
registration via `config.GlobalConfig.Register("general_setting", ...)`.
Helpers: `IsCurrencyDisplay()`, `IsCNYDisplay()`, `GetQuotaDisplayType()`.
- Expose `quota_display_type` in `/api/status` and keep legacy
`display_in_currency` for backward compatibility.
- Logger: update `LogQuota` and `FormatQuota` to support USD/CNY/TOKENS. When
CNY is selected, convert using `operation_setting.USDExchangeRate`.
- Controllers:
- `billing`: compute subscription/usage amounts based on the selected type
(USD: divide by `QuotaPerUnit`; CNY: USD→CNY; TOKENS: keep raw tokens).
- `topup` / `topup_stripe`: treat inputs as “amount” for USD/CNY and as
token-count for TOKENS; adjust min topup and pay money accordingly.
- `misc`: include `quota_display_type` in status payload.
- Compatibility: in `model/option.UpdateOption`, map updates to
`DisplayInCurrencyEnabled` → `general_setting.quota_display_type`
(true→USD, false→TOKENS). Keep exporting the legacy key in `OptionMap`.
Frontend
- Settings: replace the “display in currency” switch with a Select
(`general_setting.quota_display_type`) offering USD / CNY / Tokens.
Provide fallback mapping from legacy `DisplayInCurrencyEnabled`.
- Persist `quota_display_type` to localStorage (keep `display_in_currency`
for legacy components).
- Rendering helpers: base all quota/price rendering on `quota_display_type`;
use `usd_exchange_rate` for CNY symbol/values.
- Pricing page: default view currency follows site display type (USD/CNY),
while TOKENS mode still allows per-view currency toggling when needed.
Notes
- No database migrations required.
- Legacy clients remain functional via compatibility fields.
401 lines
11 KiB
JavaScript
401 lines
11 KiB
JavaScript
/*
|
||
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 { useState, useEffect, useContext, useRef, useMemo } from 'react';
|
||
import { useTranslation } from 'react-i18next';
|
||
import { API, copy, showError, showInfo, showSuccess } from '../../helpers';
|
||
import { Modal } from '@douyinfe/semi-ui';
|
||
import { UserContext } from '../../context/User';
|
||
import { StatusContext } from '../../context/Status';
|
||
|
||
export const useModelPricingData = () => {
|
||
const { t } = useTranslation();
|
||
const [searchValue, setSearchValue] = useState('');
|
||
const compositionRef = useRef({ isComposition: false });
|
||
const [selectedRowKeys, setSelectedRowKeys] = useState([]);
|
||
const [modalImageUrl, setModalImageUrl] = useState('');
|
||
const [isModalOpenurl, setIsModalOpenurl] = useState(false);
|
||
const [selectedGroup, setSelectedGroup] = useState('all');
|
||
const [showModelDetail, setShowModelDetail] = useState(false);
|
||
const [selectedModel, setSelectedModel] = useState(null);
|
||
const [filterGroup, setFilterGroup] = useState('all'); // 用于 Table 的可用分组筛选,"all" 表示不过滤
|
||
const [filterQuotaType, setFilterQuotaType] = useState('all'); // 计费类型筛选: 'all' | 0 | 1
|
||
const [filterEndpointType, setFilterEndpointType] = useState('all'); // 端点类型筛选: 'all' | string
|
||
const [filterVendor, setFilterVendor] = useState('all'); // 供应商筛选: 'all' | 'unknown' | string
|
||
const [filterTag, setFilterTag] = useState('all'); // 模型标签筛选: 'all' | string
|
||
const [pageSize, setPageSize] = useState(20);
|
||
const [currentPage, setCurrentPage] = useState(1);
|
||
const [currency, setCurrency] = useState('USD');
|
||
const [showWithRecharge, setShowWithRecharge] = useState(false);
|
||
const [tokenUnit, setTokenUnit] = useState('M');
|
||
const [models, setModels] = useState([]);
|
||
const [vendorsMap, setVendorsMap] = useState({});
|
||
const [loading, setLoading] = useState(true);
|
||
const [groupRatio, setGroupRatio] = useState({});
|
||
const [usableGroup, setUsableGroup] = useState({});
|
||
const [endpointMap, setEndpointMap] = useState({});
|
||
const [autoGroups, setAutoGroups] = useState([]);
|
||
|
||
const [statusState] = useContext(StatusContext);
|
||
const [userState] = useContext(UserContext);
|
||
|
||
// 充值汇率(price)与美元兑人民币汇率(usd_exchange_rate)
|
||
const priceRate = useMemo(
|
||
() => statusState?.status?.price ?? 1,
|
||
[statusState],
|
||
);
|
||
const usdExchangeRate = useMemo(
|
||
() => statusState?.status?.usd_exchange_rate ?? priceRate,
|
||
[statusState, priceRate],
|
||
);
|
||
const customExchangeRate = useMemo(
|
||
() => statusState?.status?.custom_currency_exchange_rate ?? 1,
|
||
[statusState],
|
||
);
|
||
const customCurrencySymbol = useMemo(
|
||
() => statusState?.status?.custom_currency_symbol ?? '¤',
|
||
[statusState],
|
||
);
|
||
|
||
// 默认货币与站点展示类型同步(USD/CNY),TOKENS 时仍允许切换视图内货币
|
||
const siteDisplayType = useMemo(
|
||
() => statusState?.status?.quota_display_type || 'USD',
|
||
[statusState],
|
||
);
|
||
useEffect(() => {
|
||
if (
|
||
siteDisplayType === 'USD' ||
|
||
siteDisplayType === 'CNY' ||
|
||
siteDisplayType === 'CUSTOM'
|
||
) {
|
||
setCurrency(siteDisplayType);
|
||
}
|
||
}, [siteDisplayType]);
|
||
|
||
const filteredModels = useMemo(() => {
|
||
let result = models;
|
||
|
||
// 分组筛选
|
||
if (filterGroup !== 'all') {
|
||
result = result.filter((model) =>
|
||
model.enable_groups.includes(filterGroup),
|
||
);
|
||
}
|
||
|
||
// 计费类型筛选
|
||
if (filterQuotaType !== 'all') {
|
||
result = result.filter((model) => model.quota_type === filterQuotaType);
|
||
}
|
||
|
||
// 端点类型筛选
|
||
if (filterEndpointType !== 'all') {
|
||
result = result.filter(
|
||
(model) =>
|
||
model.supported_endpoint_types &&
|
||
model.supported_endpoint_types.includes(filterEndpointType),
|
||
);
|
||
}
|
||
|
||
// 供应商筛选
|
||
if (filterVendor !== 'all') {
|
||
if (filterVendor === 'unknown') {
|
||
result = result.filter((model) => !model.vendor_name);
|
||
} else {
|
||
result = result.filter((model) => model.vendor_name === filterVendor);
|
||
}
|
||
}
|
||
|
||
// 标签筛选
|
||
if (filterTag !== 'all') {
|
||
const tagLower = filterTag.toLowerCase();
|
||
result = result.filter((model) => {
|
||
if (!model.tags) return false;
|
||
const tagsArr = model.tags
|
||
.toLowerCase()
|
||
.split(/[,;|\s]+/)
|
||
.map((tag) => tag.trim())
|
||
.filter(Boolean);
|
||
return tagsArr.includes(tagLower);
|
||
});
|
||
}
|
||
|
||
// 搜索筛选
|
||
if (searchValue.length > 0) {
|
||
const searchTerm = searchValue.toLowerCase();
|
||
result = result.filter(
|
||
(model) =>
|
||
(model.model_name &&
|
||
model.model_name.toLowerCase().includes(searchTerm)) ||
|
||
(model.description &&
|
||
model.description.toLowerCase().includes(searchTerm)) ||
|
||
(model.tags && model.tags.toLowerCase().includes(searchTerm)) ||
|
||
(model.vendor_name &&
|
||
model.vendor_name.toLowerCase().includes(searchTerm)),
|
||
);
|
||
}
|
||
|
||
return result;
|
||
}, [
|
||
models,
|
||
searchValue,
|
||
filterGroup,
|
||
filterQuotaType,
|
||
filterEndpointType,
|
||
filterVendor,
|
||
filterTag,
|
||
]);
|
||
|
||
const rowSelection = useMemo(
|
||
() => ({
|
||
selectedRowKeys,
|
||
onChange: (keys) => {
|
||
setSelectedRowKeys(keys);
|
||
},
|
||
}),
|
||
[selectedRowKeys],
|
||
);
|
||
|
||
const displayPrice = (usdPrice) => {
|
||
let priceInUSD = usdPrice;
|
||
if (showWithRecharge) {
|
||
priceInUSD = (usdPrice * priceRate) / usdExchangeRate;
|
||
}
|
||
|
||
if (currency === 'CNY') {
|
||
return `¥${(priceInUSD * usdExchangeRate).toFixed(3)}`;
|
||
} else if (currency === 'CUSTOM') {
|
||
return `${customCurrencySymbol}${(priceInUSD * customExchangeRate).toFixed(3)}`;
|
||
}
|
||
return `$${priceInUSD.toFixed(3)}`;
|
||
};
|
||
|
||
const setModelsFormat = (models, groupRatio, vendorMap) => {
|
||
for (let i = 0; i < models.length; i++) {
|
||
const m = models[i];
|
||
m.key = m.model_name;
|
||
m.group_ratio = groupRatio[m.model_name];
|
||
|
||
if (m.vendor_id && vendorMap[m.vendor_id]) {
|
||
const vendor = vendorMap[m.vendor_id];
|
||
m.vendor_name = vendor.name;
|
||
m.vendor_icon = vendor.icon;
|
||
m.vendor_description = vendor.description;
|
||
}
|
||
}
|
||
models.sort((a, b) => {
|
||
return a.quota_type - b.quota_type;
|
||
});
|
||
|
||
models.sort((a, b) => {
|
||
if (a.model_name.startsWith('gpt') && !b.model_name.startsWith('gpt')) {
|
||
return -1;
|
||
} else if (
|
||
!a.model_name.startsWith('gpt') &&
|
||
b.model_name.startsWith('gpt')
|
||
) {
|
||
return 1;
|
||
} else {
|
||
return a.model_name.localeCompare(b.model_name);
|
||
}
|
||
});
|
||
|
||
setModels(models);
|
||
};
|
||
|
||
const loadPricing = async () => {
|
||
setLoading(true);
|
||
let url = '/api/pricing';
|
||
const res = await API.get(url);
|
||
const {
|
||
success,
|
||
message,
|
||
data,
|
||
vendors,
|
||
group_ratio,
|
||
usable_group,
|
||
supported_endpoint,
|
||
auto_groups,
|
||
} = res.data;
|
||
if (success) {
|
||
setGroupRatio(group_ratio);
|
||
setUsableGroup(usable_group);
|
||
setSelectedGroup('all');
|
||
// 构建供应商 Map 方便查找
|
||
const vendorMap = {};
|
||
if (Array.isArray(vendors)) {
|
||
vendors.forEach((v) => {
|
||
vendorMap[v.id] = v;
|
||
});
|
||
}
|
||
setVendorsMap(vendorMap);
|
||
setEndpointMap(supported_endpoint || {});
|
||
setAutoGroups(auto_groups || []);
|
||
setModelsFormat(data, group_ratio, vendorMap);
|
||
} else {
|
||
showError(message);
|
||
}
|
||
setLoading(false);
|
||
};
|
||
|
||
const refresh = async () => {
|
||
await loadPricing();
|
||
};
|
||
|
||
const copyText = async (text) => {
|
||
if (await copy(text)) {
|
||
showSuccess(t('已复制:') + text);
|
||
} else {
|
||
Modal.error({ title: t('无法复制到剪贴板,请手动复制'), content: text });
|
||
}
|
||
};
|
||
|
||
const handleChange = (value) => {
|
||
const newSearchValue = value ? value : '';
|
||
setSearchValue(newSearchValue);
|
||
};
|
||
|
||
const handleCompositionStart = () => {
|
||
compositionRef.current.isComposition = true;
|
||
};
|
||
|
||
const handleCompositionEnd = (event) => {
|
||
compositionRef.current.isComposition = false;
|
||
const value = event.target.value;
|
||
const newSearchValue = value ? value : '';
|
||
setSearchValue(newSearchValue);
|
||
};
|
||
|
||
const handleGroupClick = (group) => {
|
||
setSelectedGroup(group);
|
||
setFilterGroup(group);
|
||
if (group === 'all') {
|
||
showInfo(t('已切换至最优倍率视图,每个模型使用其最低倍率分组'));
|
||
} else {
|
||
showInfo(
|
||
t('当前查看的分组为:{{group}},倍率为:{{ratio}}', {
|
||
group: group,
|
||
ratio: groupRatio[group] ?? 1,
|
||
}),
|
||
);
|
||
}
|
||
};
|
||
|
||
const openModelDetail = (model) => {
|
||
setSelectedModel(model);
|
||
setShowModelDetail(true);
|
||
};
|
||
|
||
const closeModelDetail = () => {
|
||
setShowModelDetail(false);
|
||
setTimeout(() => {
|
||
setSelectedModel(null);
|
||
}, 300);
|
||
};
|
||
|
||
useEffect(() => {
|
||
refresh().then();
|
||
}, []);
|
||
|
||
// 当筛选条件变化时重置到第一页
|
||
useEffect(() => {
|
||
setCurrentPage(1);
|
||
}, [
|
||
filterGroup,
|
||
filterQuotaType,
|
||
filterEndpointType,
|
||
filterVendor,
|
||
filterTag,
|
||
searchValue,
|
||
]);
|
||
|
||
return {
|
||
// 状态
|
||
searchValue,
|
||
setSearchValue,
|
||
selectedRowKeys,
|
||
setSelectedRowKeys,
|
||
modalImageUrl,
|
||
setModalImageUrl,
|
||
isModalOpenurl,
|
||
setIsModalOpenurl,
|
||
selectedGroup,
|
||
setSelectedGroup,
|
||
showModelDetail,
|
||
setShowModelDetail,
|
||
selectedModel,
|
||
setSelectedModel,
|
||
filterGroup,
|
||
setFilterGroup,
|
||
filterQuotaType,
|
||
setFilterQuotaType,
|
||
filterEndpointType,
|
||
setFilterEndpointType,
|
||
filterVendor,
|
||
setFilterVendor,
|
||
filterTag,
|
||
setFilterTag,
|
||
pageSize,
|
||
setPageSize,
|
||
currentPage,
|
||
setCurrentPage,
|
||
currency,
|
||
setCurrency,
|
||
showWithRecharge,
|
||
setShowWithRecharge,
|
||
tokenUnit,
|
||
setTokenUnit,
|
||
models,
|
||
loading,
|
||
groupRatio,
|
||
usableGroup,
|
||
endpointMap,
|
||
autoGroups,
|
||
|
||
// 计算属性
|
||
priceRate,
|
||
usdExchangeRate,
|
||
filteredModels,
|
||
rowSelection,
|
||
|
||
// 供应商
|
||
vendorsMap,
|
||
|
||
// 用户和状态
|
||
userState,
|
||
statusState,
|
||
|
||
// 方法
|
||
displayPrice,
|
||
refresh,
|
||
copyText,
|
||
handleChange,
|
||
handleCompositionStart,
|
||
handleCompositionEnd,
|
||
handleGroupClick,
|
||
openModelDetail,
|
||
closeModelDetail,
|
||
|
||
// 引用
|
||
compositionRef,
|
||
|
||
// 国际化
|
||
t,
|
||
};
|
||
};
|