✨ feat: add traffic limit rules and user traffic statistics
This commit is contained in:
parent
eac7b27f60
commit
f8fbf5529f
@ -182,5 +182,19 @@
|
||||
"validationFailed": "Validation failed",
|
||||
"totalNodeGroups": "Total Node Groups",
|
||||
"invalidRange": "Minimum traffic must be less than maximum traffic",
|
||||
"rangeConflict": "Traffic range conflicts with node group \"{{name}}\" (range: {{min}} - {{max}} GB)"
|
||||
"rangeConflict": "Traffic range conflicts with node group \"{{name}}\" (range: {{min}} - {{max}} GB)",
|
||||
"isExpiredGroup": "Expired Node Group",
|
||||
"isExpiredGroupDescription": "Allow expired users to use limited nodes",
|
||||
"expiredDaysLimit": "Expired Days Limit",
|
||||
"expiredDaysLimitDescription": "Number of days after expiration that users can still access nodes",
|
||||
"maxTrafficGBExpired": "Max Traffic for Expired Users (GB)",
|
||||
"maxTrafficGBExpiredDescription": "Maximum traffic allowed for expired users (0 = unlimited)",
|
||||
"speedLimit": "Speed Limit (KB/s)",
|
||||
"speedLimitDescription": "Speed limit for users in this node group (0 = unlimited)",
|
||||
"expiredGroup": "Expired Only",
|
||||
"expiredSettings": "Expired Settings",
|
||||
"days": "days",
|
||||
"expiredGroupExists": "System already has an expired node group: {{name}}",
|
||||
"nodeGroupUsedBySubscribe": "This node group is used as default node group in subscription products, cannot set as expired group",
|
||||
"expiredGroupForCalculationDescription": "Expired-only node groups cannot participate in group calculation"
|
||||
}
|
||||
|
||||
@ -74,6 +74,17 @@
|
||||
"showOriginalPriceDescription": "When enabled, the subscription card will display both the original price and the discounted price to help users understand the discount amount",
|
||||
"speedLimit": "Speed Limit ",
|
||||
"traffic": "Traffic",
|
||||
"trafficLimit": "Traffic Limit",
|
||||
"trafficLimitRules": "Traffic Limit Rules",
|
||||
"trafficLimitDescription": "Configure traffic-based speed limit rules. When traffic usage reaches the specified amount, the speed will be limited.",
|
||||
"addTrafficLimitRule": "Add Traffic Limit Rule",
|
||||
"statType": "Statistics Type",
|
||||
"selectStatType": "Select type...",
|
||||
"statTypeHour": "Hour",
|
||||
"statTypeDay": "Day",
|
||||
"statValue": "Time Value",
|
||||
"trafficUsage": "Traffic Usage (GB)",
|
||||
"speedLimitKb": "Speed Limit (kb)",
|
||||
"unitPrice": "Unit Price",
|
||||
"unitTime": "Unit Time",
|
||||
"unlimitedInventory": "Unlimited (enter -1)",
|
||||
|
||||
@ -182,6 +182,20 @@
|
||||
"validationFailed": "验证失败",
|
||||
"totalNodeGroups": "总节点组数",
|
||||
"invalidRange": "最小流量必须小于最大流量",
|
||||
"rangeConflict": "流量区间与节点组 \"{{name}}\" 冲突(区间:{{min}} - {{max}} GB)"
|
||||
"rangeConflict": "流量区间与节点组 \"{{name}}\" 冲突(区间:{{min}} - {{max}} GB)",
|
||||
"isExpiredGroup": "过期节点组",
|
||||
"isExpiredGroupDescription": "允许过期用户使用受限节点",
|
||||
"expiredDaysLimit": "过期天数限制",
|
||||
"expiredDaysLimitDescription": "用户订阅过期后仍可访问节点的天数",
|
||||
"maxTrafficGBExpired": "过期用户最大流量 (GB)",
|
||||
"maxTrafficGBExpiredDescription": "过期用户允许使用的最大流量(0 = 不限制)",
|
||||
"speedLimit": "限速 (KB/s)",
|
||||
"speedLimitDescription": "该节点组用户的速度限制(0 = 不限制)",
|
||||
"expiredGroup": "过期专用",
|
||||
"expiredSettings": "过期设置",
|
||||
"days": "天",
|
||||
"expiredGroupExists": "系统中已存在过期节点组:{{name}}",
|
||||
"nodeGroupUsedBySubscribe": "该节点组已被订阅商品设置为默认节点组,不能设为过期节点组",
|
||||
"expiredGroupForCalculationDescription": "过期专用节点组不能参与分组计算"
|
||||
}
|
||||
|
||||
|
||||
@ -74,6 +74,17 @@
|
||||
"showOriginalPriceDescription": "开启后,在订阅卡片上将会显示原价和折后价,帮助用户了解优惠幅度",
|
||||
"speedLimit": "速度限制",
|
||||
"traffic": "流量",
|
||||
"trafficLimit": "按量限速",
|
||||
"trafficLimitRules": "按量限速规则",
|
||||
"trafficLimitDescription": "配置基于流量的限速规则。当流量使用达到指定量时,将进行限速。",
|
||||
"addTrafficLimitRule": "添加限速规则",
|
||||
"statType": "统计类型",
|
||||
"selectStatType": "选择类型...",
|
||||
"statTypeHour": "小时",
|
||||
"statTypeDay": "天",
|
||||
"statValue": "时间值",
|
||||
"trafficUsage": "使用流量(GB)",
|
||||
"speedLimitKb": "限速(kb)",
|
||||
"unitPrice": "单价",
|
||||
"unitTime": "时间单位",
|
||||
"unlimitedInventory": "无限制(输入 -1)",
|
||||
|
||||
@ -40,6 +40,10 @@ const NodeGroupForm = forwardRef<
|
||||
description: "",
|
||||
sort: 0,
|
||||
for_calculation: true,
|
||||
is_expired_group: false,
|
||||
expired_days_limit: 7,
|
||||
max_traffic_gb_expired: 0,
|
||||
speed_limit: 0,
|
||||
min_traffic_gb: 0,
|
||||
max_traffic_gb: 0,
|
||||
});
|
||||
@ -53,6 +57,10 @@ const NodeGroupForm = forwardRef<
|
||||
description: initialValues.description || "",
|
||||
sort: initialValues.sort ?? 0,
|
||||
for_calculation: initialValues.for_calculation ?? true,
|
||||
is_expired_group: initialValues.is_expired_group ?? false,
|
||||
expired_days_limit: initialValues.expired_days_limit ?? 7,
|
||||
max_traffic_gb_expired: initialValues.max_traffic_gb_expired ?? 0,
|
||||
speed_limit: initialValues.speed_limit ?? 0,
|
||||
min_traffic_gb: initialValues.min_traffic_gb ?? 0,
|
||||
max_traffic_gb: initialValues.max_traffic_gb ?? 0,
|
||||
});
|
||||
@ -62,6 +70,10 @@ const NodeGroupForm = forwardRef<
|
||||
description: "",
|
||||
sort: 0,
|
||||
for_calculation: true,
|
||||
is_expired_group: false,
|
||||
expired_days_limit: 7,
|
||||
max_traffic_gb_expired: 0,
|
||||
speed_limit: 0,
|
||||
min_traffic_gb: 0,
|
||||
max_traffic_gb: 0,
|
||||
});
|
||||
@ -118,15 +130,68 @@ const NodeGroupForm = forwardRef<
|
||||
return "";
|
||||
};
|
||||
|
||||
// 检测过期节点组冲突
|
||||
const checkExpiredGroupConflict = async (isExpiredGroup: boolean): Promise<string> => {
|
||||
if (!isExpiredGroup) {
|
||||
return "";
|
||||
}
|
||||
|
||||
// 检查是否已存在其他过期节点组
|
||||
const existingExpiredGroup = allNodeGroups.find(
|
||||
(group) => group.is_expired_group && group.id !== currentGroupId
|
||||
);
|
||||
|
||||
if (existingExpiredGroup) {
|
||||
return t("expiredGroupExists", `System already has an expired node group: ${existingExpiredGroup.name}`);
|
||||
}
|
||||
|
||||
// 检查当前节点组是否被订阅商品使用
|
||||
if (currentGroupId) {
|
||||
try {
|
||||
const { getSubscribeList } = await import("@workspace/ui/services/admin/subscribe");
|
||||
const { data } = await getSubscribeList({
|
||||
page: 1,
|
||||
size: 1,
|
||||
node_group_id: currentGroupId
|
||||
});
|
||||
|
||||
if (data.data && data.data.total > 0) {
|
||||
return t("nodeGroupUsedBySubscribe", "This node group is used as default node group in subscription products, cannot set as expired group");
|
||||
}
|
||||
} catch (error) {
|
||||
console.error("Failed to check subscribe usage:", error);
|
||||
}
|
||||
}
|
||||
|
||||
return "";
|
||||
};
|
||||
|
||||
// 检查是否存在其他过期节点组(用于隐藏开关)
|
||||
const hasOtherExpiredGroup = allNodeGroups.some(
|
||||
(group) => group.is_expired_group && group.id !== currentGroupId
|
||||
);
|
||||
|
||||
// 当前是否是过期节点组(编辑模式下)
|
||||
const isCurrentExpiredGroup = initialValues?.is_expired_group ?? false;
|
||||
|
||||
const handleSubmit = async (e: React.FormEvent) => {
|
||||
e.preventDefault();
|
||||
|
||||
// 检测流量区间冲突
|
||||
// 检测过期节点组冲突
|
||||
const expiredGroupConflict = await checkExpiredGroupConflict(values.is_expired_group);
|
||||
if (expiredGroupConflict) {
|
||||
setConflictError(expiredGroupConflict);
|
||||
return;
|
||||
}
|
||||
|
||||
// 仅在非过期节点组时检测流量区间冲突
|
||||
if (!values.is_expired_group) {
|
||||
const conflict = checkTrafficRangeConflict(values.min_traffic_gb, values.max_traffic_gb);
|
||||
if (conflict) {
|
||||
setConflictError(conflict);
|
||||
return;
|
||||
}
|
||||
}
|
||||
|
||||
setSubmitting(true);
|
||||
const success = await onSubmit(values);
|
||||
@ -139,6 +204,10 @@ const NodeGroupForm = forwardRef<
|
||||
description: "",
|
||||
sort: 0,
|
||||
for_calculation: true,
|
||||
is_expired_group: false,
|
||||
expired_days_limit: 7,
|
||||
max_traffic_gb_expired: 0,
|
||||
speed_limit: 0,
|
||||
min_traffic_gb: 0,
|
||||
max_traffic_gb: 0,
|
||||
});
|
||||
@ -207,18 +276,107 @@ const NodeGroupForm = forwardRef<
|
||||
{t("forCalculation", "For Calculation")}
|
||||
</Label>
|
||||
<p className="text-sm text-muted-foreground">
|
||||
{t("forCalculationDescription", "Whether this node group participates in grouping calculation")}
|
||||
{values.is_expired_group
|
||||
? t("expiredGroupForCalculationDescription", "Expired-only node groups cannot participate in group calculation")
|
||||
: t("forCalculationDescription", "Whether this node group participates in grouping calculation")}
|
||||
</p>
|
||||
</div>
|
||||
<Switch
|
||||
id="for_calculation"
|
||||
checked={values.for_calculation}
|
||||
disabled={values.is_expired_group}
|
||||
onCheckedChange={(checked) =>
|
||||
setValues({ ...values, for_calculation: checked })
|
||||
}
|
||||
/>
|
||||
</div>
|
||||
|
||||
{/* 仅在没有其他过期节点组或当前就是过期节点组时显示 */}
|
||||
{(!hasOtherExpiredGroup || isCurrentExpiredGroup) && (
|
||||
<div className="flex items-center justify-between">
|
||||
<div className="space-y-0.5">
|
||||
<Label htmlFor="is_expired_group">
|
||||
{t("isExpiredGroup", "Expired Node Group")}
|
||||
</Label>
|
||||
<p className="text-sm text-muted-foreground">
|
||||
{t("isExpiredGroupDescription", "Allow expired users to use limited nodes")}
|
||||
</p>
|
||||
</div>
|
||||
<Switch
|
||||
id="is_expired_group"
|
||||
checked={values.is_expired_group}
|
||||
onCheckedChange={async (checked) => {
|
||||
setValues({
|
||||
...values,
|
||||
is_expired_group: checked,
|
||||
for_calculation: checked ? false : values.for_calculation,
|
||||
min_traffic_gb: checked ? 0 : values.min_traffic_gb,
|
||||
max_traffic_gb: checked ? 0 : values.max_traffic_gb,
|
||||
});
|
||||
// 实时检测过期节点组冲突
|
||||
const conflict = await checkExpiredGroupConflict(checked);
|
||||
setConflictError(conflict);
|
||||
}}
|
||||
/>
|
||||
</div>
|
||||
)}
|
||||
|
||||
{values.is_expired_group && (
|
||||
<>
|
||||
<div className="space-y-2">
|
||||
<Label htmlFor="expired_days_limit">
|
||||
{t("expiredDaysLimit", "Expired Days Limit")}
|
||||
</Label>
|
||||
<p className="text-sm text-muted-foreground">
|
||||
{t("expiredDaysLimitDescription", "Number of days after expiration that users can still access nodes")}
|
||||
</p>
|
||||
<Input
|
||||
id="expired_days_limit"
|
||||
type="number"
|
||||
min={1}
|
||||
value={values.expired_days_limit}
|
||||
onChange={(e) =>
|
||||
setValues({ ...values, expired_days_limit: parseInt(e.target.value) || 7 })
|
||||
}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div className="space-y-2">
|
||||
<Label htmlFor="max_traffic_gb_expired">
|
||||
{t("maxTrafficGBExpired", "Max Traffic for Expired Users (GB)")}
|
||||
</Label>
|
||||
<p className="text-sm text-muted-foreground">
|
||||
{t("maxTrafficGBExpiredDescription", "Maximum traffic allowed for expired users (0 = unlimited)")}
|
||||
</p>
|
||||
<Input
|
||||
id="max_traffic_gb_expired"
|
||||
type="number"
|
||||
min={0}
|
||||
value={values.max_traffic_gb_expired}
|
||||
onChange={(e) =>
|
||||
setValues({ ...values, max_traffic_gb_expired: parseInt(e.target.value) || 0 })
|
||||
}
|
||||
/>
|
||||
</div>
|
||||
|
||||
<div className="space-y-2">
|
||||
<Label htmlFor="speed_limit">
|
||||
{t("speedLimit", "Speed Limit (KB/s)")}
|
||||
</Label>
|
||||
<Input
|
||||
id="speed_limit"
|
||||
type="number"
|
||||
min={0}
|
||||
value={values.speed_limit}
|
||||
onChange={(e) =>
|
||||
setValues({ ...values, speed_limit: parseInt(e.target.value) || 0 })
|
||||
}
|
||||
/>
|
||||
</div>
|
||||
</>
|
||||
)}
|
||||
|
||||
{!values.is_expired_group && (
|
||||
<div className="space-y-2">
|
||||
<div className="flex items-center justify-between">
|
||||
<Label>{t("trafficRangeGB", "Traffic Range (GB)")}</Label>
|
||||
@ -270,6 +428,7 @@ const NodeGroupForm = forwardRef<
|
||||
</div>
|
||||
)}
|
||||
</div>
|
||||
)}
|
||||
|
||||
<div className="flex justify-end gap-2">
|
||||
<button
|
||||
|
||||
@ -77,6 +77,19 @@ export default function NodeGroups() {
|
||||
id: "name",
|
||||
accessorKey: "name",
|
||||
header: t("name", "Name"),
|
||||
cell: ({ row }: { row: any }) => {
|
||||
const isExpiredGroup = row.original.is_expired_group;
|
||||
return (
|
||||
<div className="flex items-center gap-2">
|
||||
<span>{row.getValue("name")}</span>
|
||||
{isExpiredGroup && (
|
||||
<Badge variant="destructive">
|
||||
{t("expiredGroup", "Expired")}
|
||||
</Badge>
|
||||
)}
|
||||
</div>
|
||||
);
|
||||
},
|
||||
},
|
||||
{
|
||||
id: "description",
|
||||
@ -99,7 +112,6 @@ export default function NodeGroups() {
|
||||
},
|
||||
{
|
||||
id: "traffic_range",
|
||||
accessorKey: "traffic_range",
|
||||
header: t("trafficRange", "Traffic Range (GB)"),
|
||||
cell: ({ row }: { row: any }) => {
|
||||
const min = row.original.min_traffic_gb;
|
||||
|
||||
@ -84,6 +84,7 @@ const defaultValues = {
|
||||
renewal_reset: false,
|
||||
show_original_price: false,
|
||||
deduction_mode: "auto",
|
||||
traffic_limit: [],
|
||||
};
|
||||
|
||||
export default function SubscribeForm<T extends Record<string, any>>({
|
||||
@ -129,6 +130,16 @@ export default function SubscribeForm<T extends Record<string, any>>({
|
||||
reset_cycle: z.number().optional(),
|
||||
renewal_reset: z.boolean().optional(),
|
||||
show_original_price: z.boolean().optional(),
|
||||
traffic_limit: z
|
||||
.array(
|
||||
z.object({
|
||||
stat_type: z.string(),
|
||||
stat_value: z.number().int(),
|
||||
traffic_usage: z.number(),
|
||||
speed_limit: z.number(),
|
||||
})
|
||||
)
|
||||
.optional(),
|
||||
});
|
||||
|
||||
const form = useForm<z.infer<typeof formSchema>>({
|
||||
@ -283,12 +294,14 @@ export default function SubscribeForm<T extends Record<string, any>>({
|
||||
|
||||
const tagGroups = getAllAvailableTags();
|
||||
|
||||
// Fetch node groups
|
||||
// Fetch node groups (exclude expired groups)
|
||||
const { data: nodeGroupsData } = useQuery({
|
||||
queryKey: ["nodeGroups"],
|
||||
queryFn: async () => {
|
||||
const { data } = await getNodeGroupList({ page: 1, size: 1000 });
|
||||
return data.data?.list || [];
|
||||
const allGroups = data.data?.list || [];
|
||||
// Filter out expired node groups
|
||||
return allGroups.filter((group) => !group.is_expired_group);
|
||||
},
|
||||
});
|
||||
|
||||
@ -344,7 +357,7 @@ export default function SubscribeForm<T extends Record<string, any>>({
|
||||
<Form {...form}>
|
||||
<form className="pt-4" onSubmit={form.handleSubmit(handleSubmit)}>
|
||||
<Tabs className="w-full" defaultValue="basic">
|
||||
<TabsList className="mb-6 grid w-full grid-cols-3">
|
||||
<TabsList className="mb-6 grid w-full grid-cols-4">
|
||||
<TabsTrigger
|
||||
className="flex items-center gap-2"
|
||||
value="basic"
|
||||
@ -366,6 +379,13 @@ export default function SubscribeForm<T extends Record<string, any>>({
|
||||
<Server className="h-4 w-4" />
|
||||
{t("form.nodes")}
|
||||
</TabsTrigger>
|
||||
<TabsTrigger
|
||||
className="flex items-center gap-2"
|
||||
value="traffic-limit"
|
||||
>
|
||||
<Icon className="h-4 w-4" icon="uil:tachometer-fast" />
|
||||
{t("form.trafficLimit")}
|
||||
</TabsTrigger>
|
||||
</TabsList>
|
||||
|
||||
<TabsContent className="space-y-4" value="basic">
|
||||
@ -1396,6 +1416,91 @@ export default function SubscribeForm<T extends Record<string, any>>({
|
||||
)}
|
||||
</div>
|
||||
</TabsContent>
|
||||
|
||||
{/* Traffic Limit Tab */}
|
||||
<TabsContent className="space-y-4" value="traffic-limit">
|
||||
<div className="space-y-4">
|
||||
<FormField
|
||||
control={form.control}
|
||||
name="traffic_limit"
|
||||
render={({ field }) => (
|
||||
<FormItem>
|
||||
<FormLabel>{t("form.trafficLimitRules", "Traffic Limit Rules")}</FormLabel>
|
||||
<FormControl>
|
||||
<ArrayInput
|
||||
value={field.value && field.value.length > 0 ? field.value : [{ stat_type: "day" }]}
|
||||
onChange={field.onChange}
|
||||
fields={[
|
||||
{
|
||||
name: "stat_type",
|
||||
type: "select",
|
||||
placeholder: t("form.statType", "Statistics Type"),
|
||||
value: "day",
|
||||
options: [
|
||||
{ label: t("form.statTypeHour", "Hour"), value: "hour" },
|
||||
{ label: t("form.statTypeDay", "Day"), value: "day" },
|
||||
],
|
||||
},
|
||||
{
|
||||
name: "stat_value",
|
||||
type: "number",
|
||||
placeholder: t("form.statValue", "Time Value"),
|
||||
min: 1,
|
||||
onKeyDown: (e: React.KeyboardEvent<HTMLInputElement>) => {
|
||||
if (e.key === '.' || e.key === ',') {
|
||||
e.preventDefault();
|
||||
}
|
||||
},
|
||||
formatOutput: (value: string | number) => {
|
||||
const num = Number(value);
|
||||
return isNaN(num) ? 0 : Math.floor(num);
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "traffic_usage",
|
||||
type: "number",
|
||||
placeholder: t("form.trafficUsage", "Traffic Usage (GB)"),
|
||||
min: 0,
|
||||
onKeyDown: (e: React.KeyboardEvent<HTMLInputElement>) => {
|
||||
if (e.key === '.' || e.key === ',') {
|
||||
e.preventDefault();
|
||||
}
|
||||
},
|
||||
formatOutput: (value: string | number) => {
|
||||
const num = Number(value);
|
||||
return isNaN(num) ? 0 : Math.floor(num);
|
||||
},
|
||||
},
|
||||
{
|
||||
name: "speed_limit",
|
||||
type: "number",
|
||||
placeholder: t("form.speedLimitKb", "Speed Limit (kb)"),
|
||||
min: 0,
|
||||
onKeyDown: (e: React.KeyboardEvent<HTMLInputElement>) => {
|
||||
if (e.key === '.' || e.key === ',') {
|
||||
e.preventDefault();
|
||||
}
|
||||
},
|
||||
formatOutput: (value: string | number) => {
|
||||
const num = Number(value);
|
||||
return isNaN(num) ? 0 : Math.floor(num);
|
||||
},
|
||||
},
|
||||
]}
|
||||
/>
|
||||
</FormControl>
|
||||
<FormDescription>
|
||||
{t(
|
||||
"form.trafficLimitDescription",
|
||||
"Configure traffic-based speed limit rules. When traffic usage reaches the specified amount, the speed will be limited."
|
||||
)}
|
||||
</FormDescription>
|
||||
<FormMessage />
|
||||
</FormItem>
|
||||
)}
|
||||
/>
|
||||
</div>
|
||||
</TabsContent>
|
||||
</Tabs>
|
||||
</form>
|
||||
</Form>
|
||||
|
||||
@ -31,12 +31,14 @@ export default function SubscribeTable() {
|
||||
const ref = useRef<ProTableActions>(null);
|
||||
const { fetchSubscribes } = useSubscribe();
|
||||
|
||||
// Fetch node groups for filtering
|
||||
// Fetch node groups for filtering (exclude expired groups)
|
||||
const { data: nodeGroupsData } = useQuery({
|
||||
queryKey: ["nodeGroups"],
|
||||
queryFn: async () => {
|
||||
const { data } = await getNodeGroupList({ page: 1, size: 1000 });
|
||||
return data.data?.list || [];
|
||||
const allGroups = data.data?.list || [];
|
||||
// Filter out expired node groups
|
||||
return allGroups.filter((group) => !group.is_expired_group);
|
||||
},
|
||||
});
|
||||
|
||||
|
||||
@ -76,6 +76,7 @@
|
||||
"server": "Server Management",
|
||||
"subscribe": "Subscribe",
|
||||
"ticket": "Ticket Management",
|
||||
"traffic": "Traffic Statistics",
|
||||
"wallet": "Balance"
|
||||
},
|
||||
"pagination": {
|
||||
|
||||
16
apps/user/public/assets/locales/en-US/traffic.json
Normal file
16
apps/user/public/assets/locales/en-US/traffic.json
Normal file
@ -0,0 +1,16 @@
|
||||
{
|
||||
"title": "Traffic Statistics",
|
||||
"selectSubscription": "Select Subscription",
|
||||
"days7": "7 Days",
|
||||
"days30": "30 Days",
|
||||
"totalTraffic": "Total Traffic",
|
||||
"uploadTraffic": "Upload Traffic",
|
||||
"downloadTraffic": "Download Traffic",
|
||||
"trafficTrend": "Traffic Trend",
|
||||
"trafficRatio": "Upload/Download Ratio",
|
||||
"upload": "Upload",
|
||||
"download": "Download",
|
||||
"noData": "No traffic data available",
|
||||
"date": "Date",
|
||||
"traffic": "Traffic (MB)"
|
||||
}
|
||||
@ -76,6 +76,7 @@
|
||||
"server": "服务器管理",
|
||||
"subscribe": "订阅管理",
|
||||
"ticket": "工单管理",
|
||||
"traffic": "流量统计",
|
||||
"wallet": "余额管理"
|
||||
},
|
||||
"pagination": {
|
||||
|
||||
16
apps/user/public/assets/locales/zh-CN/traffic.json
Normal file
16
apps/user/public/assets/locales/zh-CN/traffic.json
Normal file
@ -0,0 +1,16 @@
|
||||
{
|
||||
"title": "流量统计",
|
||||
"selectSubscription": "选择订阅",
|
||||
"days7": "7 天",
|
||||
"days30": "30 天",
|
||||
"totalTraffic": "总流量",
|
||||
"uploadTraffic": "上传流量",
|
||||
"downloadTraffic": "下载流量",
|
||||
"trafficTrend": "流量趋势",
|
||||
"trafficRatio": "上传/下载占比",
|
||||
"upload": "上传",
|
||||
"download": "下载",
|
||||
"noData": "暂无流量数据",
|
||||
"date": "日期",
|
||||
"traffic": "流量 (MB)"
|
||||
}
|
||||
@ -40,6 +40,11 @@ export function useNavs() {
|
||||
icon: "uil:shop",
|
||||
title: t("menu.subscribe", "Subscribe"),
|
||||
},
|
||||
{
|
||||
url: "/traffic",
|
||||
icon: "uil:chart-line",
|
||||
title: t("menu.traffic", "Traffic Statistics"),
|
||||
},
|
||||
],
|
||||
},
|
||||
{
|
||||
|
||||
@ -28,6 +28,9 @@ const mainPurchasingIndexLazyRouteImport = createFileRoute(
|
||||
'/(main)/purchasing/',
|
||||
)()
|
||||
const mainuserWalletLazyRouteImport = createFileRoute('/(main)/(user)/wallet')()
|
||||
const mainuserTrafficLazyRouteImport = createFileRoute(
|
||||
'/(main)/(user)/traffic',
|
||||
)()
|
||||
const mainuserTicketLazyRouteImport = createFileRoute('/(main)/(user)/ticket')()
|
||||
const mainuserSubscribeLazyRouteImport = createFileRoute(
|
||||
'/(main)/(user)/subscribe',
|
||||
@ -126,6 +129,15 @@ const mainuserWalletLazyRoute = mainuserWalletLazyRouteImport
|
||||
getParentRoute: () => mainuserRouteLazyRoute,
|
||||
} as any)
|
||||
.lazy(() => import('./routes/(main)/(user)/wallet.lazy').then((d) => d.Route))
|
||||
const mainuserTrafficLazyRoute = mainuserTrafficLazyRouteImport
|
||||
.update({
|
||||
id: '/traffic',
|
||||
path: '/traffic',
|
||||
getParentRoute: () => mainuserRouteLazyRoute,
|
||||
} as any)
|
||||
.lazy(() =>
|
||||
import('./routes/(main)/(user)/traffic.lazy').then((d) => d.Route),
|
||||
)
|
||||
const mainuserTicketLazyRoute = mainuserTicketLazyRouteImport
|
||||
.update({
|
||||
id: '/ticket',
|
||||
@ -220,6 +232,7 @@ export interface FileRoutesByFullPath {
|
||||
'/profile': typeof mainuserProfileLazyRoute
|
||||
'/subscribe': typeof mainuserSubscribeLazyRoute
|
||||
'/ticket': typeof mainuserTicketLazyRoute
|
||||
'/traffic': typeof mainuserTrafficLazyRoute
|
||||
'/wallet': typeof mainuserWalletLazyRoute
|
||||
'/purchasing': typeof mainPurchasingIndexLazyRoute
|
||||
'/purchasing/order': typeof mainPurchasingOrderIndexRoute
|
||||
@ -240,6 +253,7 @@ export interface FileRoutesByTo {
|
||||
'/profile': typeof mainuserProfileLazyRoute
|
||||
'/subscribe': typeof mainuserSubscribeLazyRoute
|
||||
'/ticket': typeof mainuserTicketLazyRoute
|
||||
'/traffic': typeof mainuserTrafficLazyRoute
|
||||
'/wallet': typeof mainuserWalletLazyRoute
|
||||
'/purchasing': typeof mainPurchasingIndexLazyRoute
|
||||
'/purchasing/order': typeof mainPurchasingOrderIndexRoute
|
||||
@ -263,6 +277,7 @@ export interface FileRoutesById {
|
||||
'/(main)/(user)/profile': typeof mainuserProfileLazyRoute
|
||||
'/(main)/(user)/subscribe': typeof mainuserSubscribeLazyRoute
|
||||
'/(main)/(user)/ticket': typeof mainuserTicketLazyRoute
|
||||
'/(main)/(user)/traffic': typeof mainuserTrafficLazyRoute
|
||||
'/(main)/(user)/wallet': typeof mainuserWalletLazyRoute
|
||||
'/(main)/purchasing/': typeof mainPurchasingIndexLazyRoute
|
||||
'/(main)/purchasing/order/': typeof mainPurchasingOrderIndexRoute
|
||||
@ -285,6 +300,7 @@ export interface FileRouteTypes {
|
||||
| '/profile'
|
||||
| '/subscribe'
|
||||
| '/ticket'
|
||||
| '/traffic'
|
||||
| '/wallet'
|
||||
| '/purchasing'
|
||||
| '/purchasing/order'
|
||||
@ -305,6 +321,7 @@ export interface FileRouteTypes {
|
||||
| '/profile'
|
||||
| '/subscribe'
|
||||
| '/ticket'
|
||||
| '/traffic'
|
||||
| '/wallet'
|
||||
| '/purchasing'
|
||||
| '/purchasing/order'
|
||||
@ -327,6 +344,7 @@ export interface FileRouteTypes {
|
||||
| '/(main)/(user)/profile'
|
||||
| '/(main)/(user)/subscribe'
|
||||
| '/(main)/(user)/ticket'
|
||||
| '/(main)/(user)/traffic'
|
||||
| '/(main)/(user)/wallet'
|
||||
| '/(main)/purchasing/'
|
||||
| '/(main)/purchasing/order/'
|
||||
@ -418,6 +436,13 @@ declare module '@tanstack/react-router' {
|
||||
preLoaderRoute: typeof mainuserWalletLazyRouteImport
|
||||
parentRoute: typeof mainuserRouteLazyRoute
|
||||
}
|
||||
'/(main)/(user)/traffic': {
|
||||
id: '/(main)/(user)/traffic'
|
||||
path: '/traffic'
|
||||
fullPath: '/traffic'
|
||||
preLoaderRoute: typeof mainuserTrafficLazyRouteImport
|
||||
parentRoute: typeof mainuserRouteLazyRoute
|
||||
}
|
||||
'/(main)/(user)/ticket': {
|
||||
id: '/(main)/(user)/ticket'
|
||||
path: '/ticket'
|
||||
@ -493,6 +518,7 @@ interface mainuserRouteLazyRouteChildren {
|
||||
mainuserProfileLazyRoute: typeof mainuserProfileLazyRoute
|
||||
mainuserSubscribeLazyRoute: typeof mainuserSubscribeLazyRoute
|
||||
mainuserTicketLazyRoute: typeof mainuserTicketLazyRoute
|
||||
mainuserTrafficLazyRoute: typeof mainuserTrafficLazyRoute
|
||||
mainuserWalletLazyRoute: typeof mainuserWalletLazyRoute
|
||||
}
|
||||
|
||||
@ -505,6 +531,7 @@ const mainuserRouteLazyRouteChildren: mainuserRouteLazyRouteChildren = {
|
||||
mainuserProfileLazyRoute: mainuserProfileLazyRoute,
|
||||
mainuserSubscribeLazyRoute: mainuserSubscribeLazyRoute,
|
||||
mainuserTicketLazyRoute: mainuserTicketLazyRoute,
|
||||
mainuserTrafficLazyRoute: mainuserTrafficLazyRoute,
|
||||
mainuserWalletLazyRoute: mainuserWalletLazyRoute,
|
||||
}
|
||||
|
||||
|
||||
6
apps/user/src/routes/(main)/(user)/traffic.lazy.tsx
Normal file
6
apps/user/src/routes/(main)/(user)/traffic.lazy.tsx
Normal file
@ -0,0 +1,6 @@
|
||||
import { createLazyFileRoute } from "@tanstack/react-router";
|
||||
import TrafficStatistics from "@/sections/user/traffic-statistics";
|
||||
|
||||
export const Route = createLazyFileRoute("/(main)/(user)/traffic")({
|
||||
component: TrafficStatistics,
|
||||
});
|
||||
132
apps/user/src/sections/user/traffic-statistics/index.tsx
Normal file
132
apps/user/src/sections/user/traffic-statistics/index.tsx
Normal file
@ -0,0 +1,132 @@
|
||||
import { useQuery } from "@tanstack/react-query";
|
||||
import { Card, CardContent, CardHeader, CardTitle } from "@workspace/ui/components/card";
|
||||
import { Select, SelectContent, SelectItem, SelectTrigger, SelectValue } from "@workspace/ui/components/select";
|
||||
import { Tabs, TabsList, TabsTrigger } from "@workspace/ui/components/tabs";
|
||||
import { Icon } from "@workspace/ui/composed/icon";
|
||||
import { queryUserSubscribe } from "@workspace/ui/services/user/user";
|
||||
import { getUserTrafficStats } from "@workspace/ui/services/user/traffic";
|
||||
import { useState } from "react";
|
||||
import { useTranslation } from "react-i18next";
|
||||
import TrafficTrendChart from "./traffic-trend-chart";
|
||||
import TrafficRatioChart from "./traffic-ratio-chart";
|
||||
import TrafficStatsCards from "./traffic-stats-cards";
|
||||
|
||||
export default function TrafficStatistics() {
|
||||
const { t } = useTranslation("traffic");
|
||||
const [days, setDays] = useState<7 | 30>(7);
|
||||
const [selectedSubscribeId, setSelectedSubscribeId] = useState<string | null>(null);
|
||||
|
||||
// 查询用户订阅列表
|
||||
const { data: userSubscribe = [] } = useQuery({
|
||||
queryKey: ["queryUserSubscribe"],
|
||||
queryFn: async () => {
|
||||
const { data } = await queryUserSubscribe();
|
||||
return data.data?.list || [];
|
||||
},
|
||||
});
|
||||
|
||||
// 使用 id_str 字段,避免 JavaScript 精度丢失
|
||||
const activeSubscribeId = selectedSubscribeId || (userSubscribe[0]?.id_str || null);
|
||||
|
||||
// 查询流量统计数据
|
||||
const { data: trafficStats, isLoading } = useQuery({
|
||||
queryKey: ["getUserTrafficStats", activeSubscribeId, days],
|
||||
queryFn: async () => {
|
||||
if (!activeSubscribeId) return null;
|
||||
const { data } = await getUserTrafficStats({
|
||||
user_subscribe_id: activeSubscribeId,
|
||||
days,
|
||||
});
|
||||
return data.data;
|
||||
},
|
||||
enabled: !!activeSubscribeId,
|
||||
});
|
||||
|
||||
return (
|
||||
<div className="flex min-h-[calc(100vh-64px-58px-32px-114px)] w-full flex-col gap-4">
|
||||
{/* 标题和控制栏 */}
|
||||
<div className="flex flex-col gap-4 md:flex-row md:items-center md:justify-between">
|
||||
<h2 className="flex items-center gap-1.5 font-semibold">
|
||||
<Icon className="size-5" icon="uil:chart-line" />
|
||||
{t("title", "Traffic Statistics")}
|
||||
</h2>
|
||||
<div className="flex flex-col gap-2 md:flex-row md:items-center">
|
||||
{/* 订阅选择 */}
|
||||
{userSubscribe.length > 1 && (
|
||||
<Select
|
||||
value={activeSubscribeId || undefined}
|
||||
onValueChange={(value) => setSelectedSubscribeId(value)}
|
||||
>
|
||||
<SelectTrigger className="w-full md:w-[200px]">
|
||||
<SelectValue placeholder={t("selectSubscription", "Select Subscription")} />
|
||||
</SelectTrigger>
|
||||
<SelectContent>
|
||||
{userSubscribe.map((sub) => (
|
||||
<SelectItem key={sub.id} value={sub.id_str}>
|
||||
{sub.subscribe.name}
|
||||
</SelectItem>
|
||||
))}
|
||||
</SelectContent>
|
||||
</Select>
|
||||
)}
|
||||
{/* 时间范围切换 */}
|
||||
<Tabs value={String(days)} onValueChange={(value) => setDays(Number(value) as 7 | 30)}>
|
||||
<TabsList>
|
||||
<TabsTrigger value="7">{t("days7", "7 Days")}</TabsTrigger>
|
||||
<TabsTrigger value="30">{t("days30", "30 Days")}</TabsTrigger>
|
||||
</TabsList>
|
||||
</Tabs>
|
||||
</div>
|
||||
</div>
|
||||
|
||||
{/* 统计卡片 */}
|
||||
{trafficStats && <TrafficStatsCards stats={trafficStats} />}
|
||||
|
||||
{/* 图表区域 */}
|
||||
<div className="grid gap-4 md:grid-cols-2">
|
||||
{/* 流量趋势图 */}
|
||||
<Card>
|
||||
<CardHeader>
|
||||
<CardTitle className="text-base">{t("trafficTrend", "Traffic Trend")}</CardTitle>
|
||||
</CardHeader>
|
||||
<CardContent>
|
||||
{isLoading ? (
|
||||
<div className="flex h-[300px] items-center justify-center">
|
||||
<Icon className="size-8 animate-spin" icon="uil:spinner" />
|
||||
</div>
|
||||
) : trafficStats && trafficStats.list.length > 0 ? (
|
||||
<TrafficTrendChart data={trafficStats.list} />
|
||||
) : (
|
||||
<div className="flex h-[300px] items-center justify-center text-muted-foreground">
|
||||
{t("noData", "No traffic data available")}
|
||||
</div>
|
||||
)}
|
||||
</CardContent>
|
||||
</Card>
|
||||
|
||||
{/* 流量占比图 */}
|
||||
<Card>
|
||||
<CardHeader>
|
||||
<CardTitle className="text-base">{t("trafficRatio", "Upload/Download Ratio")}</CardTitle>
|
||||
</CardHeader>
|
||||
<CardContent>
|
||||
{isLoading ? (
|
||||
<div className="flex h-[300px] items-center justify-center">
|
||||
<Icon className="size-8 animate-spin" icon="uil:spinner" />
|
||||
</div>
|
||||
) : trafficStats && (trafficStats.total_upload > 0 || trafficStats.total_download > 0) ? (
|
||||
<TrafficRatioChart
|
||||
upload={trafficStats.total_upload}
|
||||
download={trafficStats.total_download}
|
||||
/>
|
||||
) : (
|
||||
<div className="flex h-[300px] items-center justify-center text-muted-foreground">
|
||||
{t("noData", "No traffic data available")}
|
||||
</div>
|
||||
)}
|
||||
</CardContent>
|
||||
</Card>
|
||||
</div>
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@ -0,0 +1,55 @@
|
||||
import { useTranslation } from "react-i18next";
|
||||
import { PieChart, Pie, Cell, ResponsiveContainer, Legend, Tooltip } from "recharts";
|
||||
|
||||
interface TrafficRatioChartProps {
|
||||
upload: number;
|
||||
download: number;
|
||||
}
|
||||
|
||||
export default function TrafficRatioChart({ upload, download }: TrafficRatioChartProps) {
|
||||
const { t } = useTranslation("traffic");
|
||||
|
||||
const data = [
|
||||
{ name: t("upload", "Upload"), value: upload },
|
||||
{ name: t("download", "Download"), value: download },
|
||||
];
|
||||
|
||||
const COLORS = ["#10b981", "#3b82f6"];
|
||||
|
||||
// 格式化流量显示
|
||||
const formatTraffic = (value: number) => {
|
||||
if (value >= 1024 * 1024 * 1024) {
|
||||
return `${(value / (1024 * 1024 * 1024)).toFixed(2)} GB`;
|
||||
}
|
||||
if (value >= 1024 * 1024) {
|
||||
return `${(value / (1024 * 1024)).toFixed(2)} MB`;
|
||||
}
|
||||
if (value >= 1024) {
|
||||
return `${(value / 1024).toFixed(2)} KB`;
|
||||
}
|
||||
return `${value} B`;
|
||||
};
|
||||
|
||||
return (
|
||||
<ResponsiveContainer width="100%" height={300}>
|
||||
<PieChart>
|
||||
<Pie
|
||||
data={data}
|
||||
cx="50%"
|
||||
cy="50%"
|
||||
labelLine={false}
|
||||
label={({ name, percent }) => `${name}: ${(percent * 100).toFixed(1)}%`}
|
||||
outerRadius={80}
|
||||
fill="#8884d8"
|
||||
dataKey="value"
|
||||
>
|
||||
{data.map((_, index) => (
|
||||
<Cell key={`cell-${index}`} fill={COLORS[index % COLORS.length]} />
|
||||
))}
|
||||
</Pie>
|
||||
<Tooltip formatter={(value: number) => formatTraffic(value)} />
|
||||
<Legend />
|
||||
</PieChart>
|
||||
</ResponsiveContainer>
|
||||
);
|
||||
}
|
||||
@ -0,0 +1,52 @@
|
||||
import { Card, CardContent } from "@workspace/ui/components/card";
|
||||
import { Icon } from "@workspace/ui/composed/icon";
|
||||
import type { GetUserTrafficStatsResponse } from "@workspace/ui/services/user/traffic";
|
||||
import { useTranslation } from "react-i18next";
|
||||
import { Display } from "@/components/display";
|
||||
|
||||
interface TrafficStatsCardsProps {
|
||||
stats: GetUserTrafficStatsResponse;
|
||||
}
|
||||
|
||||
export default function TrafficStatsCards({ stats }: TrafficStatsCardsProps) {
|
||||
const { t } = useTranslation("traffic");
|
||||
|
||||
const cards = [
|
||||
{
|
||||
title: t("totalTraffic", "Total Traffic"),
|
||||
value: stats.total_traffic,
|
||||
icon: "uil:chart-line",
|
||||
color: "text-blue-500",
|
||||
},
|
||||
{
|
||||
title: t("uploadTraffic", "Upload Traffic"),
|
||||
value: stats.total_upload,
|
||||
icon: "uil:upload",
|
||||
color: "text-green-500",
|
||||
},
|
||||
{
|
||||
title: t("downloadTraffic", "Download Traffic"),
|
||||
value: stats.total_download,
|
||||
icon: "uil:download",
|
||||
color: "text-purple-500",
|
||||
},
|
||||
];
|
||||
|
||||
return (
|
||||
<div className="grid gap-4 md:grid-cols-3">
|
||||
{cards.map((card) => (
|
||||
<Card key={card.title}>
|
||||
<CardContent className="flex items-center justify-between p-6">
|
||||
<div className="flex flex-col gap-1">
|
||||
<span className="text-muted-foreground text-sm">{card.title}</span>
|
||||
<span className="font-bold text-2xl">
|
||||
<Display type="traffic" value={card.value} />
|
||||
</span>
|
||||
</div>
|
||||
<Icon className={`size-10 ${card.color}`} icon={card.icon} />
|
||||
</CardContent>
|
||||
</Card>
|
||||
))}
|
||||
</div>
|
||||
);
|
||||
}
|
||||
@ -0,0 +1,81 @@
|
||||
import type { GetUserTrafficStatsResponse } from "@workspace/ui/services/user/traffic";
|
||||
import { format } from "date-fns";
|
||||
import { useTranslation } from "react-i18next";
|
||||
import {
|
||||
LineChart,
|
||||
Line,
|
||||
XAxis,
|
||||
YAxis,
|
||||
CartesianGrid,
|
||||
Tooltip,
|
||||
Legend,
|
||||
ResponsiveContainer,
|
||||
} from "recharts";
|
||||
|
||||
interface TrafficTrendChartProps {
|
||||
data: GetUserTrafficStatsResponse["list"];
|
||||
}
|
||||
|
||||
export default function TrafficTrendChart({ data }: TrafficTrendChartProps) {
|
||||
const { t } = useTranslation("traffic");
|
||||
|
||||
// 转换数据格式,将字节转换为 MB(保持为数字)
|
||||
const chartData = data.map((item) => ({
|
||||
date: format(new Date(item.date), "MM-dd"),
|
||||
upload: Number((item.upload / (1024 * 1024)).toFixed(2)),
|
||||
download: Number((item.download / (1024 * 1024)).toFixed(2)),
|
||||
}));
|
||||
|
||||
// 格式化流量显示
|
||||
const formatTraffic = (value: number | string) => {
|
||||
const numValue = typeof value === "string" ? parseFloat(value) : value;
|
||||
if (numValue >= 1024) {
|
||||
return `${(numValue / 1024).toFixed(2)} GB`;
|
||||
}
|
||||
return `${numValue.toFixed(2)} MB`;
|
||||
};
|
||||
|
||||
return (
|
||||
<ResponsiveContainer width="100%" height={300}>
|
||||
<LineChart data={chartData}>
|
||||
<CartesianGrid strokeDasharray="3 3" />
|
||||
<XAxis
|
||||
dataKey="date"
|
||||
label={{ value: t("date", "Date"), position: "insideBottom", offset: -5 }}
|
||||
/>
|
||||
<YAxis
|
||||
label={{ value: t("traffic", "Traffic (MB)"), angle: -90, position: "insideLeft" }}
|
||||
/>
|
||||
<Tooltip
|
||||
formatter={(value: number | string) => formatTraffic(value)}
|
||||
labelStyle={{ color: "#000" }}
|
||||
/>
|
||||
<Legend
|
||||
verticalAlign="bottom"
|
||||
height={36}
|
||||
wrapperStyle={{
|
||||
position: "absolute",
|
||||
width: "444px",
|
||||
height: "36px",
|
||||
left: "5px",
|
||||
bottom: "-5px"
|
||||
}}
|
||||
/>
|
||||
<Line
|
||||
type="monotone"
|
||||
dataKey="upload"
|
||||
stroke="#10b981"
|
||||
name={t("upload", "Upload")}
|
||||
strokeWidth={2}
|
||||
/>
|
||||
<Line
|
||||
type="monotone"
|
||||
dataKey="download"
|
||||
stroke="#3b82f6"
|
||||
name={t("download", "Download")}
|
||||
strokeWidth={2}
|
||||
/>
|
||||
</LineChart>
|
||||
</ResponsiveContainer>
|
||||
);
|
||||
}
|
||||
13
packages/ui/src/services/admin/typings.d.ts
vendored
13
packages/ui/src/services/admin/typings.d.ts
vendored
@ -2693,8 +2693,13 @@ declare namespace API {
|
||||
description: string;
|
||||
sort: number;
|
||||
for_calculation: boolean;
|
||||
is_expired_group: boolean;
|
||||
expired_days_limit: number;
|
||||
max_traffic_gb_expired?: number;
|
||||
speed_limit: number;
|
||||
min_traffic_gb?: number;
|
||||
max_traffic_gb?: number;
|
||||
node_count?: number;
|
||||
created_at: number;
|
||||
updated_at: number;
|
||||
};
|
||||
@ -2832,6 +2837,10 @@ declare namespace API {
|
||||
description?: string;
|
||||
sort?: number;
|
||||
for_calculation?: boolean;
|
||||
is_expired_group?: boolean;
|
||||
expired_days_limit?: number;
|
||||
max_traffic_gb_expired?: number;
|
||||
speed_limit?: number;
|
||||
min_traffic_gb?: number;
|
||||
max_traffic_gb?: number;
|
||||
};
|
||||
@ -2842,6 +2851,10 @@ declare namespace API {
|
||||
description?: string;
|
||||
sort?: number;
|
||||
for_calculation?: boolean;
|
||||
is_expired_group?: boolean;
|
||||
expired_days_limit?: number;
|
||||
max_traffic_gb_expired?: number;
|
||||
speed_limit?: number;
|
||||
min_traffic_gb?: number;
|
||||
max_traffic_gb?: number;
|
||||
};
|
||||
|
||||
@ -9,6 +9,7 @@ import * as payment from "./payment";
|
||||
import * as portal from "./portal";
|
||||
import * as subscribe from "./subscribe";
|
||||
import * as ticket from "./ticket";
|
||||
import * as traffic from "./traffic";
|
||||
import * as user from "./user";
|
||||
export default {
|
||||
announcement,
|
||||
@ -18,5 +19,6 @@ export default {
|
||||
portal,
|
||||
subscribe,
|
||||
ticket,
|
||||
traffic,
|
||||
user,
|
||||
};
|
||||
|
||||
37
packages/ui/src/services/user/traffic.ts
Normal file
37
packages/ui/src/services/user/traffic.ts
Normal file
@ -0,0 +1,37 @@
|
||||
import request from "@workspace/ui/lib/request";
|
||||
|
||||
export interface GetUserTrafficStatsRequest {
|
||||
user_subscribe_id: string; // 保持字符串,避免精度问题
|
||||
days: 7 | 30;
|
||||
}
|
||||
|
||||
export interface DailyTrafficStats {
|
||||
date: string;
|
||||
upload: number;
|
||||
download: number;
|
||||
total: number;
|
||||
}
|
||||
|
||||
export interface GetUserTrafficStatsResponse {
|
||||
list: DailyTrafficStats[];
|
||||
total_upload: number;
|
||||
total_download: number;
|
||||
total_traffic: number;
|
||||
}
|
||||
|
||||
/** Get User Traffic Statistics GET /v1/public/user/traffic_stats */
|
||||
export async function getUserTrafficStats(
|
||||
params: GetUserTrafficStatsRequest,
|
||||
options?: { [key: string]: any }
|
||||
) {
|
||||
return request<API.Response & { data?: GetUserTrafficStatsResponse }>(
|
||||
`${import.meta.env.VITE_API_PREFIX || ""}/v1/public/user/traffic_stats`,
|
||||
{
|
||||
method: "GET",
|
||||
params: {
|
||||
...params,
|
||||
},
|
||||
...(options || {}),
|
||||
}
|
||||
);
|
||||
}
|
||||
1
packages/ui/src/services/user/typings.d.ts
vendored
1
packages/ui/src/services/user/typings.d.ts
vendored
@ -1184,6 +1184,7 @@ declare namespace API {
|
||||
|
||||
type UserSubscribe = {
|
||||
id: number;
|
||||
id_str: string;
|
||||
user_id: number;
|
||||
order_id: number;
|
||||
subscribe_id: number;
|
||||
|
||||
Loading…
x
Reference in New Issue
Block a user