✨ 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",
|
"validationFailed": "Validation failed",
|
||||||
"totalNodeGroups": "Total Node Groups",
|
"totalNodeGroups": "Total Node Groups",
|
||||||
"invalidRange": "Minimum traffic must be less than maximum traffic",
|
"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",
|
"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 ",
|
"speedLimit": "Speed Limit ",
|
||||||
"traffic": "Traffic",
|
"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",
|
"unitPrice": "Unit Price",
|
||||||
"unitTime": "Unit Time",
|
"unitTime": "Unit Time",
|
||||||
"unlimitedInventory": "Unlimited (enter -1)",
|
"unlimitedInventory": "Unlimited (enter -1)",
|
||||||
|
|||||||
@ -182,6 +182,20 @@
|
|||||||
"validationFailed": "验证失败",
|
"validationFailed": "验证失败",
|
||||||
"totalNodeGroups": "总节点组数",
|
"totalNodeGroups": "总节点组数",
|
||||||
"invalidRange": "最小流量必须小于最大流量",
|
"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": "开启后,在订阅卡片上将会显示原价和折后价,帮助用户了解优惠幅度",
|
"showOriginalPriceDescription": "开启后,在订阅卡片上将会显示原价和折后价,帮助用户了解优惠幅度",
|
||||||
"speedLimit": "速度限制",
|
"speedLimit": "速度限制",
|
||||||
"traffic": "流量",
|
"traffic": "流量",
|
||||||
|
"trafficLimit": "按量限速",
|
||||||
|
"trafficLimitRules": "按量限速规则",
|
||||||
|
"trafficLimitDescription": "配置基于流量的限速规则。当流量使用达到指定量时,将进行限速。",
|
||||||
|
"addTrafficLimitRule": "添加限速规则",
|
||||||
|
"statType": "统计类型",
|
||||||
|
"selectStatType": "选择类型...",
|
||||||
|
"statTypeHour": "小时",
|
||||||
|
"statTypeDay": "天",
|
||||||
|
"statValue": "时间值",
|
||||||
|
"trafficUsage": "使用流量(GB)",
|
||||||
|
"speedLimitKb": "限速(kb)",
|
||||||
"unitPrice": "单价",
|
"unitPrice": "单价",
|
||||||
"unitTime": "时间单位",
|
"unitTime": "时间单位",
|
||||||
"unlimitedInventory": "无限制(输入 -1)",
|
"unlimitedInventory": "无限制(输入 -1)",
|
||||||
|
|||||||
@ -40,6 +40,10 @@ const NodeGroupForm = forwardRef<
|
|||||||
description: "",
|
description: "",
|
||||||
sort: 0,
|
sort: 0,
|
||||||
for_calculation: true,
|
for_calculation: true,
|
||||||
|
is_expired_group: false,
|
||||||
|
expired_days_limit: 7,
|
||||||
|
max_traffic_gb_expired: 0,
|
||||||
|
speed_limit: 0,
|
||||||
min_traffic_gb: 0,
|
min_traffic_gb: 0,
|
||||||
max_traffic_gb: 0,
|
max_traffic_gb: 0,
|
||||||
});
|
});
|
||||||
@ -53,6 +57,10 @@ const NodeGroupForm = forwardRef<
|
|||||||
description: initialValues.description || "",
|
description: initialValues.description || "",
|
||||||
sort: initialValues.sort ?? 0,
|
sort: initialValues.sort ?? 0,
|
||||||
for_calculation: initialValues.for_calculation ?? true,
|
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,
|
min_traffic_gb: initialValues.min_traffic_gb ?? 0,
|
||||||
max_traffic_gb: initialValues.max_traffic_gb ?? 0,
|
max_traffic_gb: initialValues.max_traffic_gb ?? 0,
|
||||||
});
|
});
|
||||||
@ -62,6 +70,10 @@ const NodeGroupForm = forwardRef<
|
|||||||
description: "",
|
description: "",
|
||||||
sort: 0,
|
sort: 0,
|
||||||
for_calculation: true,
|
for_calculation: true,
|
||||||
|
is_expired_group: false,
|
||||||
|
expired_days_limit: 7,
|
||||||
|
max_traffic_gb_expired: 0,
|
||||||
|
speed_limit: 0,
|
||||||
min_traffic_gb: 0,
|
min_traffic_gb: 0,
|
||||||
max_traffic_gb: 0,
|
max_traffic_gb: 0,
|
||||||
});
|
});
|
||||||
@ -118,15 +130,68 @@ const NodeGroupForm = forwardRef<
|
|||||||
return "";
|
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) => {
|
const handleSubmit = async (e: React.FormEvent) => {
|
||||||
e.preventDefault();
|
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);
|
const conflict = checkTrafficRangeConflict(values.min_traffic_gb, values.max_traffic_gb);
|
||||||
if (conflict) {
|
if (conflict) {
|
||||||
setConflictError(conflict);
|
setConflictError(conflict);
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
|
||||||
setSubmitting(true);
|
setSubmitting(true);
|
||||||
const success = await onSubmit(values);
|
const success = await onSubmit(values);
|
||||||
@ -139,6 +204,10 @@ const NodeGroupForm = forwardRef<
|
|||||||
description: "",
|
description: "",
|
||||||
sort: 0,
|
sort: 0,
|
||||||
for_calculation: true,
|
for_calculation: true,
|
||||||
|
is_expired_group: false,
|
||||||
|
expired_days_limit: 7,
|
||||||
|
max_traffic_gb_expired: 0,
|
||||||
|
speed_limit: 0,
|
||||||
min_traffic_gb: 0,
|
min_traffic_gb: 0,
|
||||||
max_traffic_gb: 0,
|
max_traffic_gb: 0,
|
||||||
});
|
});
|
||||||
@ -207,18 +276,107 @@ const NodeGroupForm = forwardRef<
|
|||||||
{t("forCalculation", "For Calculation")}
|
{t("forCalculation", "For Calculation")}
|
||||||
</Label>
|
</Label>
|
||||||
<p className="text-sm text-muted-foreground">
|
<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>
|
</p>
|
||||||
</div>
|
</div>
|
||||||
<Switch
|
<Switch
|
||||||
id="for_calculation"
|
id="for_calculation"
|
||||||
checked={values.for_calculation}
|
checked={values.for_calculation}
|
||||||
|
disabled={values.is_expired_group}
|
||||||
onCheckedChange={(checked) =>
|
onCheckedChange={(checked) =>
|
||||||
setValues({ ...values, for_calculation: checked })
|
setValues({ ...values, for_calculation: checked })
|
||||||
}
|
}
|
||||||
/>
|
/>
|
||||||
</div>
|
</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="space-y-2">
|
||||||
<div className="flex items-center justify-between">
|
<div className="flex items-center justify-between">
|
||||||
<Label>{t("trafficRangeGB", "Traffic Range (GB)")}</Label>
|
<Label>{t("trafficRangeGB", "Traffic Range (GB)")}</Label>
|
||||||
@ -270,6 +428,7 @@ const NodeGroupForm = forwardRef<
|
|||||||
</div>
|
</div>
|
||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
|
)}
|
||||||
|
|
||||||
<div className="flex justify-end gap-2">
|
<div className="flex justify-end gap-2">
|
||||||
<button
|
<button
|
||||||
|
|||||||
@ -77,6 +77,19 @@ export default function NodeGroups() {
|
|||||||
id: "name",
|
id: "name",
|
||||||
accessorKey: "name",
|
accessorKey: "name",
|
||||||
header: t("name", "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",
|
id: "description",
|
||||||
@ -99,7 +112,6 @@ export default function NodeGroups() {
|
|||||||
},
|
},
|
||||||
{
|
{
|
||||||
id: "traffic_range",
|
id: "traffic_range",
|
||||||
accessorKey: "traffic_range",
|
|
||||||
header: t("trafficRange", "Traffic Range (GB)"),
|
header: t("trafficRange", "Traffic Range (GB)"),
|
||||||
cell: ({ row }: { row: any }) => {
|
cell: ({ row }: { row: any }) => {
|
||||||
const min = row.original.min_traffic_gb;
|
const min = row.original.min_traffic_gb;
|
||||||
|
|||||||
@ -84,6 +84,7 @@ const defaultValues = {
|
|||||||
renewal_reset: false,
|
renewal_reset: false,
|
||||||
show_original_price: false,
|
show_original_price: false,
|
||||||
deduction_mode: "auto",
|
deduction_mode: "auto",
|
||||||
|
traffic_limit: [],
|
||||||
};
|
};
|
||||||
|
|
||||||
export default function SubscribeForm<T extends Record<string, any>>({
|
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(),
|
reset_cycle: z.number().optional(),
|
||||||
renewal_reset: z.boolean().optional(),
|
renewal_reset: z.boolean().optional(),
|
||||||
show_original_price: 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>>({
|
const form = useForm<z.infer<typeof formSchema>>({
|
||||||
@ -283,12 +294,14 @@ export default function SubscribeForm<T extends Record<string, any>>({
|
|||||||
|
|
||||||
const tagGroups = getAllAvailableTags();
|
const tagGroups = getAllAvailableTags();
|
||||||
|
|
||||||
// Fetch node groups
|
// Fetch node groups (exclude expired groups)
|
||||||
const { data: nodeGroupsData } = useQuery({
|
const { data: nodeGroupsData } = useQuery({
|
||||||
queryKey: ["nodeGroups"],
|
queryKey: ["nodeGroups"],
|
||||||
queryFn: async () => {
|
queryFn: async () => {
|
||||||
const { data } = await getNodeGroupList({ page: 1, size: 1000 });
|
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 {...form}>
|
||||||
<form className="pt-4" onSubmit={form.handleSubmit(handleSubmit)}>
|
<form className="pt-4" onSubmit={form.handleSubmit(handleSubmit)}>
|
||||||
<Tabs className="w-full" defaultValue="basic">
|
<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
|
<TabsTrigger
|
||||||
className="flex items-center gap-2"
|
className="flex items-center gap-2"
|
||||||
value="basic"
|
value="basic"
|
||||||
@ -366,6 +379,13 @@ export default function SubscribeForm<T extends Record<string, any>>({
|
|||||||
<Server className="h-4 w-4" />
|
<Server className="h-4 w-4" />
|
||||||
{t("form.nodes")}
|
{t("form.nodes")}
|
||||||
</TabsTrigger>
|
</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>
|
</TabsList>
|
||||||
|
|
||||||
<TabsContent className="space-y-4" value="basic">
|
<TabsContent className="space-y-4" value="basic">
|
||||||
@ -1396,6 +1416,91 @@ export default function SubscribeForm<T extends Record<string, any>>({
|
|||||||
)}
|
)}
|
||||||
</div>
|
</div>
|
||||||
</TabsContent>
|
</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>
|
</Tabs>
|
||||||
</form>
|
</form>
|
||||||
</Form>
|
</Form>
|
||||||
|
|||||||
@ -31,12 +31,14 @@ export default function SubscribeTable() {
|
|||||||
const ref = useRef<ProTableActions>(null);
|
const ref = useRef<ProTableActions>(null);
|
||||||
const { fetchSubscribes } = useSubscribe();
|
const { fetchSubscribes } = useSubscribe();
|
||||||
|
|
||||||
// Fetch node groups for filtering
|
// Fetch node groups for filtering (exclude expired groups)
|
||||||
const { data: nodeGroupsData } = useQuery({
|
const { data: nodeGroupsData } = useQuery({
|
||||||
queryKey: ["nodeGroups"],
|
queryKey: ["nodeGroups"],
|
||||||
queryFn: async () => {
|
queryFn: async () => {
|
||||||
const { data } = await getNodeGroupList({ page: 1, size: 1000 });
|
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",
|
"server": "Server Management",
|
||||||
"subscribe": "Subscribe",
|
"subscribe": "Subscribe",
|
||||||
"ticket": "Ticket Management",
|
"ticket": "Ticket Management",
|
||||||
|
"traffic": "Traffic Statistics",
|
||||||
"wallet": "Balance"
|
"wallet": "Balance"
|
||||||
},
|
},
|
||||||
"pagination": {
|
"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": "服务器管理",
|
"server": "服务器管理",
|
||||||
"subscribe": "订阅管理",
|
"subscribe": "订阅管理",
|
||||||
"ticket": "工单管理",
|
"ticket": "工单管理",
|
||||||
|
"traffic": "流量统计",
|
||||||
"wallet": "余额管理"
|
"wallet": "余额管理"
|
||||||
},
|
},
|
||||||
"pagination": {
|
"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",
|
icon: "uil:shop",
|
||||||
title: t("menu.subscribe", "Subscribe"),
|
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/',
|
'/(main)/purchasing/',
|
||||||
)()
|
)()
|
||||||
const mainuserWalletLazyRouteImport = createFileRoute('/(main)/(user)/wallet')()
|
const mainuserWalletLazyRouteImport = createFileRoute('/(main)/(user)/wallet')()
|
||||||
|
const mainuserTrafficLazyRouteImport = createFileRoute(
|
||||||
|
'/(main)/(user)/traffic',
|
||||||
|
)()
|
||||||
const mainuserTicketLazyRouteImport = createFileRoute('/(main)/(user)/ticket')()
|
const mainuserTicketLazyRouteImport = createFileRoute('/(main)/(user)/ticket')()
|
||||||
const mainuserSubscribeLazyRouteImport = createFileRoute(
|
const mainuserSubscribeLazyRouteImport = createFileRoute(
|
||||||
'/(main)/(user)/subscribe',
|
'/(main)/(user)/subscribe',
|
||||||
@ -126,6 +129,15 @@ const mainuserWalletLazyRoute = mainuserWalletLazyRouteImport
|
|||||||
getParentRoute: () => mainuserRouteLazyRoute,
|
getParentRoute: () => mainuserRouteLazyRoute,
|
||||||
} as any)
|
} as any)
|
||||||
.lazy(() => import('./routes/(main)/(user)/wallet.lazy').then((d) => d.Route))
|
.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
|
const mainuserTicketLazyRoute = mainuserTicketLazyRouteImport
|
||||||
.update({
|
.update({
|
||||||
id: '/ticket',
|
id: '/ticket',
|
||||||
@ -220,6 +232,7 @@ export interface FileRoutesByFullPath {
|
|||||||
'/profile': typeof mainuserProfileLazyRoute
|
'/profile': typeof mainuserProfileLazyRoute
|
||||||
'/subscribe': typeof mainuserSubscribeLazyRoute
|
'/subscribe': typeof mainuserSubscribeLazyRoute
|
||||||
'/ticket': typeof mainuserTicketLazyRoute
|
'/ticket': typeof mainuserTicketLazyRoute
|
||||||
|
'/traffic': typeof mainuserTrafficLazyRoute
|
||||||
'/wallet': typeof mainuserWalletLazyRoute
|
'/wallet': typeof mainuserWalletLazyRoute
|
||||||
'/purchasing': typeof mainPurchasingIndexLazyRoute
|
'/purchasing': typeof mainPurchasingIndexLazyRoute
|
||||||
'/purchasing/order': typeof mainPurchasingOrderIndexRoute
|
'/purchasing/order': typeof mainPurchasingOrderIndexRoute
|
||||||
@ -240,6 +253,7 @@ export interface FileRoutesByTo {
|
|||||||
'/profile': typeof mainuserProfileLazyRoute
|
'/profile': typeof mainuserProfileLazyRoute
|
||||||
'/subscribe': typeof mainuserSubscribeLazyRoute
|
'/subscribe': typeof mainuserSubscribeLazyRoute
|
||||||
'/ticket': typeof mainuserTicketLazyRoute
|
'/ticket': typeof mainuserTicketLazyRoute
|
||||||
|
'/traffic': typeof mainuserTrafficLazyRoute
|
||||||
'/wallet': typeof mainuserWalletLazyRoute
|
'/wallet': typeof mainuserWalletLazyRoute
|
||||||
'/purchasing': typeof mainPurchasingIndexLazyRoute
|
'/purchasing': typeof mainPurchasingIndexLazyRoute
|
||||||
'/purchasing/order': typeof mainPurchasingOrderIndexRoute
|
'/purchasing/order': typeof mainPurchasingOrderIndexRoute
|
||||||
@ -263,6 +277,7 @@ export interface FileRoutesById {
|
|||||||
'/(main)/(user)/profile': typeof mainuserProfileLazyRoute
|
'/(main)/(user)/profile': typeof mainuserProfileLazyRoute
|
||||||
'/(main)/(user)/subscribe': typeof mainuserSubscribeLazyRoute
|
'/(main)/(user)/subscribe': typeof mainuserSubscribeLazyRoute
|
||||||
'/(main)/(user)/ticket': typeof mainuserTicketLazyRoute
|
'/(main)/(user)/ticket': typeof mainuserTicketLazyRoute
|
||||||
|
'/(main)/(user)/traffic': typeof mainuserTrafficLazyRoute
|
||||||
'/(main)/(user)/wallet': typeof mainuserWalletLazyRoute
|
'/(main)/(user)/wallet': typeof mainuserWalletLazyRoute
|
||||||
'/(main)/purchasing/': typeof mainPurchasingIndexLazyRoute
|
'/(main)/purchasing/': typeof mainPurchasingIndexLazyRoute
|
||||||
'/(main)/purchasing/order/': typeof mainPurchasingOrderIndexRoute
|
'/(main)/purchasing/order/': typeof mainPurchasingOrderIndexRoute
|
||||||
@ -285,6 +300,7 @@ export interface FileRouteTypes {
|
|||||||
| '/profile'
|
| '/profile'
|
||||||
| '/subscribe'
|
| '/subscribe'
|
||||||
| '/ticket'
|
| '/ticket'
|
||||||
|
| '/traffic'
|
||||||
| '/wallet'
|
| '/wallet'
|
||||||
| '/purchasing'
|
| '/purchasing'
|
||||||
| '/purchasing/order'
|
| '/purchasing/order'
|
||||||
@ -305,6 +321,7 @@ export interface FileRouteTypes {
|
|||||||
| '/profile'
|
| '/profile'
|
||||||
| '/subscribe'
|
| '/subscribe'
|
||||||
| '/ticket'
|
| '/ticket'
|
||||||
|
| '/traffic'
|
||||||
| '/wallet'
|
| '/wallet'
|
||||||
| '/purchasing'
|
| '/purchasing'
|
||||||
| '/purchasing/order'
|
| '/purchasing/order'
|
||||||
@ -327,6 +344,7 @@ export interface FileRouteTypes {
|
|||||||
| '/(main)/(user)/profile'
|
| '/(main)/(user)/profile'
|
||||||
| '/(main)/(user)/subscribe'
|
| '/(main)/(user)/subscribe'
|
||||||
| '/(main)/(user)/ticket'
|
| '/(main)/(user)/ticket'
|
||||||
|
| '/(main)/(user)/traffic'
|
||||||
| '/(main)/(user)/wallet'
|
| '/(main)/(user)/wallet'
|
||||||
| '/(main)/purchasing/'
|
| '/(main)/purchasing/'
|
||||||
| '/(main)/purchasing/order/'
|
| '/(main)/purchasing/order/'
|
||||||
@ -418,6 +436,13 @@ declare module '@tanstack/react-router' {
|
|||||||
preLoaderRoute: typeof mainuserWalletLazyRouteImport
|
preLoaderRoute: typeof mainuserWalletLazyRouteImport
|
||||||
parentRoute: typeof mainuserRouteLazyRoute
|
parentRoute: typeof mainuserRouteLazyRoute
|
||||||
}
|
}
|
||||||
|
'/(main)/(user)/traffic': {
|
||||||
|
id: '/(main)/(user)/traffic'
|
||||||
|
path: '/traffic'
|
||||||
|
fullPath: '/traffic'
|
||||||
|
preLoaderRoute: typeof mainuserTrafficLazyRouteImport
|
||||||
|
parentRoute: typeof mainuserRouteLazyRoute
|
||||||
|
}
|
||||||
'/(main)/(user)/ticket': {
|
'/(main)/(user)/ticket': {
|
||||||
id: '/(main)/(user)/ticket'
|
id: '/(main)/(user)/ticket'
|
||||||
path: '/ticket'
|
path: '/ticket'
|
||||||
@ -493,6 +518,7 @@ interface mainuserRouteLazyRouteChildren {
|
|||||||
mainuserProfileLazyRoute: typeof mainuserProfileLazyRoute
|
mainuserProfileLazyRoute: typeof mainuserProfileLazyRoute
|
||||||
mainuserSubscribeLazyRoute: typeof mainuserSubscribeLazyRoute
|
mainuserSubscribeLazyRoute: typeof mainuserSubscribeLazyRoute
|
||||||
mainuserTicketLazyRoute: typeof mainuserTicketLazyRoute
|
mainuserTicketLazyRoute: typeof mainuserTicketLazyRoute
|
||||||
|
mainuserTrafficLazyRoute: typeof mainuserTrafficLazyRoute
|
||||||
mainuserWalletLazyRoute: typeof mainuserWalletLazyRoute
|
mainuserWalletLazyRoute: typeof mainuserWalletLazyRoute
|
||||||
}
|
}
|
||||||
|
|
||||||
@ -505,6 +531,7 @@ const mainuserRouteLazyRouteChildren: mainuserRouteLazyRouteChildren = {
|
|||||||
mainuserProfileLazyRoute: mainuserProfileLazyRoute,
|
mainuserProfileLazyRoute: mainuserProfileLazyRoute,
|
||||||
mainuserSubscribeLazyRoute: mainuserSubscribeLazyRoute,
|
mainuserSubscribeLazyRoute: mainuserSubscribeLazyRoute,
|
||||||
mainuserTicketLazyRoute: mainuserTicketLazyRoute,
|
mainuserTicketLazyRoute: mainuserTicketLazyRoute,
|
||||||
|
mainuserTrafficLazyRoute: mainuserTrafficLazyRoute,
|
||||||
mainuserWalletLazyRoute: mainuserWalletLazyRoute,
|
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;
|
description: string;
|
||||||
sort: number;
|
sort: number;
|
||||||
for_calculation: boolean;
|
for_calculation: boolean;
|
||||||
|
is_expired_group: boolean;
|
||||||
|
expired_days_limit: number;
|
||||||
|
max_traffic_gb_expired?: number;
|
||||||
|
speed_limit: number;
|
||||||
min_traffic_gb?: number;
|
min_traffic_gb?: number;
|
||||||
max_traffic_gb?: number;
|
max_traffic_gb?: number;
|
||||||
|
node_count?: number;
|
||||||
created_at: number;
|
created_at: number;
|
||||||
updated_at: number;
|
updated_at: number;
|
||||||
};
|
};
|
||||||
@ -2832,6 +2837,10 @@ declare namespace API {
|
|||||||
description?: string;
|
description?: string;
|
||||||
sort?: number;
|
sort?: number;
|
||||||
for_calculation?: boolean;
|
for_calculation?: boolean;
|
||||||
|
is_expired_group?: boolean;
|
||||||
|
expired_days_limit?: number;
|
||||||
|
max_traffic_gb_expired?: number;
|
||||||
|
speed_limit?: number;
|
||||||
min_traffic_gb?: number;
|
min_traffic_gb?: number;
|
||||||
max_traffic_gb?: number;
|
max_traffic_gb?: number;
|
||||||
};
|
};
|
||||||
@ -2842,6 +2851,10 @@ declare namespace API {
|
|||||||
description?: string;
|
description?: string;
|
||||||
sort?: number;
|
sort?: number;
|
||||||
for_calculation?: boolean;
|
for_calculation?: boolean;
|
||||||
|
is_expired_group?: boolean;
|
||||||
|
expired_days_limit?: number;
|
||||||
|
max_traffic_gb_expired?: number;
|
||||||
|
speed_limit?: number;
|
||||||
min_traffic_gb?: number;
|
min_traffic_gb?: number;
|
||||||
max_traffic_gb?: number;
|
max_traffic_gb?: number;
|
||||||
};
|
};
|
||||||
|
|||||||
@ -9,6 +9,7 @@ import * as payment from "./payment";
|
|||||||
import * as portal from "./portal";
|
import * as portal from "./portal";
|
||||||
import * as subscribe from "./subscribe";
|
import * as subscribe from "./subscribe";
|
||||||
import * as ticket from "./ticket";
|
import * as ticket from "./ticket";
|
||||||
|
import * as traffic from "./traffic";
|
||||||
import * as user from "./user";
|
import * as user from "./user";
|
||||||
export default {
|
export default {
|
||||||
announcement,
|
announcement,
|
||||||
@ -18,5 +19,6 @@ export default {
|
|||||||
portal,
|
portal,
|
||||||
subscribe,
|
subscribe,
|
||||||
ticket,
|
ticket,
|
||||||
|
traffic,
|
||||||
user,
|
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 = {
|
type UserSubscribe = {
|
||||||
id: number;
|
id: number;
|
||||||
|
id_str: string;
|
||||||
user_id: number;
|
user_id: number;
|
||||||
order_id: number;
|
order_id: number;
|
||||||
subscribe_id: number;
|
subscribe_id: number;
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user