diff --git a/apps/admin/public/assets/locales/en-US/components.json b/apps/admin/public/assets/locales/en-US/components.json
index 09e97b4..964814b 100644
--- a/apps/admin/public/assets/locales/en-US/components.json
+++ b/apps/admin/public/assets/locales/en-US/components.json
@@ -40,6 +40,8 @@
"40005": "You do not have access permission, please contact the administrator if you have any questions.",
"50001": "Corresponding coupon information not found, please check and try again.",
"50002": "The coupon has been used, cannot be used again.",
+ "50003": "This coupon code is not supported by the current purchase plan.",
+ "50004": "Coupon has insufficient remaining uses.",
"60001": "Subscription has expired, please renew before using.",
"60002": "Unable to use the subscription at the moment, please try again later.",
"60003": "An existing subscription is detected. Please cancel it before proceeding.",
diff --git a/apps/admin/public/assets/locales/en-US/group.json b/apps/admin/public/assets/locales/en-US/group.json
new file mode 100644
index 0000000..2c4dd85
--- /dev/null
+++ b/apps/admin/public/assets/locales/en-US/group.json
@@ -0,0 +1,186 @@
+{
+ "actions": "Actions",
+ "autoTrigger": "Auto",
+ "averageMode": "Average Grouping",
+ "cancel": "Cancel",
+ "completed": "Completed",
+ "confirm": "Confirm",
+ "confirmDelete": "Confirm Delete",
+ "config": "Config",
+ "create": "Create",
+ "created": "Created successfully",
+ "createdAt": "Created At",
+ "createNodeGroup": "Create Node Group",
+ "createUserGroup": "Create User Group",
+ "delete": "Delete",
+ "deleted": "Deleted successfully",
+ "deleteNodeGroupConfirm": "This will delete the node group. Nodes in this group will be reassigned.",
+ "deleteUserGroupConfirm": "This will delete the user group. Users in this group will be reassigned to the default group.",
+ "description": "Description",
+ "descriptionPlaceholder": "Enter description",
+ "edit": "Edit",
+ "editNodeGroup": "Edit Node Group",
+ "editUserGroup": "Edit User Group",
+ "editUserGroupDescription": "Edit user group assignment and lock status",
+ "selectGroup": "Select a group",
+ "endTime": "End Time",
+ "errorMessage": "Error Message",
+ "export": "Export",
+ "failed": "Failed",
+ "failedCount": "Failed",
+ "groupConfig": "Group Configuration",
+ "groupConfigDescription": "Manage node groups and automatically assign node groups to user subscriptions",
+ "groupDetails": "Group Details",
+ "groupEnabled": "Group Management Enabled",
+ "groupEnabledDescription": "Enable group management to control user access to nodes",
+ "groupHistory": "Group Calculation History",
+ "groupHistoryDescription": "View group recalculation history and results",
+ "groupHistoryDetail": "Group Calculation Detail",
+ "groupId": "Group ID",
+ "groupIdPlaceholder": "Enter unique group ID",
+ "groupMode": "Group Mode",
+ "groupModeDescription": "Select the grouping algorithm for assigning users to groups",
+ "groupName": "Group Name",
+ "groupNamePlaceholder": "Enter group name",
+ "groupRecalculation": "Group Recalculation",
+ "groupRecalculationDescription": "Manually trigger node group reassignment for all active user subscriptions based on current configuration",
+ "history": "History",
+ "historyId": "History ID",
+ "id": "ID",
+ "idPrefix": "#",
+ "idle": "Idle",
+ "separator": "/",
+ "loading": "Loading...",
+ "loadFailed": "Failed to load configuration",
+ "locked": "Locked",
+ "manualTrigger": "Manual",
+ "name": "Name",
+ "namePlaceholder": "Enter name",
+ "nodeCount": "Node Count",
+ "nodeGroup": "Node Group",
+ "nodeGroupFormDescription": "Configure node group settings",
+ "nodeGroups": "Node Groups",
+ "nodeGroupsDescription": "Manage node groups for user access control",
+ "noDetails": "No details available",
+ "operator": "Operator",
+ "progress": "Progress",
+ "recalculate": "Recalculate",
+ "recalculateAll": "Reassign Node Groups",
+ "recalculationCompleted": "Recalculation completed successfully",
+ "recalculationFailed": "Recalculation failed. Please try again.",
+ "recalculationStarted": "Recalculation started",
+ "recalculationWarning": "Recalculation will reassign node groups for all active user subscriptions based on current configuration. This operation cannot be undone.",
+ "running": "Running",
+ "save": "Save",
+ "scheduleTrigger": "Schedule",
+ "sort": "Sort",
+ "sortOrder": "Sort Order",
+ "startTime": "Start Time",
+ "subscribeMode": "Subscribe-based Grouping",
+ "successCount": "Success",
+ "title": "Group Management",
+ "totalUsers": "Total Users",
+ "totalNodes": "Total Nodes",
+ "totalGroups": "Total Groups",
+ "trafficMode": "Traffic-based Grouping",
+ "triggerType": "Trigger Type",
+ "userGroup": "User Group",
+ "userGroups": "User Groups",
+ "userGroupsDescription": "Manage user groups for node access control",
+ "updated": "Updated successfully",
+ "updateFailed": "Update failed",
+ "userCount": "User Count",
+ "viewDetail": "View Detail",
+ "warning": "Warning",
+ "yes": "Yes",
+ "no": "No",
+ "saving": "Saving...",
+ "enableGrouping": "Enable Grouping",
+ "enableGroupingDescription": "When enabled, user subscriptions will be automatically assigned node groups based on the distribution mode",
+ "groupingMode": "Grouping Mode",
+ "averageModeConfig": "Average Mode Configuration",
+ "subscribeModeConfig": "Subscribe Mode Configuration",
+ "trafficModeConfig": "Traffic Mode Configuration",
+ "averageModeDescription": "Randomly assign available node groups to active user subscriptions",
+ "subscribeModeDescription": "Set default node group for user groups based on subscription plans",
+ "trafficModeDescription": "Assign node groups to user subscriptions based on traffic usage",
+ "defaultUserGroupId": "Default User Group ID",
+ "defaultUserGroupDescription": "New users will be assigned to this group",
+ "defaultUserGroupForExpiredDescription": "Users with expired subscriptions will be assigned to this group",
+ "autoCreateGroup": "Auto Create Group",
+ "autoCreateGroupDescription": "Automatically create a new user group when a new subscription plan is added",
+ "lockGroup": "Lock Group",
+ "lockGroupDescription": "Prevent automatic recalculation from changing this user's group",
+ "trafficRangesComingSoon": "Traffic ranges configuration coming soon...",
+ "currentStatus": "Current Status",
+ "trafficRangesConfig": "Traffic Ranges Configuration",
+ "trafficRangesDescription": "Configure traffic ranges for grouping users. Traffic is calculated based on user's billing cycle.",
+ "minTrafficGB": "Min Traffic (GB)",
+ "maxTrafficGB": "Max Traffic (GB)",
+ "addRange": "Add Range",
+ "remove": "Remove",
+ "note": "Note",
+ "trafficRangesNote": "Ranges must not overlap and must cover all values without gaps. Users with traffic >= the upper limit of the last range will be assigned to the last group.",
+ "defaultUserGroup": "Default User Group",
+ "defaultUserGroupForTrafficDescription": "Users with traffic exceeding all defined ranges will be assigned to this group",
+ "rangeError": "Range Error",
+ "overlapError": "Overlap Error",
+ "gapError": "Gap Error",
+ "groupByTraffic": "Group by Traffic",
+ "resetGroups": "Reset All Groups",
+ "resetGroupsTitle": "Reset All Groups",
+ "resetGroupsDescription": "This action will delete all node groups and user groups, reset all users' group ID to 0, clear all products' node group IDs, and clear all nodes' node group IDs. This action cannot be undone.",
+ "resetSuccess": "All groups have been reset successfully",
+ "resetFailed": "Failed to reset groups",
+ "saved": "Configuration saved successfully",
+ "saveFailed": "Failed to save configuration",
+ "autoCalculated": "Auto-calculated",
+ "userGroupCountAutoCalculated": "Auto-calculated from actual user groups",
+ "userGroupCount": "User Group Count",
+ "nodeGroupCountAutoCalculated": "Auto-calculated from actual node groups",
+ "nodeGroupCount": "Node Group Count",
+ "arrow": " → ",
+ "availableNodeGroups": "Available Node Groups",
+ "currentGroupingResult": "Current Grouping Result",
+ "calculationInfo": "Calculation Information",
+ "groupingDetailsStatistics": "Grouping Details Statistics",
+ "successFailedCount": "Success/Failed",
+ "latestGroupingCalculation": "Latest grouping calculation details",
+ "userList": "User List",
+ "email": "Email",
+ "noUsers": "No users found",
+ "showing": "Showing",
+ "to": "to",
+ "of": "of",
+ "previous": "Previous",
+ "next": "Next",
+ "result": "Result",
+ "bindNodeGroup": "Bind Node Group",
+ "bindNodeGroupDescription": "Select a node group to bind to user groups: {{userGroups}}",
+ "selectNodeGroup": "Select Node Group",
+ "selectNodeGroupPlaceholder": "Select a node group...",
+ "selectNodeGroupRequired": "Please select a node group",
+ "unbound": "Unbound",
+ "bindSuccess": "Successfully bound {{userGroupCount}} user group(s) to node group",
+ "bindFailed": "Failed to bind node group",
+ "groupMapping": "Group Mapping",
+ "forCalculation": "For Calculation",
+ "trafficRange": "Traffic Range (GB)",
+ "configSaved": "Configuration saved successfully",
+ "subscribeGroupMappingTitle": "Subscribe-Node Group Mapping",
+ "subscribeName": "Subscribe Plan",
+ "userGroupName": "User Group",
+ "nodeGroupName": "Node Group",
+ "notMapped": "Not Mapped",
+ "noMappingData": "No mapping data available",
+ "forCalculationDescription": "Whether this node group participates in grouping calculation",
+ "trafficRangeGB": "Traffic Range (GB)",
+ "trafficRangeDescription": "Users with traffic >= Min and < Max will be assigned to this node group",
+ "minCannotExceedMax": "Minimum traffic cannot exceed maximum traffic",
+ "rangeOverlap": "Range overlaps with node group \"{{name}}\"",
+ "nodeGroupNotFound": "Node group not found",
+ "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)"
+}
diff --git a/apps/admin/public/assets/locales/en-US/menu.json b/apps/admin/public/assets/locales/en-US/menu.json
index 4c17948..b2c12ed 100644
--- a/apps/admin/public/assets/locales/en-US/menu.json
+++ b/apps/admin/public/assets/locales/en-US/menu.json
@@ -10,6 +10,7 @@
"Document Management": "Document Management",
"Email": "Email",
"Gift": "Gift",
+ "Group Management": "Group Management",
"Login": "Login",
"Logs & Analytics": "Logs & Analytics",
"Maintenance": "Maintenance",
diff --git a/apps/admin/public/assets/locales/en-US/nodes.json b/apps/admin/public/assets/locales/en-US/nodes.json
index 05ed877..67b4fe6 100644
--- a/apps/admin/public/assets/locales/en-US/nodes.json
+++ b/apps/admin/public/assets/locales/en-US/nodes.json
@@ -1,5 +1,6 @@
{
"address": "Address",
+ "all": "All",
"cancel": "Cancel",
"confirm": "Confirm",
"confirmDeleteDesc": "This action cannot be undone.",
@@ -17,15 +18,21 @@
"enabled_off": "Disabled",
"enabled_on": "Enabled",
"name": "Name",
+ "nodeGroup": "Node Group",
+ "nodeGroups": "Node Groups",
+ "nodeGroup_description": "Assign this node to multiple groups for user access control.",
"pageTitle": "Nodes",
"port": "Port",
"protocol": "Protocol",
+ "public": "Public",
+ "selectNodeGroup": "Select node group…",
"select_protocol": "Select protocol…",
"select_server": "Select server…",
"server": "Server",
"sorted_success": "Sorted successfully",
"tags": "Tags",
"tags_description": "Permission grouping tag (incl. plan binding and delivery policies).",
+ "tags_groupMode_description": "Optional tags for display and filtering (node group name will be used as tag if empty).",
"tags_placeholder": "Use Enter or comma (,) to add multiple tags",
"updated": "Updated"
}
diff --git a/apps/admin/public/assets/locales/en-US/product.json b/apps/admin/public/assets/locales/en-US/product.json
index f590049..5c19842 100644
--- a/apps/admin/public/assets/locales/en-US/product.json
+++ b/apps/admin/public/assets/locales/en-US/product.json
@@ -1,4 +1,5 @@
{
+ "all": "All",
"cancel": "Cancel",
"confirm": "Confirm",
"confirmDelete": "Are you sure you want to delete?",
@@ -7,13 +8,16 @@
"create": "Create",
"createSubscribe": "Create Subscription",
"createSuccess": "Create Successful",
+ "currentUserGroup": "Current User Group",
+ "defaultNodeGroup": "Default Node Group",
"delete": "Delete",
+ "nodeGroups": "Node Groups",
+ "nodes": "nodes",
"deleteSuccess": "Delete Successful",
"deleteWarning": "Data cannot be recovered after deletion. Please proceed with caution.",
"deviceLimit": "IP Limit",
"edit": "Edit",
"editSubscribe": "Edit Subscription",
- "sortSuccess": "Sort completed successfully",
"form": {
"annualReset": "Annual Reset",
"basic": "Basic",
@@ -30,7 +34,6 @@
"discountPercent": "Discount Percentage",
"Hour": "Hour",
"inventory": "Subscription Limit",
- "unlimitedInventory": "Unlimited (enter -1)",
"language": "Language",
"languageDescription": "Leave empty for default without language restriction",
"languagePlaceholder": "Language identifier for the subscription, e.g., en-US, zh-CN",
@@ -40,7 +43,19 @@
"name": "Name",
"node": "Node",
"nodeGroup": "Node Group",
- "nodes": "Nodes",
+ "nodeGroups": "Node Groups",
+ "nodeGroupsDescription": "Assign this product to multiple node groups. Users will get nodes from these groups.",
+ "nodeGroupsFirstSelectionDescription": "Select node groups for this product. The first selected group will be set as the default node group.",
+ "defaultNodeGroup": "Default Node Group",
+ "defaultNodeGroupDescription": "Select the default node group for this product. This will be automatically included in the backup node groups.",
+ "selectDefaultNodeGroup": "Select a default node group...",
+ "noDefaultNodeGroup": "No Default Node Group",
+ "backupNodeGroups": "Backup Node Groups",
+ "backupNodeGroupsDescription": "Select additional backup node groups. The default node group is automatically included.",
+ "nodes": "Linked Nodes",
+ "nodesDescription": "Select nodes for this subscription",
+ "nodesInGroup": "Nodes in this group:",
+ "nodesWithoutGroupsDescription": "Nodes without group assignment will be shown here (nodes that belong to groups are managed in the Node Groups section above)",
"noLimit": "No Limit",
"NoLimit": "No Limit",
"noReset": "No Reset",
@@ -61,16 +76,42 @@
"traffic": "Traffic",
"unitPrice": "Unit Price",
"unitTime": "Unit Time",
+ "unlimitedInventory": "Unlimited (enter -1)",
"Year": "Year"
},
+ "groupMapping": "Group Mapping",
+ "groupMappingTitle": "Group Mapping",
+ "groupMappingUpdateFailed": "Failed to update group mapping",
+ "groupMappingUpdateSuccess": "Group mapping updated successfully",
+ "migrateUsers": "Migrate Users",
+ "migrateUsersTitle": "Migrate Users",
+ "migrateUsersDescription": "Migrate all users from the current user group to another group",
+ "migrateUsersWarning": "This will migrate {count} users from \"{group}\" to the target group. This action cannot be undone.",
+ "migrateUsersSuccess": "Successfully migrated {count} users to the target group",
+ "migrateUsersFailed": "Failed to migrate users",
+ "targetUserGroup": "Target User Group",
+ "selectTargetGroup": "Select a target group...",
+ "selectTargetGroupFirst": "Please select a target group first",
+ "cannotMigrateToSameGroup": "Cannot migrate to the same group",
+ "noSourceGroup": "No source group available",
+ "selectedGroup": "Selected Group",
+ "userCount": "User Count",
+ "migrating": "Migrating...",
"inventory": "Subscription Limit",
"language": "Language",
+ "loading": "Loading...",
"name": "Name",
+ "noMapping": "No mapping set",
+ "noNodes": "No nodes in this group",
"quota": "Purchase Limit/Time",
"replacement": "Reset Price/Time",
+ "save": "Save",
+ "selectGroupPlaceholder": "Select a group...",
+ "selectUserGroup": "Select User Group",
"sell": "Sell",
"show": "Display",
"sold": "Subscription Count",
+ "sortSuccess": "Sort completed successfully",
"traffic": "Traffic",
"unitPrice": "Unit Price",
"updateSuccess": "Update Successful"
diff --git a/apps/admin/public/assets/locales/en-US/redemption.json b/apps/admin/public/assets/locales/en-US/redemption.json
index 1abed9d..be07826 100644
--- a/apps/admin/public/assets/locales/en-US/redemption.json
+++ b/apps/admin/public/assets/locales/en-US/redemption.json
@@ -1,5 +1,4 @@
{
- "active": "Active",
"cancel": "Cancel",
"code": "Redemption Code",
"confirm": "Confirm",
@@ -13,7 +12,6 @@
"duration": "Redemption Duration",
"edit": "Edit",
"editRedemptionCode": "Edit Redemption Code",
- "exhausted": "Exhausted",
"form": {
"batchCount": "Batch Count",
"batchCountPlaceholder": "Batch Count",
@@ -26,16 +24,12 @@
"halfYear": "Half Year",
"month": "Month",
"quarter": "Quarter",
- "quantityRequired": "Quantity is required",
"selectPlan": "Select Redemption Plan",
"selectUnitTime": "Select Redemption Duration Unit",
"subscribePlan": "Redemption Plan",
- "subscribePlanRequired": "Subscribe plan is required",
"totalCount": "Available Uses",
"totalCountPlaceholder": "Available Uses",
- "totalCountRequired": "Total count is required",
"unitTime": "Redemption Duration Unit",
- "unitTimeRequired": "Unit time is required",
"year": "Year"
},
"id": "ID",
@@ -50,8 +44,8 @@
"status": "Status",
"subscribeId": "Subscribe ID",
"subscribePlan": "Redemption Plan",
- "totalCount": "Available Uses",
"total": "Total",
+ "totalCount": "Available Uses",
"unitTime": "Redemption Duration Unit",
"updateSuccess": "Update Success",
"usedCount": "Used",
diff --git a/apps/admin/public/assets/locales/en-US/system.json b/apps/admin/public/assets/locales/en-US/system.json
index ead55ff..7c67319 100644
--- a/apps/admin/public/assets/locales/en-US/system.json
+++ b/apps/admin/public/assets/locales/en-US/system.json
@@ -19,6 +19,7 @@
"description": "Configure currency units, symbols, and exchange rate API settings",
"title": "Currency Configuration"
},
+ "groupSettings": "Group Settings",
"invite": {
"description": "Configure user invitation and referral reward settings",
"forcedInvite": "Require Invitation to Register",
diff --git a/apps/admin/public/assets/locales/en-US/tool.json b/apps/admin/public/assets/locales/en-US/tool.json
index 0bf5b5d..c1c61a4 100644
--- a/apps/admin/public/assets/locales/en-US/tool.json
+++ b/apps/admin/public/assets/locales/en-US/tool.json
@@ -12,6 +12,7 @@
"systemReboot": "System Reboot",
"systemServices": "System Services",
"update": "Update",
+ "updateDescription": "Are you sure you want to update?",
"updateFailed": "Update failed",
"updateServerDescription": "Are you sure you want to update the server version from {{current}} to {{latest}}?",
"updateSuccess": "Update completed successfully",
diff --git a/apps/admin/public/assets/locales/en-US/translation.json b/apps/admin/public/assets/locales/en-US/translation.json
index d8c04fb..30bb51d 100644
--- a/apps/admin/public/assets/locales/en-US/translation.json
+++ b/apps/admin/public/assets/locales/en-US/translation.json
@@ -7,6 +7,10 @@
"serverRequired": "Please select a server"
},
"form": {
+ "quantityRequired": "Quantity is required",
+ "subscribePlanRequired": "Subscribe plan is required",
+ "totalCountRequired": "Total count is required",
+ "unitTimeRequired": "Unit time is required",
"validation": {
"nameRequired": "Client name is required",
"userAgentRequiredSuffix": "is required"
diff --git a/apps/admin/public/assets/locales/en-US/user.json b/apps/admin/public/assets/locales/en-US/user.json
index 71a2bbe..7dfb941 100644
--- a/apps/admin/public/assets/locales/en-US/user.json
+++ b/apps/admin/public/assets/locales/en-US/user.json
@@ -2,6 +2,7 @@
"accountEnable": "Account Enable",
"add": "Add",
"administrator": "Administrator",
+ "all": "All",
"areaCodePlaceholder": "Area code",
"authMethodsTitle": "Auth Methods",
"avatar": "Avatar",
@@ -17,6 +18,9 @@
"confirm": "Confirm",
"confirmDelete": "Confirm Delete",
"confirmOffline": "Confirm Offline",
+ "confirmResetToken": "Confirm Reset Subscription Address",
+ "confirmResumeSubscribe": "Confirm Resume Subscription",
+ "confirmStopSubscribe": "Confirm Stop Subscription",
"copySubscription": "Copy Subscription",
"copySuccess": "Copied successfully",
"create": "Create",
@@ -29,12 +33,14 @@
"deleteDescription": "This action cannot be undone.",
"deleteSubscriptionDescription": "This action cannot be undone.",
"deleteSuccess": "Deleted successfully",
- "isDeleted": "Status",
"deviceLimit": "Device Limit",
"download": "Download",
"downloadTraffic": "Download Traffic",
"edit": "Edit",
+ "editGroup": "Edit Group",
"editSubscription": "Edit Subscription",
+ "editUserGroup": "Edit User Group",
+ "editUserGroupDescription": "Edit user group assignment and lock status",
"enable": "Enable",
"expiredAt": "Expired At",
"expireTime": "expireTime",
@@ -44,6 +50,7 @@
"invalidEmailFormat": "Invalid email format",
"inviteCode": "Invite Code",
"inviteCodePlaceholder": "Enter invite code",
+ "isDeleted": "Status",
"kickOfflineConfirm": "kickOfflineConfirm",
"kickOfflineSuccess": "Device kicked offline",
"lastSeen": "Last Seen",
@@ -73,37 +80,29 @@
"referrerUserId": "Referrer User ID",
"remove": "Remove",
"resetLogs": "Reset Logs",
- "resetTraffic": "Reset Traffic",
- "toggleStatus": "Toggle Status",
- "resetSubscriptionToken": "Reset Token",
- "resetSubscriptionTokenDescription": "This will reset the subscription token. Old links will become invalid.",
- "resetSubscriptionTraffic": "Reset Traffic",
- "resetSubscriptionTrafficDescription": "This will reset the subscription traffic counters.",
- "toggleSubscriptionStatus": "Toggle Status",
- "toggleSubscriptionStatusDescription": "This will toggle the subscription status.",
"resetTime": "Reset Time",
"resetToken": "Reset Subscription Address",
+ "saving": "Saving...",
"resetTokenDescription": "This will reset the subscription address and regenerate a new token.",
"resetTokenSuccess": "Subscription address reset successfully",
- "confirmResetToken": "Confirm Reset Subscription Address",
+ "resumeSubscribe": "Resume Subscription",
+ "selectGroup": "Select a group",
+ "resumeSubscribeDescription": "This will resume the subscription and allow the user to use it.",
+ "resumeSubscribeSuccess": "Subscription resumed successfully",
+ "save": "Save",
+ "shortCode": "Short Code",
+ "speedLimit": "Speed Limit",
+ "startTime": "startTime",
+ "status": "Status",
+ "statusActive": "Active",
+ "statusDeducted": "Deducted",
+ "statusExpired": "Expired",
+ "statusFinished": "Finished",
+ "statusPending": "Pending",
+ "statusStopped": "Stopped",
"stopSubscribe": "Stop Subscription",
"stopSubscribeDescription": "This will stop the subscription temporarily. User will not be able to use it.",
"stopSubscribeSuccess": "Subscription stopped successfully",
- "confirmStopSubscribe": "Confirm Stop Subscription",
- "resumeSubscribe": "Resume Subscription",
- "resumeSubscribeDescription": "This will resume the subscription and allow the user to use it.",
- "resumeSubscribeSuccess": "Subscription resumed successfully",
- "confirmResumeSubscribe": "Confirm Resume Subscription",
- "status": "Status",
- "statusPending": "Pending",
- "statusActive": "Active",
- "statusFinished": "Finished",
- "statusExpired": "Expired",
- "statusDeducted": "Deducted",
- "statusStopped": "Stopped",
- "save": "Save",
- "speedLimit": "Speed Limit",
- "startTime": "startTime",
"subscription": "Subscription",
"subscriptionId": "subscriptionId",
"subscriptionInfo": "subscriptionInfo",
@@ -119,11 +118,13 @@
"trafficDetails": "Traffic Details",
"trafficLimit": "Traffic Limit",
"trafficStats": "Traffic Stats",
- "trafficUsage": "trafficUsage",
+ "trafficUsage": "Traffic Usage",
+ "remainingTraffic": "Remaining Traffic",
"unlimited": "unlimited",
"unverified": "Unverified",
"update": "Update",
"updateSuccess": "Updated successfully",
+ "groupUpdated": "Group updated successfully",
"upload": "Upload",
"uploadTraffic": "Upload Traffic",
"userAgent": "User Agent",
@@ -134,5 +135,17 @@
"userList": "User List",
"userName": "Username",
"userProfile": "User Profile",
- "verified": "Verified"
+ "userGroup": "User Group",
+ "verified": "Verified",
+ "locked": "Locked",
+ "lockGroup": "Lock Group",
+ "lockGroupDescription": "Prevent automatic grouping from changing this user's group",
+ "groupLocked": "Group Locked",
+ "previewNodes": "Preview Nodes",
+ "availableNodes": "Available Nodes",
+ "name": "Name",
+ "address": "Address",
+ "noNodesAvailable": "No nodes available",
+ "nodeGroup": "Node Group",
+ "publicNodes": "Public Nodes"
}
diff --git a/apps/admin/public/assets/locales/zh-CN/components.json b/apps/admin/public/assets/locales/zh-CN/components.json
index 212f3c6..c0d0b17 100644
--- a/apps/admin/public/assets/locales/zh-CN/components.json
+++ b/apps/admin/public/assets/locales/zh-CN/components.json
@@ -40,6 +40,8 @@
"40005": "您没有访问权限,如有疑问请联系管理员。",
"50001": "找不到对应的优惠券信息,请检查后重试。",
"50002": "该优惠券已被使用,无法再次使用。",
+ "50003": "",
+ "50004": "",
"60001": "订阅已过期,请续费后使用。",
"60002": "暂时无法使用该订阅,请稍后再试。",
"60003": "检测到现有订阅,请先取消后再继续。",
diff --git a/apps/admin/public/assets/locales/zh-CN/group.json b/apps/admin/public/assets/locales/zh-CN/group.json
new file mode 100644
index 0000000..396197f
--- /dev/null
+++ b/apps/admin/public/assets/locales/zh-CN/group.json
@@ -0,0 +1,187 @@
+{
+ "actions": "操作",
+ "autoTrigger": "自动",
+ "averageMode": "平均分组",
+ "cancel": "取消",
+ "completed": "已完成",
+ "confirm": "确认",
+ "confirmDelete": "确认删除",
+ "config": "配置",
+ "create": "创建",
+ "created": "创建成功",
+ "createdAt": "创建时间",
+ "createNodeGroup": "创建节点组",
+ "createUserGroup": "创建用户组",
+ "delete": "删除",
+ "deleted": "删除成功",
+ "deleteNodeGroupConfirm": "此操作将删除节点组。该组中的节点将被重新分配。",
+ "deleteUserGroupConfirm": "此操作将删除用户组。该组中的用户将被重新分配到默认组。",
+ "description": "描述",
+ "descriptionPlaceholder": "输入描述",
+ "edit": "编辑",
+ "editNodeGroup": "编辑节点组",
+ "editUserGroup": "编辑用户组",
+ "editUserGroupDescription": "编辑用户组分配和锁定状态",
+ "selectGroup": "选择一个组",
+ "endTime": "结束时间",
+ "errorMessage": "错误信息",
+ "export": "导出",
+ "failed": "失败",
+ "failedCount": "失败",
+ "groupConfig": "分组配置",
+ "groupConfigDescription": "管理节点组并自动为用户订阅分配节点组",
+ "groupDetails": "分组详情",
+ "groupEnabled": "启用分组管理",
+ "groupEnabledDescription": "启用分组管理以控制用户对节点的访问",
+ "groupHistory": "分组计算历史",
+ "groupHistoryDescription": "查看分组重算历史和结果",
+ "groupHistoryDetail": "分组计算详情",
+ "groupId": "分组ID",
+ "groupIdPlaceholder": "输入唯一的分组ID",
+ "groupMode": "分组模式",
+ "groupModeDescription": "选择分组算法以将用户分配到组",
+ "groupName": "分组名称",
+ "groupNamePlaceholder": "输入分组名称",
+ "groupRecalculation": "分组重新计算",
+ "groupRecalculationDescription": "根据当前配置手动触发所有有效用户订阅的节点组重新分配",
+ "history": "历史记录",
+ "historyId": "历史ID",
+ "id": "ID",
+ "idPrefix": "#",
+ "idle": "空闲",
+ "separator": ",",
+ "loading": "加载中...",
+ "loadFailed": "加载配置失败",
+ "locked": "已锁定",
+ "manualTrigger": "手动",
+ "name": "名称",
+ "namePlaceholder": "输入名称",
+ "nodeCount": "节点数",
+ "nodeGroup": "节点组",
+ "nodeGroupFormDescription": "配置节点组设置",
+ "nodeGroups": "节点组",
+ "nodeGroupsDescription": "管理节点组以控制用户访问权限",
+ "noDetails": "暂无详情",
+ "operator": "操作人",
+ "progress": "进度",
+ "recalculate": "重新计算",
+ "recalculateAll": "重新分配节点组",
+ "recalculationCompleted": "重新计算成功完成",
+ "recalculationFailed": "重新计算失败,请重试。",
+ "recalculationStarted": "重新计算已启动",
+ "recalculationWarning": "重新计算将根据当前配置重新分配所有有效用户订阅的节点组。此操作无法撤消。",
+ "running": "运行中",
+ "save": "保存",
+ "scheduleTrigger": "定时",
+ "sort": "排序",
+ "sortOrder": "排序顺序",
+ "startTime": "开始时间",
+ "subscribeMode": "套餐分组",
+ "successCount": "成功",
+ "title": "分组管理",
+ "totalUsers": "总用户数",
+ "totalNodes": "总节点数",
+ "totalGroups": "总分组数",
+ "trafficMode": "流量分组",
+ "triggerType": "触发类型",
+ "userGroup": "用户组",
+ "userGroups": "用户组",
+ "userGroupsDescription": "管理用户组以控制节点访问权限",
+ "updated": "更新成功",
+ "updateFailed": "更新失败",
+ "userCount": "用户数",
+ "viewDetail": "查看详情",
+ "warning": "警告",
+ "yes": "是",
+ "no": "否",
+ "saving": "保存中...",
+ "enableGrouping": "启用分组",
+ "enableGroupingDescription": "启用后,用户订阅将根据分配模式自动分配节点组",
+ "groupingMode": "分组方式",
+ "averageModeConfig": "平均模式配置",
+ "subscribeModeConfig": "订阅模式配置",
+ "trafficModeConfig": "流量模式配置",
+ "averageModeDescription": "为有效用户订阅随机分配可用节点组",
+ "subscribeModeDescription": "根据订阅套餐设置用户组的默认节点组",
+ "trafficModeDescription": "根据用户订阅的流量使用情况分配节点组",
+ "defaultUserGroupId": "默认用户组ID",
+ "defaultUserGroupDescription": "新用户将被分配到此组",
+ "defaultUserGroupForExpiredDescription": "订阅过期的用户将被分配到此组",
+ "autoCreateGroup": "自动创建组",
+ "autoCreateGroupDescription": "添加新订阅计划时自动创建新的用户组",
+ "lockGroup": "锁定分组",
+ "lockGroupDescription": "防止自动重新计算更改此用户的分组",
+ "trafficRangesComingSoon": "流量区间配置即将推出...",
+ "currentStatus": "当前状态",
+ "trafficRangesConfig": "流量区间配置",
+ "trafficRangesDescription": "配置用户流量分组区间。流量根据用户计费周期计算。",
+ "minTrafficGB": "最小流量 (GB)",
+ "maxTrafficGB": "最大流量 (GB)",
+ "addRange": "添加区间",
+ "remove": "移除",
+ "note": "注意",
+ "trafficRangesNote": "区间不能重叠且必须覆盖所有值而不留空档。流量大于最后一个区间上限的用户将被分配到最后一个组。",
+ "defaultUserGroup": "默认用户组",
+ "defaultUserGroupForTrafficDescription": "超出所有定义区间的用户将被分配到此组",
+ "rangeError": "区间错误",
+ "overlapError": "区间重叠错误",
+ "gapError": "存在空档错误",
+ "groupByTraffic": "按流量分组",
+ "resetGroups": "重置所有分组",
+ "resetGroupsTitle": "重置所有分组",
+ "resetGroupsDescription": "此操作将删除所有节点组和用户组,将所有用户的组ID重置为0,清空所有商品的节点组ID,清空所有节点的节点组ID。此操作无法撤消。",
+ "resetSuccess": "所有分组已成功重置",
+ "resetFailed": "重置分组失败",
+ "saved": "配置保存成功",
+ "saveFailed": "保存配置失败",
+ "autoCalculated": "自动统计",
+ "userGroupCountAutoCalculated": "自动统计实际用户组数量",
+ "userGroupCount": "用户组数",
+ "nodeGroupCountAutoCalculated": "自动统计实际节点组数量",
+ "nodeGroupCount": "节点组数",
+ "arrow": " → ",
+ "availableNodeGroups": "可用节点组",
+ "currentGroupingResult": "当前分组结果",
+ "calculationInfo": "计算信息",
+ "groupingDetailsStatistics": "分组详情统计",
+ "successFailedCount": "成功/失败",
+ "latestGroupingCalculation": "最新分组计算详情",
+ "userList": "用户列表",
+ "email": "邮箱",
+ "noUsers": "未找到用户",
+ "showing": "显示",
+ "to": "至",
+ "of": "共",
+ "previous": "上一页",
+ "next": "下一页",
+ "result": "结果",
+ "bindNodeGroup": "绑定节点组",
+ "bindNodeGroupDescription": "选择一个节点组绑定到以下用户组:{{userGroups}}",
+ "selectNodeGroup": "选择节点组",
+ "selectNodeGroupPlaceholder": "请选择节点组...",
+ "selectNodeGroupRequired": "请选择一个节点组",
+ "unbound": "未绑定",
+ "bindSuccess": "成功将 {{userGroupCount}} 个用户组绑定到节点组",
+ "bindFailed": "绑定节点组失败",
+ "groupMapping": "分组对应关系",
+ "forCalculation": "参与计算",
+ "trafficRange": "流量区间 (GB)",
+ "configSaved": "配置保存成功",
+ "subscribeGroupMappingTitle": "套餐-节点组对应关系",
+ "subscribeName": "订阅计划",
+ "userGroupName": "用户组",
+ "nodeGroupName": "节点组",
+ "notMapped": "未映射",
+ "noMappingData": "暂无映射数据",
+ "forCalculationDescription": "此节点组是否参与分组计算",
+ "trafficRangeGB": "流量区间 (GB)",
+ "trafficRangeDescription": "流量大于等于最小值且小于最大值的用户将被分配到此节点组",
+ "minCannotExceedMax": "最小流量不能超过最大流量",
+ "rangeOverlap": "区间与节点组 \"{{name}}\" 重叠",
+ "nodeGroupNotFound": "未找到节点组",
+ "validationFailed": "验证失败",
+ "totalNodeGroups": "总节点组数",
+ "invalidRange": "最小流量必须小于最大流量",
+ "rangeConflict": "流量区间与节点组 \"{{name}}\" 冲突(区间:{{min}} - {{max}} GB)"
+}
+
diff --git a/apps/admin/public/assets/locales/zh-CN/menu.json b/apps/admin/public/assets/locales/zh-CN/menu.json
index 54fe7ab..4a5b8dd 100644
--- a/apps/admin/public/assets/locales/zh-CN/menu.json
+++ b/apps/admin/public/assets/locales/zh-CN/menu.json
@@ -10,6 +10,7 @@
"Document Management": "文档管理",
"Email": "邮件",
"Gift": "赠送",
+ "Group Management": "分组管理",
"Login": "登录",
"Logs & Analytics": "日志与分析",
"Maintenance": "维护",
diff --git a/apps/admin/public/assets/locales/zh-CN/nodes.json b/apps/admin/public/assets/locales/zh-CN/nodes.json
index a2634e9..3c67650 100644
--- a/apps/admin/public/assets/locales/zh-CN/nodes.json
+++ b/apps/admin/public/assets/locales/zh-CN/nodes.json
@@ -1,5 +1,6 @@
{
"address": "地址",
+ "all": "全部",
"cancel": "取消",
"confirm": "确认",
"confirmDeleteDesc": "此操作无法撤销。",
@@ -17,15 +18,21 @@
"enabled_off": "已禁用",
"enabled_on": "已启用",
"name": "名称",
+ "nodeGroup": "节点分组",
+ "nodeGroups": "节点分组",
+ "nodeGroup_description": "将此节点分配到多个分组以控制用户访问。",
"pageTitle": "节点",
"port": "端口",
"protocol": "协议",
+ "public": "公共",
+ "selectNodeGroup": "选择节点分组…",
"select_protocol": "选择协议…",
"select_server": "选择服务器…",
"server": "服务器",
"sorted_success": "排序成功",
"tags": "标签",
"tags_description": "权限分组标签(包含计划绑定和投递策略)。",
+ "tags_groupMode_description": "可选标签,用于显示和过滤(如果为空,节点组名称将作为标签使用)。",
"tags_placeholder": "使用回车或逗号 (,) 添加多个标签",
"updated": "已更新"
}
diff --git a/apps/admin/public/assets/locales/zh-CN/product.json b/apps/admin/public/assets/locales/zh-CN/product.json
index d8a6e28..d5705b6 100644
--- a/apps/admin/public/assets/locales/zh-CN/product.json
+++ b/apps/admin/public/assets/locales/zh-CN/product.json
@@ -1,4 +1,5 @@
{
+ "all": "全部",
"cancel": "取消",
"confirm": "确认",
"confirmDelete": "确定要删除吗?",
@@ -7,13 +8,16 @@
"create": "创建",
"createSubscribe": "创建订阅",
"createSuccess": "创建成功",
+ "currentUserGroup": "当前用户分组",
+ "defaultNodeGroup": "默认节点组",
"delete": "删除",
+ "nodeGroups": "节点组",
+ "nodes": "个节点",
"deleteSuccess": "删除成功",
"deleteWarning": "删除后数据无法恢复,请谨慎操作。",
"deviceLimit": "IP限制",
"edit": "编辑",
"editSubscribe": "编辑订阅",
- "sortSuccess": "排序成功",
"form": {
"annualReset": "年度重置",
"basic": "基本",
@@ -30,7 +34,6 @@
"discountPercent": "折扣百分比",
"Hour": "小时",
"inventory": "订阅库存",
- "unlimitedInventory": "无限制(输入 -1)",
"language": "语言",
"languageDescription": "留空为默认无语言限制",
"languagePlaceholder": "订阅的语言标识符,例如 en-US、zh-CN",
@@ -40,7 +43,19 @@
"name": "名称",
"node": "节点",
"nodeGroup": "节点组",
- "nodes": "节点",
+ "nodeGroups": "节点组",
+ "nodeGroupsDescription": "将此商品分配到多个节点分组。用户将可以从这些分组获取节点。",
+ "nodeGroupsFirstSelectionDescription": "为此商品选择节点组。第一个选中的组将被设置为默认节点组。",
+ "defaultNodeGroup": "默认节点组",
+ "defaultNodeGroupDescription": "为此商品选择默认节点组。将自动包含在备用节点组中。",
+ "selectDefaultNodeGroup": "选择默认节点组...",
+ "noDefaultNodeGroup": "无默认节点组",
+ "backupNodeGroups": "备用节点组",
+ "backupNodeGroupsDescription": "选择其他备用节点组。默认节点组会自动包含。",
+ "nodes": "关联节点",
+ "nodesDescription": "选择此订阅的节点",
+ "nodesInGroup": "分组中的节点:",
+ "nodesWithoutGroupsDescription": "未分配到分组的节点将在此处显示(属于分组的节点在上方的节点组部分管理)",
"noLimit": "无限制",
"NoLimit": "无限制",
"noReset": "不重置",
@@ -61,16 +76,42 @@
"traffic": "流量",
"unitPrice": "单价",
"unitTime": "时间单位",
+ "unlimitedInventory": "无限制(输入 -1)",
"Year": "年"
},
+ "groupMapping": "分组映射",
+ "groupMappingTitle": "分组映射",
+ "groupMappingUpdateFailed": "更新分组映射失败",
+ "groupMappingUpdateSuccess": "分组映射更新成功",
+ "migrateUsers": "迁移用户",
+ "migrateUsersTitle": "迁移用户",
+ "migrateUsersDescription": "将当前用户组的所有用户迁移到另一个用户组",
+ "migrateUsersWarning": "这将把 {count} 个用户从 \"{group}\" 迁移到目标用户组。此操作无法撤销。",
+ "migrateUsersSuccess": "成功将 {count} 个用户迁移到目标用户组",
+ "migrateUsersFailed": "迁移用户失败",
+ "targetUserGroup": "目标用户组",
+ "selectTargetGroup": "选择目标用户组...",
+ "selectTargetGroupFirst": "请先选择目标用户组",
+ "cannotMigrateToSameGroup": "无法迁移到相同的用户组",
+ "noSourceGroup": "没有可用的源用户组",
+ "selectedGroup": "已选择的分组",
+ "userCount": "用户数量",
+ "migrating": "迁移中...",
"inventory": "订阅库存",
"language": "语言",
+ "loading": "加载中...",
"name": "名称",
+ "noMapping": "未设置映射",
+ "noNodes": "该分组下没有节点",
"quota": "购买限制/次",
"replacement": "重置价格/次",
+ "save": "保存",
+ "selectGroupPlaceholder": "选择分组...",
+ "selectUserGroup": "选择用户分组",
"sell": "销售",
"show": "显示",
"sold": "订阅数量",
+ "sortSuccess": "排序成功",
"traffic": "流量",
"unitPrice": "单价",
"updateSuccess": "更新成功"
diff --git a/apps/admin/public/assets/locales/zh-CN/redemption.json b/apps/admin/public/assets/locales/zh-CN/redemption.json
index 4752261..a74a702 100644
--- a/apps/admin/public/assets/locales/zh-CN/redemption.json
+++ b/apps/admin/public/assets/locales/zh-CN/redemption.json
@@ -1,5 +1,4 @@
{
- "active": "有效",
"cancel": "取消",
"code": "兑换码",
"confirm": "确认",
@@ -13,7 +12,6 @@
"duration": "兑换可用时长",
"edit": "编辑",
"editRedemptionCode": "编辑兑换码",
- "exhausted": "已用尽",
"form": {
"batchCount": "批次数量",
"batchCountPlaceholder": "批次数量",
@@ -26,16 +24,12 @@
"halfYear": "半年",
"month": "月",
"quarter": "季度",
- "quantityRequired": "数量为必填项",
"selectPlan": "选择兑换套餐",
"selectUnitTime": "选择兑换时长单位",
"subscribePlan": "兑换套餐",
- "subscribePlanRequired": "兑换套餐为必填项",
"totalCount": "兑换码可用次数",
"totalCountPlaceholder": "兑换码可用次数",
- "totalCountRequired": "兑换码可用次数为必填项",
"unitTime": "兑换时长单位",
- "unitTimeRequired": "兑换时长单位为必填项",
"year": "年"
},
"id": "ID",
@@ -50,8 +44,8 @@
"status": "状态",
"subscribeId": "套餐ID",
"subscribePlan": "兑换套餐",
- "totalCount": "兑换码可用次数",
"total": "总计",
+ "totalCount": "兑换码可用次数",
"unitTime": "兑换时长单位",
"updateSuccess": "更新成功",
"usedCount": "已使用数量",
diff --git a/apps/admin/public/assets/locales/zh-CN/system.json b/apps/admin/public/assets/locales/zh-CN/system.json
index adfda2d..3b3c01e 100644
--- a/apps/admin/public/assets/locales/zh-CN/system.json
+++ b/apps/admin/public/assets/locales/zh-CN/system.json
@@ -19,6 +19,7 @@
"description": "配置货币单位、符号和汇率 API 设置",
"title": "货币配置"
},
+ "groupSettings": "",
"invite": {
"description": "配置用户邀请和推荐奖励设置",
"forcedInvite": "强制邀请注册",
diff --git a/apps/admin/public/assets/locales/zh-CN/tool.json b/apps/admin/public/assets/locales/zh-CN/tool.json
index 32a6b2a..dad025d 100644
--- a/apps/admin/public/assets/locales/zh-CN/tool.json
+++ b/apps/admin/public/assets/locales/zh-CN/tool.json
@@ -12,6 +12,7 @@
"systemReboot": "系统重启",
"systemServices": "系统服务",
"update": "更新",
+ "updateDescription": "",
"updateFailed": "更新失败",
"updateServerDescription": "确定要将服务器版本从 {{current}} 更新到 {{latest}} 吗?",
"updateSuccess": "更新成功",
diff --git a/apps/admin/public/assets/locales/zh-CN/translation.json b/apps/admin/public/assets/locales/zh-CN/translation.json
index bf51533..4890a5c 100644
--- a/apps/admin/public/assets/locales/zh-CN/translation.json
+++ b/apps/admin/public/assets/locales/zh-CN/translation.json
@@ -7,6 +7,10 @@
"serverRequired": "请选择服务器"
},
"form": {
+ "quantityRequired": "",
+ "subscribePlanRequired": "",
+ "totalCountRequired": "",
+ "unitTimeRequired": "",
"validation": {
"nameRequired": "客户端名称必填",
"userAgentRequiredSuffix": "是必填项"
diff --git a/apps/admin/public/assets/locales/zh-CN/user.json b/apps/admin/public/assets/locales/zh-CN/user.json
index 7d4fb06..1c2a0d0 100644
--- a/apps/admin/public/assets/locales/zh-CN/user.json
+++ b/apps/admin/public/assets/locales/zh-CN/user.json
@@ -2,6 +2,7 @@
"accountEnable": "账户启用",
"add": "添加",
"administrator": "管理员",
+ "all": "全部",
"areaCodePlaceholder": "区号",
"authMethodsTitle": "认证方式",
"avatar": "头像",
@@ -17,6 +18,9 @@
"confirm": "确认",
"confirmDelete": "确认删除",
"confirmOffline": "确认下线",
+ "confirmResetToken": "确认重置订阅地址",
+ "confirmResumeSubscribe": "确认恢复订阅",
+ "confirmStopSubscribe": "确认暂停订阅",
"copySubscription": "复制订阅",
"copySuccess": "复制成功",
"create": "创建",
@@ -29,12 +33,14 @@
"deleteDescription": "此操作无法撤销。",
"deleteSubscriptionDescription": "此操作无法撤销。",
"deleteSuccess": "删除成功",
- "isDeleted": "状态",
"deviceLimit": "IP限制",
"download": "下载",
"downloadTraffic": "下载流量",
"edit": "编辑",
+ "editGroup": "编辑分组",
"editSubscription": "编辑订阅",
+ "editUserGroup": "编辑用户组",
+ "editUserGroupDescription": "编辑用户组分配和锁定状态",
"enable": "启用",
"expiredAt": "过期时间",
"expireTime": "过期时间",
@@ -44,6 +50,7 @@
"invalidEmailFormat": "邮箱格式无效",
"inviteCode": "邀请码",
"inviteCodePlaceholder": "输入邀请码",
+ "isDeleted": "状态",
"kickOfflineConfirm": "确认踢下线",
"kickOfflineSuccess": "设备已踢下线",
"lastSeen": "最后上线",
@@ -73,37 +80,29 @@
"referrerUserId": "推荐人用户 ID",
"remove": "移除",
"resetLogs": "重置日志",
- "resetTraffic": "重置流量",
- "toggleStatus": "切换状态",
- "resetSubscriptionToken": "重置令牌",
- "resetSubscriptionTokenDescription": "将重置订阅令牌,旧订阅链接会失效。",
- "resetSubscriptionTraffic": "重置流量",
- "resetSubscriptionTrafficDescription": "将重置该订阅的流量统计。",
- "toggleSubscriptionStatus": "切换状态",
- "toggleSubscriptionStatusDescription": "将切换该订阅的启用/停用状态。",
"resetTime": "重置时间",
"resetToken": "重置订阅地址",
"resetTokenDescription": "这将重置订阅地址并重新生成新的令牌。",
+ "saving": "保存中...",
"resetTokenSuccess": "订阅地址重置成功",
- "confirmResetToken": "确认重置订阅地址",
+ "resumeSubscribe": "恢复订阅",
+ "selectGroup": "选择一个组",
+ "resumeSubscribeDescription": "这将恢复订阅,允许用户继续使用。",
+ "resumeSubscribeSuccess": "订阅已恢复",
+ "save": "保存",
+ "shortCode": "短码",
+ "speedLimit": "速度限制",
+ "startTime": "开始时间",
+ "status": "状态",
+ "statusActive": "活跃",
+ "statusDeducted": "已扣除",
+ "statusExpired": "已过期",
+ "statusFinished": "已完成",
+ "statusPending": "待处理",
+ "statusStopped": "已停止",
"stopSubscribe": "暂停订阅",
"stopSubscribeDescription": "这将暂时停止订阅。用户将无法使用。",
"stopSubscribeSuccess": "订阅已暂停",
- "confirmStopSubscribe": "确认暂停订阅",
- "resumeSubscribe": "恢复订阅",
- "resumeSubscribeDescription": "这将恢复订阅,允许用户继续使用。",
- "resumeSubscribeSuccess": "订阅已恢复",
- "confirmResumeSubscribe": "确认恢复订阅",
- "status": "状态",
- "statusPending": "待处理",
- "statusActive": "活跃",
- "statusFinished": "已完成",
- "statusExpired": "已过期",
- "statusDeducted": "已扣除",
- "statusStopped": "已停止",
- "save": "保存",
- "speedLimit": "速度限制",
- "startTime": "开始时间",
"subscription": "订阅",
"subscriptionId": "订阅 ID",
"subscriptionInfo": "订阅信息",
@@ -120,10 +119,12 @@
"trafficLimit": "流量限制",
"trafficStats": "流量统计",
"trafficUsage": "流量使用",
+ "remainingTraffic": "剩余流量",
"unlimited": "无限制",
"unverified": "未验证",
"update": "更新",
"updateSuccess": "更新成功",
+ "groupUpdated": "分组更新成功",
"upload": "上传",
"uploadTraffic": "上传流量",
"userAgent": "用户代理",
@@ -134,5 +135,17 @@
"userList": "用户列表",
"userName": "用户名",
"userProfile": "用户资料",
- "verified": "已验证"
+ "userGroup": "用户分组",
+ "verified": "已验证",
+ "locked": "锁定",
+ "lockGroup": "锁定分组",
+ "lockGroupDescription": "防止自动分组更改此用户的分组",
+ "groupLocked": "分组已锁定",
+ "previewNodes": "预览节点",
+ "availableNodes": "可用节点",
+ "name": "名称",
+ "address": "地址",
+ "noNodesAvailable": "无可用节点",
+ "nodeGroup": "节点组",
+ "publicNodes": "公共节点"
}
diff --git a/apps/admin/src/layout/navs.ts b/apps/admin/src/layout/navs.ts
index 07ea8ba..e4b8c65 100644
--- a/apps/admin/src/layout/navs.ts
+++ b/apps/admin/src/layout/navs.ts
@@ -34,6 +34,11 @@ export function useNavs() {
url: "/dashboard/nodes",
icon: "flat-color-icons:mind-map",
},
+ {
+ title: t("Group Management", "Group Management"),
+ url: "/dashboard/group",
+ icon: "flat-color-icons:department",
+ },
{
title: t("Subscribe Config", "Subscribe Config"),
url: "/dashboard/subscribe",
diff --git a/apps/admin/src/routeTree.gen.ts b/apps/admin/src/routeTree.gen.ts
index 65c03c2..97268bd 100644
--- a/apps/admin/src/routeTree.gen.ts
+++ b/apps/admin/src/routeTree.gen.ts
@@ -39,6 +39,8 @@ const DashboardOrderIndexLazyRouteImport =
const DashboardMarketingIndexLazyRouteImport = createFileRoute(
'/dashboard/marketing/',
)()
+const DashboardGroupIndexLazyRouteImport =
+ createFileRoute('/dashboard/group/')()
const DashboardDocumentIndexLazyRouteImport = createFileRoute(
'/dashboard/document/',
)()
@@ -189,6 +191,13 @@ const DashboardMarketingIndexLazyRoute =
} as any).lazy(() =>
import('./routes/dashboard/marketing/index.lazy').then((d) => d.Route),
)
+const DashboardGroupIndexLazyRoute = DashboardGroupIndexLazyRouteImport.update({
+ id: '/group/',
+ path: '/group/',
+ getParentRoute: () => DashboardRouteLazyRoute,
+} as any).lazy(() =>
+ import('./routes/dashboard/group/index.lazy').then((d) => d.Route),
+)
const DashboardDocumentIndexLazyRoute =
DashboardDocumentIndexLazyRouteImport.update({
id: '/document/',
@@ -345,6 +354,7 @@ export interface FileRoutesByFullPath {
'/dashboard/auth-control': typeof DashboardAuthControlIndexLazyRoute
'/dashboard/coupon': typeof DashboardCouponIndexLazyRoute
'/dashboard/document': typeof DashboardDocumentIndexLazyRoute
+ '/dashboard/group': typeof DashboardGroupIndexLazyRoute
'/dashboard/marketing': typeof DashboardMarketingIndexLazyRoute
'/dashboard/order': typeof DashboardOrderIndexLazyRoute
'/dashboard/payment': typeof DashboardPaymentIndexLazyRoute
@@ -377,6 +387,7 @@ export interface FileRoutesByTo {
'/dashboard/auth-control': typeof DashboardAuthControlIndexLazyRoute
'/dashboard/coupon': typeof DashboardCouponIndexLazyRoute
'/dashboard/document': typeof DashboardDocumentIndexLazyRoute
+ '/dashboard/group': typeof DashboardGroupIndexLazyRoute
'/dashboard/marketing': typeof DashboardMarketingIndexLazyRoute
'/dashboard/order': typeof DashboardOrderIndexLazyRoute
'/dashboard/payment': typeof DashboardPaymentIndexLazyRoute
@@ -411,6 +422,7 @@ export interface FileRoutesById {
'/dashboard/auth-control/': typeof DashboardAuthControlIndexLazyRoute
'/dashboard/coupon/': typeof DashboardCouponIndexLazyRoute
'/dashboard/document/': typeof DashboardDocumentIndexLazyRoute
+ '/dashboard/group/': typeof DashboardGroupIndexLazyRoute
'/dashboard/marketing/': typeof DashboardMarketingIndexLazyRoute
'/dashboard/order/': typeof DashboardOrderIndexLazyRoute
'/dashboard/payment/': typeof DashboardPaymentIndexLazyRoute
@@ -446,6 +458,7 @@ export interface FileRouteTypes {
| '/dashboard/auth-control'
| '/dashboard/coupon'
| '/dashboard/document'
+ | '/dashboard/group'
| '/dashboard/marketing'
| '/dashboard/order'
| '/dashboard/payment'
@@ -478,6 +491,7 @@ export interface FileRouteTypes {
| '/dashboard/auth-control'
| '/dashboard/coupon'
| '/dashboard/document'
+ | '/dashboard/group'
| '/dashboard/marketing'
| '/dashboard/order'
| '/dashboard/payment'
@@ -511,6 +525,7 @@ export interface FileRouteTypes {
| '/dashboard/auth-control/'
| '/dashboard/coupon/'
| '/dashboard/document/'
+ | '/dashboard/group/'
| '/dashboard/marketing/'
| '/dashboard/order/'
| '/dashboard/payment/'
@@ -627,6 +642,13 @@ declare module '@tanstack/react-router' {
preLoaderRoute: typeof DashboardMarketingIndexLazyRouteImport
parentRoute: typeof DashboardRouteLazyRoute
}
+ '/dashboard/group/': {
+ id: '/dashboard/group/'
+ path: '/group'
+ fullPath: '/dashboard/group'
+ preLoaderRoute: typeof DashboardGroupIndexLazyRouteImport
+ parentRoute: typeof DashboardRouteLazyRoute
+ }
'/dashboard/document/': {
id: '/dashboard/document/'
path: '/document'
@@ -770,6 +792,7 @@ interface DashboardRouteLazyRouteChildren {
DashboardAuthControlIndexLazyRoute: typeof DashboardAuthControlIndexLazyRoute
DashboardCouponIndexLazyRoute: typeof DashboardCouponIndexLazyRoute
DashboardDocumentIndexLazyRoute: typeof DashboardDocumentIndexLazyRoute
+ DashboardGroupIndexLazyRoute: typeof DashboardGroupIndexLazyRoute
DashboardMarketingIndexLazyRoute: typeof DashboardMarketingIndexLazyRoute
DashboardOrderIndexLazyRoute: typeof DashboardOrderIndexLazyRoute
DashboardPaymentIndexLazyRoute: typeof DashboardPaymentIndexLazyRoute
@@ -802,6 +825,7 @@ const DashboardRouteLazyRouteChildren: DashboardRouteLazyRouteChildren = {
DashboardAuthControlIndexLazyRoute: DashboardAuthControlIndexLazyRoute,
DashboardCouponIndexLazyRoute: DashboardCouponIndexLazyRoute,
DashboardDocumentIndexLazyRoute: DashboardDocumentIndexLazyRoute,
+ DashboardGroupIndexLazyRoute: DashboardGroupIndexLazyRoute,
DashboardMarketingIndexLazyRoute: DashboardMarketingIndexLazyRoute,
DashboardOrderIndexLazyRoute: DashboardOrderIndexLazyRoute,
DashboardPaymentIndexLazyRoute: DashboardPaymentIndexLazyRoute,
diff --git a/apps/admin/src/routes/dashboard/group/index.lazy.tsx b/apps/admin/src/routes/dashboard/group/index.lazy.tsx
new file mode 100644
index 0000000..f723df2
--- /dev/null
+++ b/apps/admin/src/routes/dashboard/group/index.lazy.tsx
@@ -0,0 +1,6 @@
+import { createLazyFileRoute } from "@tanstack/react-router";
+import Group from "@/sections/group";
+
+export const Route = createLazyFileRoute("/dashboard/group/")({
+ component: Group,
+});
diff --git a/apps/admin/src/sections/group/average-mode-tab.tsx b/apps/admin/src/sections/group/average-mode-tab.tsx
new file mode 100644
index 0000000..4151bbf
--- /dev/null
+++ b/apps/admin/src/sections/group/average-mode-tab.tsx
@@ -0,0 +1,264 @@
+"use client";
+
+import { Badge } from "@workspace/ui/components/badge";
+import { Button } from "@workspace/ui/components/button";
+import {
+ Card,
+ CardContent,
+ CardDescription,
+ CardHeader,
+ CardTitle,
+} from "@workspace/ui/components/card";
+import { Input } from "@workspace/ui/components/input";
+import { Label } from "@workspace/ui/components/label";
+import { Loader2 } from "lucide-react";
+import { useEffect, useState, useRef } from "react";
+import { useTranslation } from "react-i18next";
+import { useQuery } from "@tanstack/react-query";
+import { toast } from "sonner";
+import {
+ getGroupConfig,
+ getNodeGroupList,
+ getRecalculationStatus,
+ recalculateGroup,
+} from "@workspace/ui/services/admin/group";
+
+export default function AverageModeTab() {
+ const { t } = useTranslation("group");
+ const [recalculating, setRecalculating] = useState(false);
+ const [loadingStatus, setLoadingStatus] = useState(false);
+
+ const [averageConfig, setAverageConfig] = useState({
+ node_group_count: 0,
+ });
+
+ const [status, setStatus] = useState<{
+ state: string;
+ progress: number;
+ total: number;
+ } | null>(null);
+
+ const hasLoadedConfig = useRef(true);
+
+ const { data: nodeGroupsData } = useQuery({
+ queryKey: ["nodeGroups"],
+ queryFn: async () => {
+ const { data } = await getNodeGroupList({ page: 1, size: 1000 });
+ return data.data?.list || [];
+ },
+ });
+
+ const loadConfig = async () => {
+ try {
+ const { data } = await getGroupConfig();
+ if (data.data?.config?.average_config) {
+ setAverageConfig(data.data.config.average_config as any);
+ }
+ hasLoadedConfig.current = true;
+ } catch (error) {
+ console.error("Failed to load group config:", error);
+ toast.error(t("loadFailed", "Failed to load configuration"));
+ }
+ };
+
+ const loadStatus = async () => {
+ setLoadingStatus(true);
+ try {
+ const { data } = await getRecalculationStatus();
+ if (data.data) {
+ setStatus(data.data);
+ }
+ } catch (error) {
+ console.error("Failed to load recalculation status:", error);
+ } finally {
+ setLoadingStatus(false);
+ }
+ };
+
+ useEffect(() => {
+ loadConfig();
+ loadStatus();
+ }, []);
+
+ useEffect(() => {
+ if (nodeGroupsData) {
+ const nodeGroupCount = nodeGroupsData?.length || 0;
+
+ if (averageConfig.node_group_count !== nodeGroupCount) {
+ setAverageConfig({
+ ...averageConfig,
+ node_group_count: nodeGroupCount,
+ });
+ }
+ }
+ }, [nodeGroupsData]);
+
+ useEffect(() => {
+ const interval = setInterval(() => {
+ if (status?.state === "running") {
+ loadStatus();
+ }
+ }, 2000);
+ return () => clearInterval(interval);
+ }, [status?.state]);
+
+ const handleRecalculate = async () => {
+ setRecalculating(true);
+ try {
+ await recalculateGroup({ mode: "average" });
+ toast.success(t("recalculationStarted", "Recalculation started"));
+ loadStatus();
+ } catch (error) {
+ console.error("Failed to start recalculation:", error);
+ toast.error(t("recalculationFailed", "Failed to start recalculation"));
+ } finally {
+ setRecalculating(false);
+ }
+ };
+
+ const getStateLabel = (state: string) => {
+ switch (state) {
+ case "running":
+ return t("running", "Running");
+ case "completed":
+ return t("completed", "Completed");
+ case "failed":
+ return t("failed", "Failed");
+ default:
+ return t("idle", "Idle");
+ }
+ };
+
+ const getStateVariant = (state: string) => {
+ switch (state) {
+ case "running":
+ return "default";
+ case "completed":
+ return "secondary";
+ case "failed":
+ return "destructive";
+ default:
+ return "outline";
+ }
+ };
+
+ return (
+
+ {/* Configuration Card */}
+
+
+ {t("averageModeConfig", "Average Mode Configuration")}
+
+ {t(
+ "averageModeDescription",
+ "Randomly assign node groups to user subscriptions based on subscribe configuration"
+ )}
+
+
+
+
+
+
+ {t("availableNodeGroups", "Available Node Groups")}
+
+
+
+ {t("nodeGroupCountAutoCalculated", "Auto-calculated from actual node groups")}
+
+
+
+
+
+
+ {/* Recalculation Card */}
+
+
+ {t("groupRecalculation", "Group Recalculation")}
+
+ {t(
+ "groupRecalculationDescription",
+ "Manually trigger a full recalculation of all user groups based on current configuration"
+ )}
+
+
+
+ {/* Current Status */}
+
+
+
+ {t("currentStatus", "Current Status")}
+
+ {loadingStatus ? (
+
+ ) : status ? (
+
+ {getStateLabel(status.state)}
+
+ ) : null}
+
+
+ {status?.state === "running" && (
+
+
+ {t("progress", "Progress")}
+
+ {status.progress} / {status.total || 0}
+
+
+
+
0 ? (status.progress / status.total) * 100 : 0}%`,
+ }}
+ />
+
+
+ )}
+
+ {status?.state === "completed" && (
+
+ {t("recalculationCompleted", "Recalculation completed successfully")}
+
+ )}
+
+ {status?.state === "failed" && (
+
+ {t("recalculationFailed", "Recalculation failed. Please try again.")}
+
+ )}
+
+
+ {/* Recalculate Button */}
+
+
+ {recalculating && (
+
+ )}
+ {t("recalculateAll", "Recalculate All Users")}
+
+
+
+ {/* Warning */}
+
+ {t("warning", "Warning")}: {" "}
+ {t(
+ "recalculationWarning",
+ "Recalculation will reassign all users to new groups based on current configuration. This operation cannot be undone."
+ )}
+
+
+
+
+ );
+}
diff --git a/apps/admin/src/sections/group/bind-node-groups-dialog.tsx b/apps/admin/src/sections/group/bind-node-groups-dialog.tsx
new file mode 100644
index 0000000..debdeac
--- /dev/null
+++ b/apps/admin/src/sections/group/bind-node-groups-dialog.tsx
@@ -0,0 +1,174 @@
+"use client";
+
+import { Button } from "@workspace/ui/components/button";
+import {
+ Dialog,
+ DialogContent,
+ DialogDescription,
+ DialogFooter,
+ DialogHeader,
+ DialogTitle,
+ DialogTrigger,
+} from "@workspace/ui/components/dialog";
+import { Label } from "@workspace/ui/components/label";
+import {
+ Select,
+ SelectContent,
+ SelectItem,
+ SelectTrigger,
+ SelectValue,
+} from "@workspace/ui/components/select";
+import { Loader2 } from "lucide-react";
+import { useEffect, useState } from "react";
+import { useTranslation } from "react-i18next";
+import { useQuery } from "@tanstack/react-query";
+import { toast } from "sonner";
+import { getNodeGroupList, bindNodeGroups } from "@workspace/ui/services/admin/group";
+
+interface BindNodeGroupsDialogProps {
+ userGroupIds: number[];
+ userGroupNames: string[];
+ onOpenChange?: (open: boolean) => void;
+ onSuccess?: () => void;
+}
+
+export default function BindNodeGroupsDialog({
+ userGroupIds,
+ userGroupNames,
+ onOpenChange,
+ onSuccess,
+}: BindNodeGroupsDialogProps) {
+ const { t } = useTranslation("group");
+ const [open, setOpen] = useState(false);
+ const [saving, setSaving] = useState(false);
+ const [selectedNodeGroupId, setSelectedNodeGroupId] = useState();
+
+ const { data: nodeGroupsData, isLoading } = useQuery({
+ queryKey: ["nodeGroups"],
+ queryFn: async () => {
+ const { data } = await getNodeGroupList({ page: 1, size: 1000 });
+ return data.data?.list || [];
+ },
+ });
+
+ useEffect(() => {
+ if (open && nodeGroupsData) {
+ // Load current binding when dialog opens
+ loadCurrentBinding();
+ }
+ }, [open]);
+
+ const loadCurrentBinding = () => {
+ // Get first user group's current node group binding
+ // For batch binding, we'll default to unbound
+ setSelectedNodeGroupId(undefined);
+ };
+
+ const handleBind = async () => {
+ if (selectedNodeGroupId === undefined) {
+ toast.error(t("selectNodeGroupRequired", "Please select a node group"));
+ return;
+ }
+
+ setSaving(true);
+ try {
+ await bindNodeGroups({
+ user_group_ids: userGroupIds,
+ node_group_id: selectedNodeGroupId === 0 ? null : selectedNodeGroupId,
+ } as API.BindNodeGroupsRequest);
+
+ toast.success(
+ t("bindSuccess", "Successfully bound {{userGroupCount}} user groups to node group").replace(
+ /{{userGroupCount}}/g,
+ String(userGroupIds.length)
+ )
+ );
+
+ setOpen(false);
+ onOpenChange?.(false);
+ onSuccess?.();
+ } catch (error) {
+ console.error("Failed to bind node group:", error);
+ toast.error(t("bindFailed", "Failed to bind node group"));
+ } finally {
+ setSaving(false);
+ }
+ };
+
+ const displayNames =
+ userGroupNames.length > 2
+ ? `${userGroupNames.slice(0, 2).join(", ")}... (${userGroupIds.length})`
+ : userGroupNames.join(", ");
+
+ return (
+ {
+ setOpen(newOpen);
+ onOpenChange?.(newOpen);
+ }}>
+
+
+ {t("bindNodeGroup", "Bind Node Group")}
+
+
+
+
+ {t("bindNodeGroup", "Bind Node Group")}
+
+ {t(
+ "bindNodeGroupDescription",
+ "Select a node group to bind to user groups: {{userGroups}}",
+ { userGroups: displayNames }
+ ).replace(/{{userGroups}}/g, displayNames)}
+
+
+
+
+ {isLoading ? (
+
+
+
+ ) : (
+
+ {t("selectNodeGroup", "Select Node Group")}
+ setSelectedNodeGroupId(parseInt(val) || undefined)}
+ >
+
+
+
+
+
+ {t("unbound", "Unbound")}
+
+ {nodeGroupsData?.map((nodeGroup) => (
+
+ {nodeGroup.name}
+
+ ))}
+
+
+
+ )}
+
+
+
+ {
+ setOpen(false);
+ onOpenChange?.(false);
+ }}
+ disabled={saving}
+ >
+ {t("cancel", "Cancel")}
+
+
+ {saving && }
+ {t("confirm", "Confirm")}
+
+
+
+
+ );
+}
diff --git a/apps/admin/src/sections/group/current-group-results.tsx b/apps/admin/src/sections/group/current-group-results.tsx
new file mode 100644
index 0000000..c76e655
--- /dev/null
+++ b/apps/admin/src/sections/group/current-group-results.tsx
@@ -0,0 +1,375 @@
+"use client";
+
+import {
+ Card,
+ CardContent,
+ CardDescription,
+ CardHeader,
+ CardTitle,
+} from "@workspace/ui/components/card";
+import {
+ Dialog,
+ DialogContent,
+ DialogDescription,
+ DialogHeader,
+ DialogTitle,
+} from "@workspace/ui/components/dialog";
+import {
+ Table,
+ TableBody,
+ TableCell,
+ TableHead,
+ TableHeader,
+ TableRow,
+} from "@workspace/ui/components/table";
+import { Loader2 } from "lucide-react";
+import { useEffect, useState } from "react";
+import { useTranslation } from "react-i18next";
+import { useQuery } from "@tanstack/react-query";
+import { getGroupHistory, getGroupHistoryDetail, getNodeGroupList } from "@workspace/ui/services/admin/group";
+
+export default function CurrentGroupResults() {
+ const { t } = useTranslation("group");
+ const [loading, setLoading] = useState(true);
+ const [latestResult, setLatestResult] = useState(null);
+ const [latestDetails, setLatestDetails] = useState([]);
+ const [detailsLoading, setDetailsLoading] = useState(false);
+
+ // User list dialog state
+ const [userListOpen, setUserListOpen] = useState(false);
+ const [selectedNodeGroupName, setSelectedNodeGroupName] = useState("");
+ const [userList, setUserList] = useState([]);
+ const [userListLoading, setUserListLoading] = useState(false);
+ const [userListTotal, setUserListTotal] = useState(0);
+
+ // Fetch node groups
+ const { data: nodeGroups } = useQuery({
+ queryKey: ["nodeGroups"],
+ queryFn: async () => {
+ const { data } = await getNodeGroupList({ page: 1, size: 1000 });
+ return data.data?.list || [];
+ },
+ });
+
+ useEffect(() => {
+ const loadData = async () => {
+ try {
+ // Load latest result
+ const { data: historyData } = await getGroupHistory({
+ page: 1,
+ size: 1,
+ });
+
+ if (historyData.data?.list && historyData.data.list.length > 0) {
+ const latest = historyData.data.list[0];
+ if (!latest) return;
+ setLatestResult(latest);
+
+ // Fetch details
+ setDetailsLoading(true);
+ try {
+ const { data: detailData } = await getGroupHistoryDetail({
+ id: latest.id,
+ });
+
+ if (detailData.data?.config_snapshot?.group_details) {
+ setLatestDetails(detailData.data.config_snapshot.group_details);
+ } else {
+ setLatestDetails([]);
+ }
+ } catch (error) {
+ console.error("Failed to fetch latest result details:", error);
+ setLatestDetails([]);
+ } finally {
+ setDetailsLoading(false);
+ }
+ }
+ } catch (error) {
+ console.error("Failed to load data:", error);
+ } finally {
+ setLoading(false);
+ }
+ };
+
+ loadData();
+ }, []);
+
+ const handleShowUserList = async (nodeGroupId: number, nodeGroupName: string) => {
+ setSelectedNodeGroupName(nodeGroupName);
+ setUserListOpen(true);
+ setUserListLoading(true);
+
+ // 从历史详情记录中获取用户数据
+ const detail = latestDetails.find((d: any) => {
+ const detailNodeGroupId = d.NodeGroupId || d.node_group_id;
+ return detailNodeGroupId === nodeGroupId;
+ });
+
+ if (detail) {
+ const userDataJSON = detail.UserData || detail.user_data;
+ if (userDataJSON) {
+ try {
+ const userData = JSON.parse(userDataJSON);
+ setUserList(userData);
+ setUserListTotal(userData.length);
+ } catch (error) {
+ console.error("Failed to parse user data:", error);
+ setUserList([]);
+ setUserListTotal(0);
+ }
+ } else {
+ setUserList([]);
+ setUserListTotal(0);
+ }
+ } else {
+ setUserList([]);
+ setUserListTotal(0);
+ }
+ setUserListLoading(false);
+ };
+
+ if (loading) {
+ return (
+
+
+ {t("currentGroupingResult", "Current Grouping Result")}
+
+ {t("loading", "Loading...")}
+
+
+
+ );
+ }
+
+ return (
+
+ {/* Latest Result Card */}
+ {!latestResult ? (
+
+
+ {t("currentGroupingResult", "Current Grouping Result")}
+
+
+
+ {t("noDetails", "No details available")}
+
+
+
+ ) : (
+
+
+ {t("currentGroupingResult", "Current Grouping Result")}
+
+ {t("latestGroupingCalculation", "Latest grouping calculation details")}
+
+
+
+ {/* Calculation Info */}
+
+
{t("calculationInfo", "Calculation Information")}
+
+
+
{t("groupMode", "Group Mode")}
+
+ {(latestResult.GroupMode || latestResult.group_mode) === "average"
+ ? t("averageMode", "Average Mode")
+ : (latestResult.GroupMode || latestResult.group_mode) === "subscribe"
+ ? t("subscribeMode", "Subscribe Mode")
+ : t("trafficMode", "Traffic Mode")}
+
+
+
+
{t("state", "State")}
+
+ {(latestResult.State || latestResult.state) === "completed"
+ ? t("completed", "Completed")
+ : (latestResult.State || latestResult.state) === "running"
+ ? t("running", "Running")
+ : (latestResult.State || latestResult.state) === "failed"
+ ? t("failed", "Failed")
+ : t("idle", "Idle")}
+
+
+
+
{t("triggerType", "Trigger Type")}
+
+ {(latestResult.TriggerType || latestResult.trigger_type) === "manual"
+ ? t("manualTrigger", "Manual")
+ : (latestResult.TriggerType || latestResult.trigger_type) === "auto"
+ ? t("autoTrigger", "Auto")
+ : t("scheduleTrigger", "Schedule")}
+
+
+
+
{t("successFailedCount", "Success/Failed")}
+
+ {latestResult.SuccessCount || latestResult.success_count || 0} / {latestResult.FailedCount || latestResult.failed_count || 0}
+
+
+
+
{t("startTime", "Start Time")}
+
+ {latestResult.StartTime || latestResult.start_time
+ ? new Date((latestResult.StartTime || latestResult.start_time) * 1000).toLocaleString()
+ : "-"}
+
+
+
+
{t("endTime", "End Time")}
+
+ {latestResult.EndTime || latestResult.end_time
+ ? new Date((latestResult.EndTime || latestResult.end_time) * 1000).toLocaleString()
+ : "-"}
+
+
+
+
+
+ {/* Grouping Details */}
+
+
{t("groupingDetailsStatistics", "Grouping Details Statistics")}
+
+
+
+ {latestDetails.reduce((sum: number, d: any) => sum + (d.UserCount || d.user_count || 0), 0)}
+
+
+ {t("totalUsers", "Total Users")}
+
+
+
+
+ {latestDetails.reduce((sum: number, d: any) => sum + (d.NodeCount || d.node_count || 0), 0)}
+
+
+ {t("totalNodes", "Total Nodes")}
+
+
+
+
{latestDetails.length}
+
+ {t("totalNodeGroups", "Total Node Groups")}
+
+
+
+
+
+ {detailsLoading ? (
+
+
+
+ {t("loading", "Loading...")}
+
+
+ ) : latestDetails.length > 0 ? (
+ <>
+ {/* Details Table */}
+
+
+
+
+
+ {t("nodeGroup", "Node Group")}
+
+
+ {t("userCount", "User Count")}
+
+
+ {t("nodeCount", "Node Count")}
+
+
+
+
+ {latestDetails.map((detail: any, index: number) => {
+ const nodeGroupId = detail.NodeGroupId || detail.node_group_id;
+ const nodeGroup = nodeGroups?.find((ng) => ng.id === nodeGroupId);
+ const nodeGroupName = nodeGroup?.name || `${t("idPrefix", "#")}${nodeGroupId}`;
+ const userCount = detail.UserCount || detail.user_count || 0;
+
+ return (
+
+
+
+
{nodeGroupName}
+
{t("id", "ID")}: {nodeGroupId}
+
+
+
+ handleShowUserList(nodeGroupId, nodeGroupName)}
+ disabled={userCount === 0}
+ >
+ {userCount}
+
+
+
+ {detail.NodeCount || detail.node_count || 0}
+
+
+ );
+ })}
+
+
+
+ >
+ ) : (
+
+ {t("noDetails", "No details available")}
+
+ )}
+
+
+ )}
+
+ {/* User List Dialog */}
+
+
+
+
+ {selectedNodeGroupName} - {t("userList", "User List")}
+
+
+ {t("totalUsers", "Total Users")}: {userListTotal}
+
+
+
+ {userListLoading ? (
+
+
+
+ {t("loading", "Loading...")}
+
+
+ ) : userList.length > 0 ? (
+
+
+
+ {t("id", "ID")}
+ {t("email", "Email")}
+
+
+
+ {userList.map((user) => (
+
+ {user.id}
+
+ {user.email || "-"}
+
+
+ ))}
+
+
+ ) : (
+
+ {t("noUsers", "No users found")}
+
+ )}
+
+
+
+
+ );
+}
diff --git a/apps/admin/src/sections/group/group-config.tsx b/apps/admin/src/sections/group/group-config.tsx
new file mode 100644
index 0000000..2ec3dd0
--- /dev/null
+++ b/apps/admin/src/sections/group/group-config.tsx
@@ -0,0 +1,264 @@
+"use client";
+
+import { Button } from "@workspace/ui/components/button";
+import {
+ Card,
+ CardContent,
+ CardDescription,
+ CardHeader,
+ CardTitle,
+} from "@workspace/ui/components/card";
+import { Loader2 } from "lucide-react";
+import { useEffect, useState } from "react";
+import { useTranslation } from "react-i18next";
+import { toast } from "sonner";
+import {
+ getGroupConfig,
+ updateGroupConfig,
+ resetGroups,
+} from "@workspace/ui/services/admin/group";
+import {
+ AlertDialog,
+ AlertDialogAction,
+ AlertDialogCancel,
+ AlertDialogContent,
+ AlertDialogDescription,
+ AlertDialogFooter,
+ AlertDialogHeader,
+ AlertDialogTitle,
+ AlertDialogTrigger,
+} from "@workspace/ui/components/alert-dialog";
+
+export default function GroupConfig() {
+ const { t } = useTranslation("group");
+ const [saving, setSaving] = useState(false);
+ const [resetting, setResetting] = useState(false);
+ const [showResetDialog, setShowResetDialog] = useState(false);
+ const [config, setConfig] = useState<{
+ enabled: boolean;
+ mode: "average" | "subscribe" | "traffic";
+ }>({
+ enabled: false,
+ mode: "average",
+ });
+
+ const loadConfig = async () => {
+ try {
+ const { data } = await getGroupConfig();
+ if (data.data) {
+ setConfig({
+ enabled: data.data.enabled || false,
+ mode: (data.data.mode || "average") as "average" | "subscribe" | "traffic",
+ });
+ }
+ } catch (error) {
+ console.error("Failed to load group config:", error);
+ toast.error(t("loadFailed", "Failed to load configuration"));
+ }
+ };
+
+ useEffect(() => {
+ loadConfig();
+ }, []);
+
+ const handleUpdateEnabled = async (enabled: boolean) => {
+ setSaving(true);
+ try {
+ const payload: any = {
+ enabled,
+ mode: config.mode,
+ };
+ await updateGroupConfig(payload);
+ setConfig({ ...config, enabled });
+ toast.success(t("saved", "Configuration saved successfully"));
+ } catch (error) {
+ console.error("Failed to update group config:", error);
+ toast.error(t("saveFailed", "Failed to save configuration"));
+ } finally {
+ setSaving(false);
+ }
+ };
+
+ const handleUpdateMode = async (mode: "average" | "subscribe" | "traffic") => {
+ setSaving(true);
+ try {
+ const payload: any = {
+ enabled: config.enabled,
+ mode,
+ };
+ await updateGroupConfig(payload);
+ setConfig({ ...config, mode });
+ toast.success(t("saved", "Configuration saved successfully"));
+ } catch (error) {
+ console.error("Failed to update group config:", error);
+ toast.error(t("saveFailed", "Failed to save configuration"));
+ } finally {
+ setSaving(false);
+ }
+ };
+
+ const handleResetGroups = async () => {
+ setResetting(true);
+ try {
+ await resetGroups({ confirm: true });
+ toast.success(t("resetSuccess", "All groups have been reset successfully"));
+ setShowResetDialog(false);
+ // Reload config after reset
+ await loadConfig();
+ } catch (error) {
+ console.error("Failed to reset groups:", error);
+ toast.error(t("resetFailed", "Failed to reset groups"));
+ } finally {
+ setResetting(false);
+ }
+ };
+
+ return (
+
+
+
+ {t("groupConfig", "Group Configuration")}
+
+ {t(
+ "groupConfigDescription",
+ "Configure user group and node group settings"
+ )}
+
+
+
+ {/* Enable/Disable */}
+
+
+
+ {t("enableGrouping", "Enable Grouping")}
+
+
+ {t(
+ "enableGroupingDescription",
+ "When enabled, users will only see nodes from their assigned group"
+ )}
+
+
+
handleUpdateEnabled(e.target.checked)}
+ disabled={saving}
+ className="h-4 w-4"
+ />
+
+
+ {/* Mode Selection */}
+ {config.enabled && (
+
+
+ {t("groupingMode", "Grouping Mode")}
+
+
+
handleUpdateMode("average")}
+ disabled={saving}
+ className={`rounded-lg border p-4 text-left transition-colors ${
+ config.mode === "average"
+ ? "border-primary bg-primary/10"
+ : "border-border hover:bg-muted"
+ } ${saving ? "opacity-50 cursor-not-allowed" : ""}`}
+ >
+
+ {t("averageMode", "Average Mode")}
+
+
+ {t(
+ "averageModeDescription",
+ "Distribute users evenly across groups"
+ )}
+
+
+
+
handleUpdateMode("subscribe")}
+ disabled={saving}
+ className={`rounded-lg border p-4 text-left transition-colors ${
+ config.mode === "subscribe"
+ ? "border-primary bg-primary/10"
+ : "border-border hover:bg-muted"
+ } ${saving ? "opacity-50 cursor-not-allowed" : ""}`}
+ >
+
+ {t("subscribeMode", "Subscribe Mode")}
+
+
+ {t(
+ "subscribeModeDescription",
+ "Group users by their subscription plan"
+ )}
+
+
+
+
handleUpdateMode("traffic")}
+ disabled={saving}
+ className={`rounded-lg border p-4 text-left transition-colors ${
+ config.mode === "traffic"
+ ? "border-primary bg-primary/10"
+ : "border-border hover:bg-muted"
+ } ${saving ? "opacity-50 cursor-not-allowed" : ""}`}
+ >
+
+ {t("trafficMode", "Traffic Mode")}
+
+
+ {t(
+ "trafficModeDescription",
+ "Group users by their traffic usage"
+ )}
+
+
+
+
+ )}
+
+ {/* Reset Button */}
+
+
+
+
+ {t("resetGroups", "Reset All Groups")}
+
+
+
+
+
+ {t("resetGroupsTitle", "Reset All Groups")}
+
+
+ {t(
+ "resetGroupsDescription",
+ "This action will delete all node groups and user groups, reset all users' group ID to 0, clear all products' node group IDs, and clear all nodes' node group IDs. This action cannot be undone."
+ )}
+
+
+
+
+ {t("cancel", "Cancel")}
+
+
+ {resetting && }
+ {t("confirm", "Confirm")}
+
+
+
+
+
+
+
+
+ );
+}
diff --git a/apps/admin/src/sections/group/group-history.tsx b/apps/admin/src/sections/group/group-history.tsx
new file mode 100644
index 0000000..140da99
--- /dev/null
+++ b/apps/admin/src/sections/group/group-history.tsx
@@ -0,0 +1,499 @@
+"use client";
+
+import { Badge } from "@workspace/ui/components/badge";
+import { Button } from "@workspace/ui/components/button";
+import {
+ Card,
+ CardContent,
+ CardDescription,
+ CardHeader,
+ CardTitle,
+} from "@workspace/ui/components/card";
+import {
+ Dialog,
+ DialogContent,
+ DialogDescription,
+ DialogHeader,
+ DialogTitle,
+} from "@workspace/ui/components/dialog";
+import {
+ Table,
+ TableBody,
+ TableCell,
+ TableHead,
+ TableHeader,
+ TableRow,
+} from "@workspace/ui/components/table";
+import {
+ ProTable,
+ type ProTableActions,
+} from "@workspace/ui/composed/pro-table/pro-table";
+import {
+ getGroupHistory,
+ getGroupHistoryDetail,
+ getNodeGroupList,
+} from "@workspace/ui/services/admin/group";
+import { useRef, useState } from "react";
+import { useTranslation } from "react-i18next";
+import { formatDate } from "@/utils/common";
+import { Loader2 } from "lucide-react";
+import { useQuery } from "@tanstack/react-query";
+
+export default function GroupHistory() {
+ const { t } = useTranslation("group");
+ const ref = useRef(null);
+ const [detailOpen, setDetailOpen] = useState(false);
+ const [detailLoading, setDetailLoading] = useState(false);
+ const [selectedHistory, setSelectedHistory] = useState(null);
+ const [details, setDetails] = useState([]);
+ const [nodeGroupMap, setNodeGroupMap] = useState>(new Map());
+
+ // User list dialog state
+ const [userListOpen, setUserListOpen] = useState(false);
+ const [selectedNodeGroupName, setSelectedNodeGroupName] = useState("");
+ const [userList, setUserList] = useState([]);
+ const [userListTotal, setUserListTotal] = useState(0);
+
+ // Fetch all node groups
+ const { data: nodeGroups } = useQuery({
+ queryKey: ["getNodeGroupListForDetail"],
+ queryFn: async () => {
+ const { data } = await getNodeGroupList({
+ page: 1,
+ size: 100,
+ });
+ return data.data?.list || [];
+ },
+ });
+
+ // Build ID to name maps when groups are loaded
+ if (nodeGroups) {
+ const newNodeGroupMap = new Map();
+ nodeGroups.forEach((ng: API.NodeGroup) => {
+ newNodeGroupMap.set(ng.id, ng.name);
+ });
+ if (newNodeGroupMap.size !== nodeGroupMap.size) {
+ setNodeGroupMap(newNodeGroupMap);
+ }
+ }
+
+ const getModeLabel = (mode: string) => {
+ switch (mode) {
+ case "average":
+ return t("averageMode", "Average");
+ case "subscribe":
+ return t("subscribeMode", "Subscribe");
+ case "traffic":
+ return t("trafficMode", "Traffic");
+ default:
+ return mode;
+ }
+ };
+
+ const getTriggerTypeLabel = (type: string) => {
+ switch (type) {
+ case "manual":
+ return t("manualTrigger", "Manual");
+ case "auto":
+ return t("autoTrigger", "Auto");
+ case "schedule":
+ return t("scheduleTrigger", "Schedule");
+ default:
+ return type;
+ }
+ };
+
+ const handleViewDetail = async (record: API.GroupHistory) => {
+ setSelectedHistory(record);
+ setDetailOpen(true);
+ setDetailLoading(true);
+ try {
+ const { data } = await getGroupHistoryDetail({
+ id: record.id,
+ });
+
+ console.log("Group history detail response:", data);
+
+ // 从返回的数据中获取详情列表
+ // data.data.config_snapshot.group_details 包含分组详情
+ if (data.data?.config_snapshot?.group_details) {
+ setDetails(data.data.config_snapshot.group_details);
+ } else {
+ console.warn("No group_details found in response:", data);
+ setDetails([]);
+ }
+ } catch (error) {
+ console.error("Failed to fetch history details:", error);
+ setDetails([]);
+ } finally {
+ setDetailLoading(false);
+ }
+ };
+
+ const handleShowUserList = async (nodeGroupId: number, nodeGroupName: string) => {
+ setSelectedNodeGroupName(nodeGroupName);
+ setUserListOpen(true);
+
+ // 从历史详情记录中获取用户数据
+ const detail = details.find((d: any) => {
+ const detailNodeGroupId = d.NodeGroupId || d.node_group_id;
+ return detailNodeGroupId === nodeGroupId;
+ });
+
+ if (detail) {
+ const userDataJSON = detail.UserData || detail.user_data;
+ if (userDataJSON) {
+ try {
+ const userData = JSON.parse(userDataJSON);
+ setUserList(userData);
+ setUserListTotal(userData.length);
+ } catch (error) {
+ console.error("Failed to parse user data:", error);
+ setUserList([]);
+ setUserListTotal(0);
+ }
+ } else {
+ setUserList([]);
+ setUserListTotal(0);
+ }
+ } else {
+ setUserList([]);
+ setUserListTotal(0);
+ }
+ };
+
+ return (
+
+
+
+ {t("groupHistory", "Group Calculation History")}
+
+ {t("groupHistoryDescription", "View group recalculation history and results")}
+
+
+
+
+ action={ref}
+ request={async (params) => {
+ const { data } = await getGroupHistory({
+ page: params.page || 1,
+ size: params.size || 10,
+ });
+ return {
+ list: data.data?.list || [],
+ total: data.data?.total || 0,
+ };
+ }}
+ columns={[
+ {
+ id: "id",
+ accessorKey: "id",
+ header: t("id", "ID"),
+ cell: ({ row }: { row: any }) => (
+
+ {t("idPrefix", "#")}{row.getValue("id")}
+
+ ),
+ },
+ {
+ id: "group_mode",
+ accessorKey: "group_mode",
+ header: t("groupMode", "Group Mode"),
+ cell: ({ row }: { row: any }) => (
+
+ {getModeLabel(row.getValue("group_mode"))}
+
+ ),
+ },
+ {
+ id: "trigger_type",
+ accessorKey: "trigger_type",
+ header: t("triggerType", "Trigger Type"),
+ cell: ({ row }: { row: any }) => (
+
+ {getTriggerTypeLabel(row.getValue("trigger_type"))}
+
+ ),
+ },
+ {
+ id: "total_users",
+ accessorKey: "total_users",
+ header: t("totalUsers", "Total Users"),
+ cell: ({ row }: { row: any }) => (
+ {row.getValue("total_users")}
+ ),
+ },
+ {
+ id: "result",
+ accessorKey: "error_log",
+ header: t("result", "Result"),
+ cell: ({ row }: { row: any }) => {
+ const record = row.original;
+ return (
+
+
+ {t("successCount", "Success")}: {record.success_count}
+ {" "}{t("separator", "/")}{" "}
+ {t("failedCount", "Failed")}: {record.failed_count}
+
+ {record.error_log && (
+
+ {t("failed", "Failed")}
+
+ )}
+ {!record.error_log && record.failed_count === 0 && (
+
+ {t("completed", "Completed")}
+
+ )}
+
+ );
+ },
+ },
+ {
+ id: "created_at",
+ accessorKey: "created_at",
+ header: t("createdAt", "Created At"),
+ cell: ({ row }: { row: any }) => formatDate(row.getValue("created_at")),
+ },
+ ]}
+ actions={{
+ render: (row: any) => [
+ handleViewDetail(row)}
+ >
+ {t("viewDetail", "View Detail")}
+ ,
+ ],
+ }}
+ header={{
+ title: t("groupHistory", "Group Calculation History"),
+ }}
+ />
+
+
+
+ {/* Detail Dialog */}
+
+
+
+
+ {t("groupHistoryDetail", "Group Calculation Detail")}
+
+
+ {t("historyId", "History ID")}: {selectedHistory?.id}
+
+
+
+ {selectedHistory && (
+ <>
+
+
+
+ {t("groupMode", "Group Mode")}
+
+
+ {getModeLabel(selectedHistory.group_mode)}
+
+
+
+
+ {t("triggerType", "Trigger Type")}
+
+
+ {getTriggerTypeLabel(selectedHistory.trigger_type)}
+
+
+
+
+ {t("totalUsers", "Total Users")}
+
+
{selectedHistory.total_users}
+
+
+
+ {t("result", "Result")}
+
+
+ {t("successCount", "Success")}: {selectedHistory.success_count}
+ {" "}{t("separator", "/")}{" "}
+ {t("failedCount", "Failed")}: {selectedHistory.failed_count}
+
+
+ {selectedHistory.start_time && (
+
+
+ {t("startTime", "Start Time")}
+
+
+ {formatDate(selectedHistory.start_time)}
+
+
+ )}
+ {selectedHistory.end_time && (
+
+
+ {t("endTime", "End Time")}
+
+
+ {formatDate(selectedHistory.end_time)}
+
+
+ )}
+
+
+ {selectedHistory.error_log && (
+
+
+ {t("errorMessage", "Error Message")}
+
+
+ {selectedHistory.error_log}
+
+
+ )}
+ >
+ )}
+
+
+
+ {t("groupDetails", "Group Details")}
+
+ {detailLoading ? (
+
+
+
+ {t("loading", "Loading...")}
+
+
+ ) : details.length > 0 ? (
+ <>
+ {/* 统计信息 */}
+
+
+
+ {details.reduce((sum: number, d: any) => sum + (d.UserCount || d.user_count || 0), 0)}
+
+
+ {t("totalUsers", "Total Users")}
+
+
+
+
+ {details.reduce((sum: number, d: any) => sum + (d.NodeCount || d.node_count || 0), 0)}
+
+
+ {t("totalNodes", "Total Nodes")}
+
+
+
+
{details.length}
+
+ {t("totalNodeGroups", "Total Node Groups")}
+
+
+
+
+ {/* 详情表格 */}
+
+
+
+
+
+ {t("nodeGroup", "Node Group")}
+
+
+ {t("userCount", "User Count")}
+
+
+ {t("nodeCount", "Node Count")}
+
+
+
+
+ {details.map((detail: any, index: number) => {
+ const nodeGroupId = detail.NodeGroupId || detail.node_group_id;
+ const nodeGroupName = nodeGroupMap.get(nodeGroupId) || `${t("idPrefix", "#")}${nodeGroupId}`;
+
+ return (
+
+
+
+
{nodeGroupName}
+
{t("id", "ID")}: {nodeGroupId}
+
+
+
+ handleShowUserList(nodeGroupId, nodeGroupName)}
+ disabled={(detail.UserCount || detail.user_count || 0) === 0}
+ >
+ {detail.UserCount || detail.user_count || 0}
+
+
+
+ {detail.NodeCount || detail.node_count || 0}
+
+
+ );
+ })}
+
+
+
+ >
+ ) : (
+
+ {t("noDetails", "No details available")}
+
+ )}
+
+
+
+
+
+ {/* User List Dialog */}
+
+
+
+
+ {selectedNodeGroupName} - {t("userList", "User List")}
+
+
+ {t("totalUsers", "Total Users")}: {userListTotal}
+
+
+
+ {userList.length > 0 ? (
+
+
+
+ {t("id", "ID")}
+ {t("email", "Email")}
+
+
+
+ {userList.map((user) => (
+
+ {user.id}
+
+ {user.email || "-"}
+
+
+ ))}
+
+
+ ) : (
+
+ {t("noUsers", "No users found")}
+
+ )}
+
+
+
+
+ );
+}
diff --git a/apps/admin/src/sections/group/group-recalculate.tsx b/apps/admin/src/sections/group/group-recalculate.tsx
new file mode 100644
index 0000000..07f01b4
--- /dev/null
+++ b/apps/admin/src/sections/group/group-recalculate.tsx
@@ -0,0 +1,228 @@
+"use client";
+
+import { Badge } from "@workspace/ui/components/badge";
+import { Button } from "@workspace/ui/components/button";
+import {
+ Card,
+ CardContent,
+ CardDescription,
+ CardHeader,
+ CardTitle,
+} from "@workspace/ui/components/card";
+import { Loader2 } from "lucide-react";
+import { useEffect, useState } from "react";
+import { useTranslation } from "react-i18next";
+import { toast } from "sonner";
+import {
+ getRecalculationStatus,
+ recalculateGroup,
+} from "@workspace/ui/services/admin/group";
+
+export default function GroupRecalculate() {
+ const { t } = useTranslation("group");
+ const [recalculating, setRecalculating] = useState(null);
+ const [loadingStatus, setLoadingStatus] = useState(false);
+ const [status, setStatus] = useState<{
+ state: string;
+ progress: number;
+ total: number;
+ } | null>(null);
+
+ const loadStatus = async () => {
+ setLoadingStatus(true);
+ try {
+ const { data } = await getRecalculationStatus();
+ if (data.data) {
+ setStatus(data.data);
+ }
+ } catch (error) {
+ console.error("Failed to load recalculation status:", error);
+ } finally {
+ setLoadingStatus(false);
+ }
+ };
+
+ useEffect(() => {
+ loadStatus();
+
+ // Poll status every 2 seconds when recalculating
+ const interval = setInterval(() => {
+ if (status?.state === "running") {
+ loadStatus();
+ }
+ }, 2000);
+ return () => clearInterval(interval);
+ }, [status?.state]);
+
+ const handleRecalculate = async (mode: "average" | "subscribe" | "traffic") => {
+ setRecalculating(mode);
+ try {
+ await recalculateGroup({ mode });
+ toast.success(t("recalculationStarted", "Recalculation started"));
+ loadStatus();
+ } catch (error) {
+ console.error("Failed to start recalculation:", error);
+ toast.error(t("recalculationFailed", "Failed to start recalculation"));
+ } finally {
+ setRecalculating(null);
+ }
+ };
+
+ const getStateLabel = (state: string) => {
+ switch (state) {
+ case "running":
+ return t("running", "Running");
+ case "completed":
+ return t("completed", "Completed");
+ case "failed":
+ return t("failed", "Failed");
+ default:
+ return t("idle", "Idle");
+ }
+ };
+
+ const getStateVariant = (state: string) => {
+ switch (state) {
+ case "running":
+ return "default";
+ case "completed":
+ return "secondary";
+ case "failed":
+ return "destructive";
+ default:
+ return "outline";
+ }
+ };
+
+ return (
+
+
+
+ {t("groupRecalculation", "Group Recalculation")}
+
+ {t(
+ "groupRecalculationDescription",
+ "Manually trigger a full recalculation of all user groups based on current configuration"
+ )}
+
+
+
+ {/* Current Status */}
+
+
+
+ {t("currentStatus", "Current Status")}
+
+ {loadingStatus ? (
+
+ ) : status ? (
+
+ {getStateLabel(status.state)}
+
+ ) : null}
+
+
+ {status?.state === "running" && (
+
+
+ {t("progress", "Progress")}
+
+ {status.progress} / {status.total || 0}
+
+
+
+
0 ? (status.progress / status.total) * 100 : 0}%`,
+ }}
+ />
+
+
+ )}
+
+ {status?.state === "completed" && (
+
+ {t("recalculationCompleted", "Recalculation completed successfully")}
+
+ )}
+
+ {status?.state === "failed" && (
+
+ {t("recalculationFailed", "Recalculation failed. Please try again.")}
+
+ )}
+
+
+ {/* Recalculate Buttons */}
+
+
+ {/* Average Mode Recalculate */}
+
+
+ {t("averageMode", "Average Mode")}
+
+
handleRecalculate("average")}
+ disabled={recalculating === "average" || status?.state === "running"}
+ className="w-full"
+ variant="outline"
+ >
+ {recalculating === "average" && (
+
+ )}
+ {t("recalculate", "Recalculate")}
+
+
+
+ {/* Subscribe Mode Recalculate */}
+
+
+ {t("subscribeMode", "Subscribe Mode")}
+
+
handleRecalculate("subscribe")}
+ disabled={recalculating === "subscribe" || status?.state === "running"}
+ className="w-full"
+ variant="outline"
+ >
+ {recalculating === "subscribe" && (
+
+ )}
+ {t("recalculate", "Recalculate")}
+
+
+
+ {/* Traffic Mode Recalculate */}
+
+
+ {t("trafficMode", "Traffic Mode")}
+
+
handleRecalculate("traffic")}
+ disabled={recalculating === "traffic" || status?.state === "running"}
+ className="w-full"
+ variant="outline"
+ >
+ {recalculating === "traffic" && (
+
+ )}
+ {t("recalculate", "Recalculate")}
+
+
+
+
+
+ {/* Warning */}
+
+ {t("warning", "Warning")}: {" "}
+ {t(
+ "recalculationWarning",
+ "Recalculation will reassign all users to new groups based on current configuration. This operation cannot be undone."
+ )}
+
+
+
+
+ );
+}
diff --git a/apps/admin/src/sections/group/index.tsx b/apps/admin/src/sections/group/index.tsx
new file mode 100644
index 0000000..12c397e
--- /dev/null
+++ b/apps/admin/src/sections/group/index.tsx
@@ -0,0 +1,85 @@
+"use client";
+
+import { Tabs, TabsContent, TabsList, TabsTrigger } from "@workspace/ui/components/tabs";
+import { useTranslation } from "react-i18next";
+// import UserGroups from "./user-groups";
+import NodeGroups from "./node-groups";
+import GroupHistory from "./group-history";
+import GroupConfig from "./group-config";
+import AverageModeTab from "./average-mode-tab";
+import SubscribeModeTab from "./subscribe-mode-tab";
+import TrafficModeTab from "./traffic-mode-tab";
+import CurrentGroupResults from "./current-group-results";
+
+export default function Group() {
+ const { t } = useTranslation("group");
+
+ return (
+
+
+ {t("title", "Group Management")}
+
+
+
+
+
+ {t("config", "Config")}
+
+ {/*
+ {t("userGroups", "User Groups")}
+ */}
+
+ {t("nodeGroups", "Node Groups")}
+
+
+ {t("averageMode", "Average Mode")}
+
+
+ {t("subscribeMode", "Subscribe Mode")}
+
+
+ {t("trafficMode", "Traffic Mode")}
+
+
+ {t("currentGroupingResult", "Current Grouping Result")}
+
+
+ {t("history", "History")}
+
+
+
+
+
+
+
+ {/*
+
+ */}
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+
+ );
+}
diff --git a/apps/admin/src/sections/group/node-group-form.tsx b/apps/admin/src/sections/group/node-group-form.tsx
new file mode 100644
index 0000000..7fd553f
--- /dev/null
+++ b/apps/admin/src/sections/group/node-group-form.tsx
@@ -0,0 +1,300 @@
+"use client";
+
+import {
+ Dialog,
+ DialogContent,
+ DialogDescription,
+ DialogHeader,
+ DialogTitle,
+ DialogTrigger,
+} from "@workspace/ui/components/dialog";
+import { Input } from "@workspace/ui/components/input";
+import { Label } from "@workspace/ui/components/label";
+import { Textarea } from "@workspace/ui/components/textarea";
+import { Switch } from "@workspace/ui/components/switch";
+import { AlertCircle, Loader2 } from "lucide-react";
+import { forwardRef, useEffect, useState } from "react";
+import { useTranslation } from "react-i18next";
+
+interface NodeGroupFormProps {
+ initialValues?: Partial;
+ allNodeGroups?: API.NodeGroup[];
+ currentGroupId?: number;
+ loading?: boolean;
+ onSubmit: (values: Record) => Promise;
+ title: string;
+ trigger: React.ReactNode;
+}
+
+const NodeGroupForm = forwardRef<
+ HTMLButtonElement,
+ NodeGroupFormProps
+>(({ initialValues, allNodeGroups = [], currentGroupId, loading, onSubmit, title, trigger }, ref) => {
+ const { t } = useTranslation("group");
+ const [open, setOpen] = useState(false);
+ const [submitting, setSubmitting] = useState(false);
+ const [conflictError, setConflictError] = useState("");
+
+ const [values, setValues] = useState({
+ name: "",
+ description: "",
+ sort: 0,
+ for_calculation: true,
+ min_traffic_gb: 0,
+ max_traffic_gb: 0,
+ });
+
+ useEffect(() => {
+ if (open) {
+ setConflictError(""); // 重置冲突错误
+ if (initialValues) {
+ setValues({
+ name: initialValues.name || "",
+ description: initialValues.description || "",
+ sort: initialValues.sort ?? 0,
+ for_calculation: initialValues.for_calculation ?? true,
+ min_traffic_gb: initialValues.min_traffic_gb ?? 0,
+ max_traffic_gb: initialValues.max_traffic_gb ?? 0,
+ });
+ } else {
+ setValues({
+ name: "",
+ description: "",
+ sort: 0,
+ for_calculation: true,
+ min_traffic_gb: 0,
+ max_traffic_gb: 0,
+ });
+ }
+ }
+ }, [initialValues, open]);
+
+ // 检测流量区间冲突
+ const checkTrafficRangeConflict = (minTraffic: number, maxTraffic: number): string => {
+ // 如果 min=0 且 max=0,表示不参与流量分组,跳过所有验证
+ if (minTraffic === 0 && maxTraffic === 0) {
+ return "";
+ }
+
+ // 验证区间有效性:min 必须 < max(除非 max=0 表示无上限)
+ if (minTraffic > 0 && maxTraffic > 0 && minTraffic >= maxTraffic) {
+ return t("invalidRange", "Min traffic must be less than max traffic");
+ }
+
+ // 处理 max=0 的情况,表示无上限,使用一个很大的数代替
+ const actualMax = maxTraffic === 0 ? Number.MAX_VALUE : maxTraffic;
+
+ // 检查与其他节点组的冲突
+ for (const group of allNodeGroups) {
+ // 跳过当前编辑的节点组
+ if (currentGroupId && group.id === currentGroupId) {
+ continue;
+ }
+
+ // 跳过没有设置流量区间的节点组(min=0 且 max=0 表示未配置)
+ const existingMin = group.min_traffic_gb ?? 0;
+ const existingMax = group.max_traffic_gb ?? 0;
+ if (existingMin === 0 && existingMax === 0) {
+ continue;
+ }
+
+ // 处理现有节点组 max=0 的情况
+ const actualExistingMax = existingMax === 0 ? Number.MAX_VALUE : existingMax;
+
+ // 检测区间重叠
+ // 两个区间 [min1, max1] 和 [min2, max2] 重叠的条件:
+ // max1 > min2 && max2 > min1
+ const hasOverlap = actualMax > existingMin && actualExistingMax > minTraffic;
+
+ if (hasOverlap) {
+ return t("rangeConflict", {
+ name: group.name,
+ min: existingMin.toString(),
+ max: existingMax === 0 ? "∞" : existingMax.toString(),
+ });
+ }
+ }
+
+ return "";
+ };
+
+ const handleSubmit = async (e: React.FormEvent) => {
+ e.preventDefault();
+
+ // 检测流量区间冲突
+ const conflict = checkTrafficRangeConflict(values.min_traffic_gb, values.max_traffic_gb);
+ if (conflict) {
+ setConflictError(conflict);
+ return;
+ }
+
+ setSubmitting(true);
+ const success = await onSubmit(values);
+ setSubmitting(false);
+ if (success) {
+ setOpen(false);
+ setConflictError("");
+ setValues({
+ name: "",
+ description: "",
+ sort: 0,
+ for_calculation: true,
+ min_traffic_gb: 0,
+ max_traffic_gb: 0,
+ });
+ }
+ };
+
+ return (
+
+
+ {trigger}
+
+
+
+ {title}
+
+ {t("nodeGroupFormDescription", "Configure node group settings")}
+
+
+
+
+
+ );
+});
+
+NodeGroupForm.displayName = "NodeGroupForm";
+
+export default NodeGroupForm;
diff --git a/apps/admin/src/sections/group/node-groups.tsx b/apps/admin/src/sections/group/node-groups.tsx
new file mode 100644
index 0000000..8811693
--- /dev/null
+++ b/apps/admin/src/sections/group/node-groups.tsx
@@ -0,0 +1,224 @@
+"use client";
+
+import { Badge } from "@workspace/ui/components/badge";
+import { Button } from "@workspace/ui/components/button";
+import {
+ Card,
+ CardContent,
+ CardDescription,
+ CardHeader,
+ CardTitle,
+} from "@workspace/ui/components/card";
+import { ConfirmButton } from "@workspace/ui/composed/confirm-button";
+import {
+ ProTable,
+ type ProTableActions,
+} from "@workspace/ui/composed/pro-table/pro-table";
+import {
+ createNodeGroup,
+ deleteNodeGroup,
+ getNodeGroupList,
+ updateNodeGroup,
+} from "@workspace/ui/services/admin/group";
+import { useEffect, useRef, useState } from "react";
+import { useTranslation } from "react-i18next";
+import { toast } from "sonner";
+import NodeGroupForm from "./node-group-form";
+
+export default function NodeGroups() {
+ const { t } = useTranslation("group");
+ const [loading, setLoading] = useState(false);
+ const [allNodeGroups, setAllNodeGroups] = useState([]);
+ const ref = useRef(null);
+
+ // 获取所有节点组数据(用于冲突检测)
+ useEffect(() => {
+ const fetchAllNodeGroups = async () => {
+ try {
+ const { data } = await getNodeGroupList({ page: 1, size: 1000 });
+ setAllNodeGroups(data.data?.list || []);
+ } catch (error) {
+ console.error("Failed to fetch node groups:", error);
+ }
+ };
+ fetchAllNodeGroups();
+ }, []);
+
+ return (
+
+
+
+ {t("nodeGroups", "Node Groups")}
+
+ {t("nodeGroupsDescription", "Manage node groups for user access control")}
+
+
+
+
+ action={ref}
+ request={async (params) => {
+ const { data } = await getNodeGroupList({
+ page: params.page || 1,
+ size: params.size || 10,
+ });
+ return {
+ list: data.data?.list || [],
+ total: data.data?.total || 0,
+ };
+ }}
+ columns={[
+ {
+ id: "id",
+ accessorKey: "id",
+ header: t("id", "ID"),
+ cell: ({ row }: { row: any }) => #{row.getValue("id")} ,
+ },
+ {
+ id: "name",
+ accessorKey: "name",
+ header: t("name", "Name"),
+ },
+ {
+ id: "description",
+ accessorKey: "description",
+ header: t("description", "Description"),
+ cell: ({ row }: { row: any }) => row.getValue("description") || "--",
+ },
+ {
+ id: "for_calculation",
+ accessorKey: "for_calculation",
+ header: t("forCalculation", "For Calculation"),
+ cell: ({ row }: { row: any }) => {
+ const value = row.getValue("for_calculation");
+ return value ? (
+ {t("yes", "Yes")}
+ ) : (
+ {t("no", "No")}
+ );
+ },
+ },
+ {
+ id: "traffic_range",
+ accessorKey: "traffic_range",
+ header: t("trafficRange", "Traffic Range (GB)"),
+ cell: ({ row }: { row: any }) => {
+ const min = row.original.min_traffic_gb;
+ const max = row.original.max_traffic_gb;
+ if (min !== undefined && max !== undefined) {
+ return `${min} - ${max}`;
+ }
+ if (min !== undefined) {
+ return `≥ ${min}`;
+ }
+ if (max !== undefined) {
+ return `≤ ${max}`;
+ }
+ return "--";
+ },
+ },
+ {
+ id: "sort",
+ accessorKey: "sort",
+ header: t("sort", "Sort"),
+ },
+ ]}
+ actions={{
+ render: (row: any) => [
+ {
+ setLoading(true);
+ try {
+ await updateNodeGroup({
+ id: row.id,
+ ...values,
+ } as API.UpdateNodeGroupRequest);
+ toast.success(t("updated", "Updated successfully"));
+ // 刷新节点组列表
+ const { data } = await getNodeGroupList({ page: 1, size: 1000 });
+ setAllNodeGroups(data.data?.list || []);
+ ref.current?.refresh();
+ setLoading(false);
+ return true;
+ } catch {
+ setLoading(false);
+ return false;
+ }
+ }}
+ title={t("editNodeGroup", "Edit Node Group")}
+ trigger={
+
+ {t("edit", "Edit")}
+
+ }
+ />,
+ {
+ await deleteNodeGroup({ id: row.id });
+ toast.success(t("deleted", "Deleted successfully"));
+ // 刷新节点组列表
+ const { data } = await getNodeGroupList({ page: 1, size: 1000 });
+ setAllNodeGroups(data.data?.list || []);
+ ref.current?.refresh();
+ setLoading(false);
+ }}
+ title={t("confirmDelete", "Confirm Delete")}
+ trigger={
+
+ {t("delete", "Delete")}
+
+ }
+ />,
+ ],
+ }}
+ header={{
+ title: t("nodeGroups", "Node Groups"),
+ toolbar: (
+ {
+ setLoading(true);
+ try {
+ await createNodeGroup(values as API.CreateNodeGroupRequest);
+ toast.success(t("created", "Created successfully"));
+ // 刷新节点组列表
+ const { data } = await getNodeGroupList({ page: 1, size: 1000 });
+ setAllNodeGroups(data.data?.list || []);
+ ref.current?.refresh();
+ setLoading(false);
+ return true;
+ } catch {
+ setLoading(false);
+ return false;
+ }
+ }}
+ title={t("createNodeGroup", "Create Node Group")}
+ trigger={
+
+ {t("create", "Create")}
+
+ }
+ />
+ ),
+ }}
+ />
+
+
+
+ );
+}
diff --git a/apps/admin/src/sections/group/subscribe-mode-tab.tsx b/apps/admin/src/sections/group/subscribe-mode-tab.tsx
new file mode 100644
index 0000000..ff0a408
--- /dev/null
+++ b/apps/admin/src/sections/group/subscribe-mode-tab.tsx
@@ -0,0 +1,260 @@
+"use client";
+
+import { Badge } from "@workspace/ui/components/badge";
+import { Button } from "@workspace/ui/components/button";
+import {
+ Card,
+ CardContent,
+ CardDescription,
+ CardHeader,
+ CardTitle,
+} from "@workspace/ui/components/card";
+import {
+ Table,
+ TableBody,
+ TableCell,
+ TableRow,
+} from "@workspace/ui/components/table";
+import { Loader2 } from "lucide-react";
+import { useEffect, useState } from "react";
+import { useTranslation } from "react-i18next";
+import { toast } from "sonner";
+import { useQuery } from "@tanstack/react-query";
+import {
+ getRecalculationStatus,
+ recalculateGroup,
+ getSubscribeGroupMapping,
+} from "@workspace/ui/services/admin/group";
+
+interface SubscribeGroupMapping {
+ subscribe_name: string;
+ node_group_name: string;
+}
+
+export default function SubscribeModeTab() {
+ const { t } = useTranslation("group");
+ const [recalculating, setRecalculating] = useState(false);
+ const [loadingStatus, setLoadingStatus] = useState(false);
+
+ const [status, setStatus] = useState<{
+ state: string;
+ progress: number;
+ total: number;
+ } | null>(null);
+
+ // Fetch subscribe group mapping
+ const { data: mappingData, isLoading: mappingLoading } = useQuery({
+ queryKey: ["subscribeGroupMapping"],
+ queryFn: async () => {
+ const { data } = await getSubscribeGroupMapping();
+ return (data.data?.list || []) as SubscribeGroupMapping[];
+ },
+ });
+
+
+ const loadStatus = async () => {
+ setLoadingStatus(true);
+ try {
+ const { data } = await getRecalculationStatus();
+ if (data.data) {
+ setStatus(data.data);
+ }
+ } catch (error) {
+ console.error("Failed to load recalculation status:", error);
+ } finally {
+ setLoadingStatus(false);
+ }
+ };
+
+ useEffect(() => {
+ loadStatus();
+ }, []);
+
+ useEffect(() => {
+ const interval = setInterval(() => {
+ if (status?.state === "running") {
+ loadStatus();
+ }
+ }, 2000);
+ return () => clearInterval(interval);
+ }, [status?.state]);
+
+ const handleRecalculate = async () => {
+ setRecalculating(true);
+ try {
+ await recalculateGroup({ mode: "subscribe" });
+ toast.success(t("recalculationStarted", "Recalculation started"));
+ loadStatus();
+ } catch (error) {
+ console.error("Failed to start recalculation:", error);
+ toast.error(t("recalculationFailed", "Failed to start recalculation"));
+ } finally {
+ setRecalculating(false);
+ }
+ };
+
+ const getStateLabel = (state: string) => {
+ switch (state) {
+ case "running":
+ return t("running", "Running");
+ case "completed":
+ return t("completed", "Completed");
+ case "failed":
+ return t("failed", "Failed");
+ default:
+ return t("idle", "Idle");
+ }
+ };
+
+ const getStateVariant = (state: string) => {
+ switch (state) {
+ case "running":
+ return "default";
+ case "completed":
+ return "secondary";
+ case "failed":
+ return "destructive";
+ default:
+ return "outline";
+ }
+ };
+
+ return (
+
+ {/* Configuration Card */}
+
+
+ {t("subscribeModeConfig", "Subscribe Mode Configuration")}
+
+ {t("subscribeModeDescription", "Group users by their purchased subscription plan")}
+
+
+
+
+ {/* Subscribe Group Mapping Card */}
+
+
+ {t("subscribeGroupMappingTitle", "套餐-节点组对应关系")}
+
+
+ {mappingLoading ? (
+
+
+
+ ) : (
+
+
+ {mappingData && mappingData.length > 0 ? (
+ mappingData
+ .filter(
+ (item: SubscribeGroupMapping) => item.subscribe_name && item.node_group_name
+ )
+ .map((item: SubscribeGroupMapping, index: number) => (
+
+
+ {item.subscribe_name}
+
+ {t("arrow", "→")}
+
+ {item.node_group_name}
+
+
+ ))
+ ) : (
+
+
+ {t("noMappingData", "No mapping data available")}
+
+
+ )}
+
+
+ )}
+
+
+
+ {/* Recalculation Card */}
+
+
+ {t("groupRecalculation", "Group Recalculation")}
+
+ {t(
+ "groupRecalculationDescription",
+ "Manually trigger a full recalculation of all user groups based on current configuration"
+ )}
+
+
+
+ {/* Current Status */}
+
+
+
+ {t("currentStatus", "Current Status")}
+
+ {loadingStatus ? (
+
+ ) : status ? (
+
+ {getStateLabel(status.state)}
+
+ ) : null}
+
+
+ {status?.state === "running" && (
+
+
+ {t("progress", "Progress")}
+
+ {status.progress} / {status.total || 0}
+
+
+
+
0 ? (status.progress / status.total) * 100 : 0}%`,
+ }}
+ />
+
+
+ )}
+
+ {status?.state === "completed" && (
+
+ {t("recalculationCompleted", "Recalculation completed successfully")}
+
+ )}
+
+ {status?.state === "failed" && (
+
+ {t("recalculationFailed", "Recalculation failed. Please try again.")}
+
+ )}
+
+
+ {/* Recalculate Button */}
+
+
+ {recalculating && (
+
+ )}
+ {t("recalculateAll", "Recalculate All Users")}
+
+
+
+ {/* Warning */}
+
+ {t("warning", "Warning")}: {" "}
+ {t(
+ "recalculationWarning",
+ "Recalculation will reassign all users to new groups based on current configuration. This operation cannot be undone."
+ )}
+
+
+
+
+ );
+}
diff --git a/apps/admin/src/sections/group/traffic-mode-tab.tsx b/apps/admin/src/sections/group/traffic-mode-tab.tsx
new file mode 100644
index 0000000..8661118
--- /dev/null
+++ b/apps/admin/src/sections/group/traffic-mode-tab.tsx
@@ -0,0 +1,228 @@
+"use client";
+
+import { Badge } from "@workspace/ui/components/badge";
+import { Button } from "@workspace/ui/components/button";
+import {
+ Card,
+ CardContent,
+ CardDescription,
+ CardHeader,
+ CardTitle,
+} from "@workspace/ui/components/card";
+import { Loader2 } from "lucide-react";
+import { useState } from "react";
+import { useTranslation } from "react-i18next";
+import { toast } from "sonner";
+import { useQuery } from "@tanstack/react-query";
+import {
+ getNodeGroupList,
+ updateNodeGroup,
+ getRecalculationStatus,
+ recalculateGroup,
+} from "@workspace/ui/services/admin/group";
+import TrafficRangeConfig from "./traffic-ranges-config";
+
+export default function TrafficModeTab() {
+ const { t } = useTranslation("group");
+ const [recalculating, setRecalculating] = useState(false);
+ const [loadingStatus, setLoadingStatus] = useState(false);
+
+ const [status, setStatus] = useState<{
+ state: string;
+ progress: number;
+ total: number;
+ } | null>(null);
+
+ // Fetch node groups
+ const { data: nodeGroupsData, isLoading: isLoadingNodeGroups, refetch: refetchNodeGroups } = useQuery({
+ queryKey: ["nodeGroups"],
+ queryFn: async () => {
+ const { data } = await getNodeGroupList({ page: 1, size: 1000 });
+ return data.data?.list || [];
+ },
+ });
+
+ const loadStatus = async () => {
+ setLoadingStatus(true);
+ try {
+ const { data } = await getRecalculationStatus();
+ if (data.data) {
+ setStatus(data.data);
+ }
+ } catch (error) {
+ console.error("Failed to load recalculation status:", error);
+ } finally {
+ setLoadingStatus(false);
+ }
+ };
+
+ const handleTrafficUpdate = async (nodeGroupId: number, fields: { min_traffic_gb?: number; max_traffic_gb?: number }) => {
+ try {
+ await updateNodeGroup({
+ id: nodeGroupId,
+ ...fields,
+ });
+ toast.success(t("configSaved", "Configuration saved successfully"));
+ // Refetch node groups to get updated data
+ refetchNodeGroups();
+ } catch (error) {
+ console.error("Failed to update node group:", error);
+ toast.error(t("saveFailed", "Failed to save configuration"));
+ }
+ };
+
+ const handleRecalculate = async () => {
+ setRecalculating(true);
+ try {
+ await recalculateGroup({ mode: "traffic" });
+ toast.success(t("recalculationStarted", "Recalculation started"));
+ loadStatus();
+ } catch (error) {
+ console.error("Failed to start recalculation:", error);
+ toast.error(t("recalculationFailed", "Failed to start recalculation"));
+ } finally {
+ setRecalculating(false);
+ }
+ };
+
+ const getStateLabel = (state: string) => {
+ switch (state) {
+ case "running":
+ return t("running", "Running");
+ case "completed":
+ return t("completed", "Completed");
+ case "failed":
+ return t("failed", "Failed");
+ default:
+ return t("idle", "Idle");
+ }
+ };
+
+ const getStateVariant = (state: string) => {
+ switch (state) {
+ case "running":
+ return "default";
+ case "completed":
+ return "secondary";
+ case "failed":
+ return "destructive";
+ default:
+ return "outline";
+ }
+ };
+
+ return (
+
+ {/* Node Groups Traffic Configuration Card */}
+
+
+ {t("trafficModeConfig", "Traffic Mode Configuration")}
+
+ {t(
+ "trafficModeDescription",
+ "Configure traffic ranges for node groups. Users will be assigned to node groups based on their traffic usage."
+ )}
+
+
+
+ {isLoadingNodeGroups ? (
+
+
+
+ {t("loading", "Loading...")}
+
+
+ ) : (
+
+ )}
+
+
+
+ {/* Recalculation Card */}
+
+
+ {t("groupRecalculation", "Group Recalculation")}
+
+ {t(
+ "groupRecalculationDescription",
+ "Manually trigger a full recalculation of all user groups based on current configuration"
+ )}
+
+
+
+ {/* Current Status */}
+
+
+
+ {t("currentStatus", "Current Status")}
+
+ {loadingStatus ? (
+
+ ) : status ? (
+
+ {getStateLabel(status.state)}
+
+ ) : null}
+
+
+ {status?.state === "running" && (
+
+
+ {t("progress", "Progress")}
+
+ {status.progress} / {status.total || 0}
+
+
+
+
0 ? (status.progress / status.total) * 100 : 0}%`,
+ }}
+ />
+
+
+ )}
+
+ {status?.state === "completed" && (
+
+ {t("recalculationCompleted", "Recalculation completed successfully")}
+
+ )}
+
+ {status?.state === "failed" && (
+
+ {t("recalculationFailed", "Recalculation failed. Please try again.")}
+
+ )}
+
+
+ {/* Recalculate Button */}
+
+
+ {recalculating && (
+
+ )}
+ {t("recalculateAll", "Recalculate All Users")}
+
+
+
+ {/* Warning */}
+
+ {t("warning", "Warning")}: {" "}
+ {t(
+ "recalculationWarning",
+ "Recalculation will reassign all users to new groups based on current configuration. This operation cannot be undone."
+ )}
+
+
+
+
+ );
+}
diff --git a/apps/admin/src/sections/group/traffic-ranges-config.tsx b/apps/admin/src/sections/group/traffic-ranges-config.tsx
new file mode 100644
index 0000000..a15e84f
--- /dev/null
+++ b/apps/admin/src/sections/group/traffic-ranges-config.tsx
@@ -0,0 +1,247 @@
+"use client";
+
+import { Input } from "@workspace/ui/components/input";
+import { Loader2 } from "lucide-react";
+import { useTranslation } from "react-i18next";
+import { useState } from "react";
+import { toast } from "sonner";
+
+interface NodeGroup {
+ id: number;
+ name: string;
+ min_traffic_gb?: number;
+ max_traffic_gb?: number;
+}
+
+interface TrafficRangeConfigProps {
+ nodeGroups: NodeGroup[];
+ onTrafficUpdate: (nodeGroupId: number, fields: { min_traffic_gb?: number; max_traffic_gb?: number }) => Promise;
+}
+
+interface UpdatingNode {
+ nodeGroupId: number;
+ field: 'min_traffic_gb' | 'max_traffic_gb';
+}
+
+interface NodeGroupTempValues {
+ min_traffic_gb?: number;
+ max_traffic_gb?: number;
+}
+
+export default function TrafficRangeConfig({ nodeGroups, onTrafficUpdate }: TrafficRangeConfigProps) {
+ const { t } = useTranslation("group");
+ const [updatingNodes, setUpdatingNodes] = useState([]);
+ // 使用对象存储每个节点组的临时值
+ const [temporaryValues, setTemporaryValues] = useState>({});
+
+ // Get the display value (temporary or actual)
+ const getDisplayValue = (nodeGroupId: number, field: 'min_traffic_gb' | 'max_traffic_gb'): number => {
+ const temp = temporaryValues[nodeGroupId];
+ if (temp && temp[field] !== undefined) {
+ return temp[field]!;
+ }
+ const nodeGroup = nodeGroups.find(ng => ng.id === nodeGroupId);
+ return field === 'min_traffic_gb' ? (nodeGroup?.min_traffic_gb ?? 0) : (nodeGroup?.max_traffic_gb ?? 0);
+ };
+
+ // Validate traffic ranges: no overlaps
+ const validateTrafficRange = (
+ nodeGroupId: number,
+ minTraffic: number,
+ maxTraffic: number
+ ): { valid: boolean; error?: string } => {
+ // 如果 min=0 且 max=0,表示不参与流量分组,跳过验证
+ if (minTraffic === 0 && maxTraffic === 0) {
+ return { valid: true };
+ }
+
+ // Check if min >= max (both > 0)
+ if (minTraffic > 0 && maxTraffic > 0 && minTraffic >= maxTraffic) {
+ return { valid: false, error: t("minCannotExceedMax", "Minimum traffic cannot exceed maximum traffic") };
+ }
+
+ // Check for overlaps with other node groups
+ const otherGroups = nodeGroups
+ .filter(ng => ng.id !== nodeGroupId)
+ .map(ng => {
+ const temp = temporaryValues[ng.id];
+ return {
+ id: ng.id,
+ name: ng.name,
+ min: temp?.min_traffic_gb !== undefined ? temp.min_traffic_gb : (ng.min_traffic_gb ?? 0),
+ max: temp?.max_traffic_gb !== undefined ? temp.max_traffic_gb : (ng.max_traffic_gb ?? 0),
+ };
+ })
+ .filter(ng => !(ng.min === 0 && ng.max === 0)) // 跳过未配置流量区间的组
+ .sort((a, b) => a.min - b.min);
+
+ for (const other of otherGroups) {
+ // Handle max=0 as no limit (infinity)
+ const otherMax = other.max === 0 ? Number.MAX_VALUE : other.max;
+ const currentMax = maxTraffic === 0 ? Number.MAX_VALUE : maxTraffic;
+
+ // Check for overlap: two ranges [min1, max1] and [min2, max2] overlap if:
+ // max1 > min2 && max2 > min1
+ if (currentMax > other.min && otherMax > minTraffic) {
+ return {
+ valid: false,
+ error: t("rangeOverlap", "Range overlaps with node group \"{{name}}\"", { name: other.name })
+ };
+ }
+ }
+
+ return { valid: true };
+ };
+
+ const handleTrafficBlur = async (nodeGroupId: number) => {
+ const nodeGroup = nodeGroups.find(ng => ng.id === nodeGroupId);
+ if (!nodeGroup) return;
+
+ const tempValues = temporaryValues[nodeGroupId];
+ if (!tempValues) return;
+
+ // 获取当前的临时值或实际值
+ const currentMin = tempValues.min_traffic_gb !== undefined
+ ? tempValues.min_traffic_gb
+ : (nodeGroup.min_traffic_gb ?? 0);
+ const currentMax = tempValues.max_traffic_gb !== undefined
+ ? tempValues.max_traffic_gb
+ : (nodeGroup.max_traffic_gb ?? 0);
+
+ // 只要有一个字段被修改了就保存
+ const hasMinChange = tempValues.min_traffic_gb !== undefined;
+ const hasMaxChange = tempValues.max_traffic_gb !== undefined;
+ if (!hasMinChange && !hasMaxChange) {
+ return;
+ }
+
+ // 验证
+ const validation = validateTrafficRange(nodeGroupId, currentMin, currentMax);
+ if (!validation.valid) {
+ toast.error(validation.error || t("validationFailed", "Validation failed"));
+ return;
+ }
+
+ // 检查值是否真的改变了
+ const originalMin = nodeGroup.min_traffic_gb ?? 0;
+ const originalMax = nodeGroup.max_traffic_gb ?? 0;
+ if (currentMin === originalMin && currentMax === originalMax) {
+ return;
+ }
+
+ // 标记为更新中(只标记被修改的字段)
+ if (hasMinChange) {
+ setUpdatingNodes(prev => [...prev, { nodeGroupId, field: 'min_traffic_gb' }]);
+ }
+ if (hasMaxChange) {
+ setUpdatingNodes(prev => [...prev, { nodeGroupId, field: 'max_traffic_gb' }]);
+ }
+
+ try {
+ // 一次性传递两个字段
+ const fieldsToUpdate: { min_traffic_gb?: number; max_traffic_gb?: number } = {};
+ if (currentMin !== originalMin) {
+ fieldsToUpdate.min_traffic_gb = currentMin;
+ }
+ if (currentMax !== originalMax) {
+ fieldsToUpdate.max_traffic_gb = currentMax;
+ }
+
+ if (Object.keys(fieldsToUpdate).length > 0) {
+ await onTrafficUpdate(nodeGroupId, fieldsToUpdate);
+ }
+ } finally {
+ // 移除更新状态
+ setUpdatingNodes(prev => prev.filter(u => !(u.nodeGroupId === nodeGroupId)));
+ }
+ };
+
+ const isUpdating = (nodeGroupId: number) => {
+ return updatingNodes.some(u => u.nodeGroupId === nodeGroupId);
+ };
+
+ return (
+ <>
+
+
+
{t("nodeGroup", "Node Group")}
+
{t("minTrafficGB", "Min (GB)")}
+
{t("maxTrafficGB", "Max (GB)")}
+
+
+ {nodeGroups.map((nodeGroup) => (
+
+
+
{nodeGroup.name}
+
{t("id", "ID")}: {nodeGroup.id}
+
+
+
{
+ const newValue = parseFloat(e.target.value) || 0;
+ // 更新临时状态
+ setTemporaryValues(prev => ({
+ ...prev,
+ [nodeGroup.id]: {
+ ...prev[nodeGroup.id],
+ min_traffic_gb: newValue,
+ max_traffic_gb: prev[nodeGroup.id]?.max_traffic_gb,
+ },
+ }));
+ }}
+ onBlur={() => handleTrafficBlur(nodeGroup.id)}
+ disabled={isUpdating(nodeGroup.id)}
+ />
+ {isUpdating(nodeGroup.id) && (
+
+
+
+ )}
+
+
+
{
+ const newValue = parseFloat(e.target.value) || 0;
+ // 更新临时状态
+ setTemporaryValues(prev => ({
+ ...prev,
+ [nodeGroup.id]: {
+ ...prev[nodeGroup.id],
+ min_traffic_gb: prev[nodeGroup.id]?.min_traffic_gb,
+ max_traffic_gb: newValue,
+ },
+ }));
+ }}
+ onBlur={() => handleTrafficBlur(nodeGroup.id)}
+ disabled={isUpdating(nodeGroup.id)}
+ />
+ {isUpdating(nodeGroup.id) && (
+
+
+
+ )}
+
+
+ ))}
+
+
+
+ {t("note", "Note")}: {" "}
+ {t(
+ "trafficRangesNote",
+ "Set traffic ranges for each node group. Users will be assigned to node groups based on their traffic usage. Leave both values as 0 to not use this node group for traffic-based assignment."
+ )}
+
+ >
+ );
+}
diff --git a/apps/admin/src/sections/group/user-group-form.tsx b/apps/admin/src/sections/group/user-group-form.tsx
new file mode 100644
index 0000000..9cb805b
--- /dev/null
+++ b/apps/admin/src/sections/group/user-group-form.tsx
@@ -0,0 +1,210 @@
+"use client";
+
+import {
+ Dialog,
+ DialogContent,
+ DialogDescription,
+ DialogHeader,
+ DialogTitle,
+ DialogTrigger,
+} from "@workspace/ui/components/dialog";
+import { Input } from "@workspace/ui/components/input";
+import { Label } from "@workspace/ui/components/label";
+import { Textarea } from "@workspace/ui/components/textarea";
+import {
+ Select,
+ SelectContent,
+ SelectItem,
+ SelectTrigger,
+ SelectValue,
+} from "@workspace/ui/components/select";
+import { Switch } from "@workspace/ui/components/switch";
+import { Loader2 } from "lucide-react";
+import { forwardRef, useEffect, useState } from "react";
+import { useTranslation } from "react-i18next";
+
+interface UserGroupFormProps {
+ initialValues?: Partial;
+ loading?: boolean;
+ nodeGroups?: API.NodeGroup[];
+ onSubmit: (values: Record) => Promise;
+ title: string;
+ trigger: React.ReactNode;
+}
+
+const UserGroupForm = forwardRef<
+ HTMLButtonElement,
+ UserGroupFormProps
+>(({ initialValues, loading, nodeGroups = [], onSubmit, title, trigger }, ref) => {
+ const { t } = useTranslation("group");
+ const [open, setOpen] = useState(false);
+ const [submitting, setSubmitting] = useState(false);
+
+ const [values, setValues] = useState({
+ name: "",
+ description: "",
+ sort: 0,
+ node_group_id: null as number | null,
+ for_calculation: true,
+ });
+
+ useEffect(() => {
+ if (open) {
+ if (initialValues) {
+ setValues({
+ name: initialValues.name || "",
+ description: initialValues.description || "",
+ sort: initialValues.sort ?? 0,
+ node_group_id: initialValues.node_group_id || null,
+ for_calculation: initialValues.for_calculation ?? true,
+ });
+ } else {
+ setValues({
+ name: "",
+ description: "",
+ sort: 0,
+ node_group_id: null,
+ for_calculation: true,
+ });
+ }
+ }
+ }, [initialValues, open]);
+
+ const handleSubmit = async (e: React.FormEvent) => {
+ e.preventDefault();
+ setSubmitting(true);
+ const success = await onSubmit(values);
+ setSubmitting(false);
+ if (success) {
+ setOpen(false);
+ setValues({
+ name: "",
+ description: "",
+ sort: 0,
+ node_group_id: null,
+ for_calculation: true,
+ });
+ }
+ };
+
+ return (
+
+
+ {trigger}
+
+
+
+ {title}
+
+ {t("userGroupFormDescription", "Configure user group settings")}
+
+
+
+
+
+ {t("name", "Name")} *
+
+
+ setValues({ ...values, name: e.target.value })
+ }
+ placeholder={t("namePlaceholder", "Enter name")}
+ required
+ />
+
+
+
+ {t("forCalculation", "For Calculation")}
+
+ setValues({ ...values, for_calculation: checked })
+ }
+ />
+
+
+
+
+ {t("description", "Description")}
+
+
+ setValues({ ...values, description: e.target.value })
+ }
+ placeholder={t("descriptionPlaceholder", "Enter description")}
+ rows={3}
+ />
+
+
+
+ {t("nodeGroup", "Node Group")}
+
+ setValues({
+ ...values,
+ node_group_id: val === "0" ? null : parseInt(val),
+ })
+ }
+ >
+
+
+
+
+
+ {t("unbound", "Unbound")}
+
+ {nodeGroups.map((nodeGroup) => (
+
+ {nodeGroup.name}
+
+ ))}
+
+
+
+
+
+ {t("sort", "Sort Order")}
+
+ setValues({ ...values, sort: parseInt(e.target.value) || 0 })
+ }
+ min={0}
+ />
+
+
+
+ setOpen(false)}
+ className="rounded-md border px-4 py-2 text-sm"
+ disabled={submitting || loading}
+ >
+ {t("cancel", "Cancel")}
+
+
+ {submitting && }
+ {t("save", "Save")}
+
+
+
+
+
+ );
+});
+
+UserGroupForm.displayName = "UserGroupForm";
+
+export default UserGroupForm;
diff --git a/apps/admin/src/sections/group/user-groups.tsx b/apps/admin/src/sections/group/user-groups.tsx
new file mode 100644
index 0000000..3e51da3
--- /dev/null
+++ b/apps/admin/src/sections/group/user-groups.tsx
@@ -0,0 +1,200 @@
+"use client";
+
+import { Button } from "@workspace/ui/components/button";
+import { Badge } from "@workspace/ui/components/badge";
+import {
+ Card,
+ CardContent,
+ CardDescription,
+ CardHeader,
+ CardTitle,
+} from "@workspace/ui/components/card";
+import { ConfirmButton } from "@workspace/ui/composed/confirm-button";
+import {
+ ProTable,
+ type ProTableActions,
+} from "@workspace/ui/composed/pro-table/pro-table";
+import {
+ createUserGroup,
+ deleteUserGroup,
+ getUserGroupList,
+ getNodeGroupList,
+ updateUserGroup,
+} from "@workspace/ui/services/admin/group";
+import { useRef, useState } from "react";
+import { useTranslation } from "react-i18next";
+import { useQuery } from "@tanstack/react-query";
+import { toast } from "sonner";
+import UserGroupForm from "./user-group-form";
+
+export default function UserGroups() {
+ const { t } = useTranslation("group");
+ const [loading, setLoading] = useState(false);
+ const ref = useRef(null);
+
+ const { data: nodeGroupsData } = useQuery({
+ queryKey: ["nodeGroups"],
+ queryFn: async () => {
+ const { data } = await getNodeGroupList({ page: 1, size: 1000 });
+ return data.data?.list || [];
+ },
+ });
+
+ return (
+
+
+
+ {t("userGroups", "User Groups")}
+
+ {t("userGroupsDescription", "Manage user groups for node access control")}
+
+
+
+
+ action={ref}
+ request={async (params) => {
+ const { data } = await getUserGroupList({
+ page: params.page || 1,
+ size: params.size || 10,
+ });
+ return {
+ list: data.data?.list || [],
+ total: data.data?.total || 0,
+ };
+ }}
+ columns={[
+ {
+ id: "id",
+ accessorKey: "id",
+ header: t("id", "ID"),
+ cell: ({ row }: { row: any }) => #{row.getValue("id")} ,
+ },
+ {
+ id: "name",
+ accessorKey: "name",
+ header: t("name", "Name"),
+ },
+ {
+ id: "for_calculation",
+ accessorKey: "for_calculation",
+ header: t("forCalculation", "For Calculation"),
+ cell: ({ row }: { row: any }) => {
+ const forCalculation = row.getValue("for_calculation");
+ return forCalculation ? (
+ {t("yes", "Yes")}
+ ) : (
+ {t("no", "No")}
+ );
+ },
+ },
+ {
+ id: "description",
+ accessorKey: "description",
+ header: t("description", "Description"),
+ cell: ({ row }: { row: any }) => row.getValue("description") || "--",
+ },
+ {
+ id: "node_group_id",
+ accessorKey: "node_group_id",
+ header: t("nodeGroup", "Node Group"),
+ cell: ({ row }: { row: any }) => {
+ const nodeGroupId = row.getValue("node_group_id");
+ const group = nodeGroupsData?.find((g) => g.id === nodeGroupId);
+ return group?.name || "--";
+ },
+ },
+ {
+ id: "sort",
+ accessorKey: "sort",
+ header: t("sort", "Sort"),
+ },
+ ]}
+ actions={{
+ render: (row: any) => [
+ {
+ setLoading(true);
+ try {
+ await updateUserGroup({
+ id: row.id,
+ ...values,
+ } as unknown as API.UpdateUserGroupRequest);
+ toast.success(t("updated", "Updated successfully"));
+ ref.current?.refresh();
+ setLoading(false);
+ return true;
+ } catch {
+ setLoading(false);
+ return false;
+ }
+ }}
+ title={t("editUserGroup", "Edit User Group")}
+ trigger={
+
+ {t("edit", "Edit")}
+
+ }
+ />,
+ {
+ await deleteUserGroup({ id: row.id });
+ toast.success(t("deleted", "Deleted successfully"));
+ ref.current?.refresh();
+ setLoading(false);
+ }}
+ title={t("confirmDelete", "Confirm Delete")}
+ trigger={
+
+ {t("delete", "Delete")}
+
+ }
+ />,
+ ],
+ }}
+ header={{
+ title: t("userGroups", "User Groups"),
+ toolbar: (
+ {
+ setLoading(true);
+ try {
+ await createUserGroup(values as API.CreateUserGroupRequest);
+ toast.success(t("created", "Created successfully"));
+ ref.current?.refresh();
+ setLoading(false);
+ return true;
+ } catch {
+ setLoading(false);
+ return false;
+ }
+ }}
+ title={t("createUserGroup", "Create User Group")}
+ trigger={
+
+ {t("create", "Create")}
+
+ }
+ />
+ ),
+ }}
+ />
+
+
+
+ );
+}
diff --git a/apps/admin/src/sections/log/reset-subscribe/index.tsx b/apps/admin/src/sections/log/reset-subscribe/index.tsx
index 1dfae0c..99ae57a 100644
--- a/apps/admin/src/sections/log/reset-subscribe/index.tsx
+++ b/apps/admin/src/sections/log/reset-subscribe/index.tsx
@@ -25,14 +25,12 @@ export default function ResetSubscribeLogPage() {
const initialFilters = {
date: sp.date || today,
- user_subscribe_id: sp.user_subscribe_id
- ? Number(sp.user_subscribe_id)
- : undefined,
+ user_subscribe_id: sp.user_subscribe_id || undefined,
};
return (
columns={[
{
@@ -83,7 +81,7 @@ export default function ResetSubscribeLogPage() {
page: pagination.page,
size: pagination.size,
date: (filter as any)?.date,
- user_subscribe_id: (filter as any)?.user_subscribe_id,
+ user_subscribe_id: (filter as any)?.user_subscribe_id ? Number((filter as any)?.user_subscribe_id) : undefined,
});
const list = (data?.data?.list || []) as any[];
const total = Number(data?.data?.total || list.length);
diff --git a/apps/admin/src/sections/log/subscribe-traffic/index.tsx b/apps/admin/src/sections/log/subscribe-traffic/index.tsx
index 5a133d1..4ac58d8 100644
--- a/apps/admin/src/sections/log/subscribe-traffic/index.tsx
+++ b/apps/admin/src/sections/log/subscribe-traffic/index.tsx
@@ -17,14 +17,12 @@ export default function SubscribeTrafficLogPage() {
const initialFilters = {
date: sp.date || today,
user_id: sp.user_id ? Number(sp.user_id) : undefined,
- user_subscribe_id: sp.user_subscribe_id
- ? Number(sp.user_subscribe_id)
- : undefined,
+ user_subscribe_id: sp.user_subscribe_id || undefined,
};
return (
actions={{
render: (row) => [
@@ -95,7 +93,7 @@ export default function SubscribeTrafficLogPage() {
size: pagination.size,
date: (filter as any)?.date,
user_id: (filter as any)?.user_id,
- user_subscribe_id: (filter as any)?.user_subscribe_id,
+ user_subscribe_id: (filter as any)?.user_subscribe_id ? Number((filter as any)?.user_subscribe_id) : undefined,
});
const list =
((data?.data?.list || []) as API.UserSubscribeTrafficLog[]) || [];
diff --git a/apps/admin/src/sections/log/subscribe/index.tsx b/apps/admin/src/sections/log/subscribe/index.tsx
index 3a739a8..60fce92 100644
--- a/apps/admin/src/sections/log/subscribe/index.tsx
+++ b/apps/admin/src/sections/log/subscribe/index.tsx
@@ -23,12 +23,10 @@ export default function SubscribeLogPage() {
const initialFilters = {
date: sp.date || today,
user_id: sp.user_id ? Number(sp.user_id) : undefined,
- user_subscribe_id: sp.user_subscribe_id
- ? Number(sp.user_subscribe_id)
- : undefined,
+ user_subscribe_id: sp.user_subscribe_id || undefined,
};
return (
-
+
columns={[
{
accessorKey: "user",
@@ -96,7 +94,7 @@ export default function SubscribeLogPage() {
size: pagination.size,
date: (filter as any)?.date,
user_id: (filter as any)?.user_id,
- user_subscribe_id: (filter as any)?.user_subscribe_id,
+ user_subscribe_id: (filter as any)?.user_subscribe_id ? Number((filter as any)?.user_subscribe_id) : undefined,
});
const list = (data?.data?.list || []) as any[];
const total = Number(data?.data?.total || list.length);
diff --git a/apps/admin/src/sections/nodes/index.tsx b/apps/admin/src/sections/nodes/index.tsx
index 788a915..edd8547 100644
--- a/apps/admin/src/sections/nodes/index.tsx
+++ b/apps/admin/src/sections/nodes/index.tsx
@@ -16,7 +16,9 @@ import {
toggleNodeStatus,
updateNode,
} from "@workspace/ui/services/admin/server";
-import { useRef, useState } from "react";
+import { getGroupConfig, getNodeGroupList } from "@workspace/ui/services/admin/group";
+import { useQuery } from "@tanstack/react-query";
+import { useMemo, useRef, useState } from "react";
import { useTranslation } from "react-i18next";
import { toast } from "sonner";
import { useNode } from "@/stores/node";
@@ -32,13 +34,132 @@ export default function Nodes() {
const { getServerName, getServerAddress, getProtocolPort } = useServer();
const { fetchNodes, fetchTags } = useNode();
+ // Fetch node groups for display
+ const { data: nodeGroupsData } = useQuery({
+ queryKey: ["nodeGroups"],
+ queryFn: async () => {
+ const { data } = await getNodeGroupList({ page: 1, size: 1000 });
+ return data.data?.list || [];
+ },
+ });
+
+ // Fetch group config to check if group feature is enabled
+ const { data: groupConfigData } = useQuery({
+ queryKey: ["groupConfig"],
+ queryFn: async () => {
+ const { data } = await getGroupConfig();
+ return data.data;
+ },
+ });
+
+ const isGroupEnabled = groupConfigData?.enabled || false;
+
+ // Dynamic columns based on group feature status
+ const columns = useMemo(() => {
+ const baseColumns = [
+ {
+ id: "enabled",
+ header: t("enabled", "Enabled"),
+ cell: ({ row }: { row: any }) => (
+ {
+ await toggleNodeStatus({ id: row.original.id, enable: v });
+ toast.success(
+ v ? t("enabled_on", "Enabled") : t("enabled_off", "Disabled")
+ );
+ ref.current?.refresh();
+ fetchNodes();
+ fetchTags();
+ }}
+ />
+ ),
+ },
+ {
+ id: "name",
+ accessorKey: "name",
+ header: t("name", "Name"),
+ },
+ {
+ id: "address_port",
+ header: `${t("address", "Address")}:${t("port", "Port")}`,
+ cell: ({ row }: { row: any }) =>
+ `${row.original.address || "—"}:${row.original.port || "—"}`,
+ },
+ {
+ id: "server_id",
+ header: t("server", "Server"),
+ cell: ({ row }: { row: any }) =>
+ `${getServerName(row.original.server_id)}:${getServerAddress(row.original.server_id)}`,
+ },
+ {
+ id: "protocol",
+ header: ` ${t("protocol", "Protocol")}:${t("port", "Port")}`,
+ cell: ({ row }: { row: any }) =>
+ `${row.original.protocol}:${getProtocolPort(row.original.server_id, row.original.protocol)}`,
+ },
+ {
+ id: "tags",
+ header: t("tags", "Tags"),
+ cell: ({ row }: { row: any }) => (
+
+ {(row.original.tags || []).length === 0
+ ? "—"
+ : row.original.tags.map((tg: string) => (
+
+ {tg}
+
+ ))}
+
+ ),
+ },
+ ];
+
+ // Add Node Groups column when group feature is enabled
+ if (isGroupEnabled) {
+ baseColumns.push({
+ id: "node_group_ids",
+ header: t("nodeGroups", "Node Groups"),
+ cell: ({ row }: { row: any }) => {
+ const groupIds = row.original.node_group_ids as number[] || [];
+
+ // Public node indicator (when node_group_ids is empty)
+ if (groupIds.length === 0) {
+ return (
+
+
+ {t("public", "Public")}
+
+
+ );
+ }
+
+ return (
+
+ {groupIds.map((groupId) => {
+ const group = nodeGroupsData?.find((g) => g.id === groupId);
+ return (
+
+ {group?.name || String(groupId)}
+
+ );
+ })}
+
+ );
+ },
+ });
+ }
+
+ return baseColumns;
+ }, [isGroupEnabled, nodeGroupsData, t, getServerName, getServerAddress, getProtocolPort]);
+
return (
-
+
action={ref}
actions={{
render: (row) => [
{
@@ -47,6 +168,7 @@ export default function Nodes() {
const body: API.UpdateNodeRequest = {
...row,
...values,
+ node_group_ids: values.node_group_ids?.map((id: string | number) => Number(id)) || [],
} as any;
await updateNode(body);
toast.success(t("updated", "Updated"));
@@ -135,62 +257,7 @@ export default function Nodes() {
];
},
}}
- columns={[
- {
- id: "enabled",
- header: t("enabled", "Enabled"),
- cell: ({ row }) => (
- {
- await toggleNodeStatus({ id: row.original.id, enable: v });
- toast.success(
- v ? t("enabled_on", "Enabled") : t("enabled_off", "Disabled")
- );
- ref.current?.refresh();
- fetchNodes();
- fetchTags();
- }}
- />
- ),
- },
- { accessorKey: "name", header: t("name", "Name") },
-
- {
- id: "address_port",
- header: `${t("address", "Address")}:${t("port", "Port")}`,
- cell: ({ row }) =>
- `${row.original.address || "—"}:${row.original.port || "—"}`,
- },
-
- {
- id: "server_id",
- header: t("server", "Server"),
- cell: ({ row }) =>
- `${getServerName(row.original.server_id)}:${getServerAddress(row.original.server_id)}`,
- },
- {
- id: "protocol",
- header: ` ${t("protocol", "Protocol")}:${t("port", "Port")}`,
- cell: ({ row }) =>
- `${row.original.protocol}:${getProtocolPort(row.original.server_id, row.original.protocol)}`,
- },
- {
- accessorKey: "tags",
- header: t("tags", "Tags"),
- cell: ({ row }) => (
-
- {(row.original.tags || []).length === 0
- ? "—"
- : row.original.tags.map((tg) => (
-
- {tg}
-
- ))}
-
- ),
- },
- ]}
+ columns={columns}
header={{
title: t("pageTitle", "Nodes"),
toolbar: (
@@ -199,15 +266,18 @@ export default function Nodes() {
onSubmit={async (values) => {
setLoading(true);
try {
- const body: API.CreateNodeRequest = {
+ const body: any = {
name: values.name,
server_id: Number(values.server_id!),
protocol: values.protocol,
address: values.address,
port: Number(values.port!),
tags: values.tags || [],
- enabled: false,
};
+ // Add node_group_ids if it exists
+ if (values.node_group_ids) {
+ body.node_group_ids = values.node_group_ids.map((id: string | number) => Number(id));
+ }
await createNode(body);
toast.success(t("created", "Created"));
ref.current?.refresh();
@@ -259,13 +329,35 @@ export default function Nodes() {
}
return updatedItems;
}}
- params={[{ key: "search" }]}
+ params={[
+ {
+ key: "search",
+ },
+ ...(isGroupEnabled
+ ? [
+ {
+ key: "node_group_id",
+ placeholder: t("nodeGroups", "Node Groups"),
+ options: [
+ { label: t("all", "All"), value: "" },
+ ...(nodeGroupsData?.map((item) => ({
+ label: item.name,
+ value: String(item.id),
+ })) || []),
+ ],
+ },
+ ]
+ : []),
+ ]}
request={async (pagination, filter) => {
- const { data } = await filterNodeList({
+ const filters = {
page: pagination.page,
size: pagination.size,
search: filter?.search || undefined,
- });
+ node_group_id: filter?.node_group_id ? Number(filter.node_group_id) : undefined,
+ };
+
+ const { data } = await filterNodeList(filters);
const list = (data?.data?.list || []) as API.Node[];
const total = Number(data?.data?.total || list.length);
return { list, total };
diff --git a/apps/admin/src/sections/nodes/node-form.tsx b/apps/admin/src/sections/nodes/node-form.tsx
index fba748c..a9afd0e 100644
--- a/apps/admin/src/sections/nodes/node-form.tsx
+++ b/apps/admin/src/sections/nodes/node-form.tsx
@@ -2,6 +2,7 @@
import { zodResolver } from "@hookform/resolvers/zod";
import { Button } from "@workspace/ui/components/button";
+import { Checkbox } from "@workspace/ui/components/checkbox";
import {
Form,
FormControl,
@@ -11,6 +12,7 @@ import {
FormLabel,
FormMessage,
} from "@workspace/ui/components/form";
+import { Label } from "@workspace/ui/components/label";
import { ScrollArea } from "@workspace/ui/components/scroll-area";
import {
Sheet,
@@ -23,6 +25,8 @@ import {
import { Combobox } from "@workspace/ui/composed/combobox";
import { EnhancedInput } from "@workspace/ui/composed/enhanced-input";
import TagInput from "@workspace/ui/composed/tag-input";
+import { getGroupConfig, getNodeGroupList } from "@workspace/ui/services/admin/group";
+import { useQuery } from "@tanstack/react-query";
import type { TFunction } from "i18next";
import { useEffect, useMemo, useState } from "react";
import { useForm } from "react-hook-form";
@@ -54,7 +58,7 @@ const buildSchema = (t: TFunction) =>
server_id: z
.number({ message: t("errors.serverRequired", "Please select a server") })
.int()
- .gt(0, t("errors.serverRequired", "Please select a server"))
+ .positive(t("errors.serverRequired", "Please select a server"))
.optional(),
protocol: z
.string()
@@ -71,6 +75,7 @@ const buildSchema = (t: TFunction) =>
.min(1, t("errors.portRange", "Port must be between 1 and 65535"))
.max(65_535, t("errors.portRange", "Port must be between 1 and 65535")),
tags: z.array(z.string()),
+ node_group_ids: z.optional(z.array(z.string()).default([])),
});
export type NodeFormValues = z.infer>;
@@ -112,8 +117,10 @@ export default function NodeForm(props: {
address: "",
port: 0,
tags: [],
+ node_group_ids: [],
...initialValues,
},
+ mode: "onSubmit", // Only validate on form submission
});
const serverId = form.watch("server_id");
@@ -125,17 +132,54 @@ export default function NodeForm(props: {
const availableProtocols = getAvailableProtocols(serverId);
+ // Fetch node groups
+ const { data: nodeGroupsData } = useQuery({
+ queryKey: ["nodeGroups"],
+ queryFn: async () => {
+ const { data } = await getNodeGroupList({ page: 1, size: 1000 });
+ return data.data?.list || [];
+ },
+ });
+
+ // Fetch group config to check if group feature is enabled
+ const { data: groupConfigData } = useQuery({
+ queryKey: ["groupConfig"],
+ queryFn: async () => {
+ const { data } = await getGroupConfig();
+ return data.data;
+ },
+ });
+
+ const isGroupEnabled = groupConfigData?.enabled || false;
+
useEffect(() => {
if (initialValues) {
- form.reset({
+ const resetValues: NodeFormValues = {
name: "",
server_id: undefined,
protocol: "",
address: "",
port: 0,
tags: [],
- ...initialValues,
- });
+ node_group_ids: [],
+ };
+
+ // Copy only the values we need from initialValues
+ if (initialValues.name) resetValues.name = initialValues.name;
+ if (initialValues.server_id) resetValues.server_id = initialValues.server_id;
+ if (initialValues.protocol) resetValues.protocol = initialValues.protocol;
+ if (initialValues.address) resetValues.address = initialValues.address;
+ if (initialValues.port) resetValues.port = initialValues.port;
+ if (initialValues.tags) resetValues.tags = initialValues.tags;
+
+ // Convert node_group_ids from number[] to string[], ensure it's always an array
+ if (initialValues.node_group_ids && Array.isArray(initialValues.node_group_ids)) {
+ resetValues.node_group_ids = initialValues.node_group_ids.map((id: string | number) => String(id));
+ } else {
+ resetValues.node_group_ids = [];
+ }
+
+ form.reset(resetValues);
}
// eslint-disable-next-line react-hooks/exhaustive-deps
}, [initialValues]);
@@ -360,6 +404,7 @@ export default function NodeForm(props: {
)}
/>
+ {/* Tags field - always shown */}
- {t(
- "tags_description",
- "Permission grouping tag (incl. plan binding and delivery policies)."
- )}
+ {isGroupEnabled
+ ? t(
+ "tags_groupMode_description",
+ "Optional tags for display and filtering (node group will be used as tag if empty)."
+ )
+ : t(
+ "tags_description",
+ "Permission grouping tag (incl. plan binding and delivery policies)."
+ )}
)}
/>
+ {/* Show Node Group field only when group feature is enabled */}
+ {isGroupEnabled && (
+ (
+
+ {t("nodeGroup", "Node Group")}
+
+
+ {nodeGroupsData?.map((g) => (
+
+ {
+ // Ensure field.value is always an array
+ const currentValue = Array.isArray(field.value) ? field.value : [];
+ if (checked) {
+ const newValue = [...currentValue, String(g.id)];
+ form.setValue(field.name, newValue, {
+ shouldValidate: true,
+ shouldDirty: true,
+ });
+ } else {
+ const newValue = currentValue.filter((v: string) => v !== String(g.id));
+ form.setValue(field.name, newValue, {
+ shouldValidate: true,
+ shouldDirty: true,
+ });
+ }
+ }}
+ />
+
+ {g.name}
+
+
+ ))}
+
+
+
+ {t(
+ "nodeGroup_description",
+ "Assign this node to multiple groups for user access control."
+ )}
+
+
+
+ )}
+ />
+ )}
diff --git a/apps/admin/src/sections/product/migrate-users-dialog.tsx b/apps/admin/src/sections/product/migrate-users-dialog.tsx
new file mode 100644
index 0000000..6633c97
--- /dev/null
+++ b/apps/admin/src/sections/product/migrate-users-dialog.tsx
@@ -0,0 +1,215 @@
+"use client";
+
+import { Button } from "@workspace/ui/components/button";
+import {
+ Dialog,
+ DialogContent,
+ DialogDescription,
+ DialogFooter,
+ DialogHeader,
+ DialogTitle,
+ DialogTrigger,
+} from "@workspace/ui/components/dialog";
+import {
+ Select,
+ SelectContent,
+ SelectItem,
+ SelectTrigger,
+ SelectValue,
+} from "@workspace/ui/components/select";
+import { Label } from "@workspace/ui/components/label";
+import { useQuery } from "@tanstack/react-query";
+import { useState } from "react";
+import { useTranslation } from "react-i18next";
+import { toast } from "sonner";
+import { Badge } from "@workspace/ui/components/badge";
+import { AlertTriangle } from "lucide-react";
+import {
+ getSubscribeMapping,
+ getUserGroupList,
+ migrateUsersToGroup,
+} from "@workspace/ui/services/admin/group";
+
+interface MigrateUsersDialogProps {
+ subscribeId: number;
+ subscribeName: string;
+}
+
+export default function MigrateUsersDialog({
+ subscribeId,
+ subscribeName,
+}: MigrateUsersDialogProps) {
+ const { t } = useTranslation("product");
+ const [open, setOpen] = useState(false);
+ const [selectedGroupId, setSelectedGroupId] = useState(0);
+ const [migrating, setMigrating] = useState(false);
+
+ // Fetch subscribe mapping to get current user group
+ const { data: mappingsData, isLoading: mappingsLoading } = useQuery({
+ enabled: open,
+ queryKey: ["subscribeMapping"],
+ queryFn: async () => {
+ const { data } = await getSubscribeMapping({ page: 1, size: 1000 });
+ return data.data;
+ },
+ });
+
+ // Fetch user groups for the dropdown
+ const { data: userGroupsData } = useQuery({
+ enabled: open,
+ queryKey: ["userGroups"],
+ queryFn: async () => {
+ const { data } = await getUserGroupList({ page: 1, size: 1000 });
+ return data.data?.list || [];
+ },
+ });
+
+ // Find the current mapping for this subscribe
+ const currentMapping = mappingsData?.list?.find(
+ (m) => m.subscribe_id === subscribeId
+ );
+
+ const currentGroupId = currentMapping?.user_group_id;
+ const currentUserGroup = currentGroupId
+ ? userGroupsData?.find((g) => g.id === currentGroupId)
+ : undefined;
+
+ const handleMigrate = async () => {
+ if (!selectedGroupId) {
+ toast.error(t("selectTargetGroupFirst", "Please select a target group first"));
+ return;
+ }
+
+ if (selectedGroupId === currentGroupId) {
+ toast.error(t("cannotMigrateToSameGroup", "Cannot migrate to the same group"));
+ return;
+ }
+
+ setMigrating(true);
+
+ try {
+ await migrateUsersToGroup({
+ from_user_group_id: currentGroupId!,
+ to_user_group_id: selectedGroupId,
+ include_locked: false,
+ });
+
+ toast.success(t("migrateUsersSuccess", "Successfully migrated users to the target group"));
+
+ setOpen(false);
+ } catch (error) {
+ console.error("Failed to migrate users:", error);
+ toast.error(t("migrateUsersFailed", "Failed to migrate users"));
+ } finally {
+ setMigrating(false);
+ }
+ };
+
+ const availableGroups = userGroupsData?.filter(
+ (g) => !currentGroupId || g.id !== currentGroupId
+ );
+
+ return (
+
+
+ {t("migrateUsers", "Migrate Users")}
+
+
+
+ {t("migrateUsersTitle", "Migrate Users")} - {subscribeName}
+
+ {t(
+ "migrateUsersDescription",
+ "Migrate all users from the current user group to another group"
+ )}
+
+
+
+ {mappingsLoading ? (
+
+ {t("loading", "Loading...")}
+
+ ) : (
+
+ {/* Current User Group Info */}
+
+
{t("currentUserGroup", "Current User Group")}:
+
+ {currentUserGroup ? (
+ <>
+ {currentUserGroup.name}
+ >
+ ) : (
+
+ {t("noMapping", "No mapping set")}
+
+ )}
+
+
+
+ {/* Warning Message */}
+ {currentUserGroup && (
+
+
+
+ {t("migrateUsersWarning", "This will migrate users from \"{group}\" to the target group. This action cannot be undone.")
+ .replace("{group}", currentUserGroup.name || "")
+ }
+
+
+ )}
+
+ {/* Target User Group Selection */}
+
+ {t("targetUserGroup", "Target User Group")}:
+ setSelectedGroupId(Number(val))}
+ disabled={!currentGroupId}
+ >
+
+
+
+
+ {availableGroups?.map((group) => (
+
+ {group.name}
+
+ ))}
+
+
+
+
+ {/* Selected Target Group Info */}
+ {selectedGroupId && (
+
+ {t("selectedGroup", "Selected Group")}:
+
+ {availableGroups?.find((g) => g.id === selectedGroupId)?.name}
+
+
+ )}
+
+ )}
+
+
+ setOpen(false)}>
+ {t("cancel", "Cancel")}
+
+
+ {migrating ? t("migrating", "Migrating...") : t("confirm", "Confirm")}
+
+
+
+
+ );
+}
diff --git a/apps/admin/src/sections/product/subscribe-form.tsx b/apps/admin/src/sections/product/subscribe-form.tsx
index fefa2e8..f744c71 100644
--- a/apps/admin/src/sections/product/subscribe-form.tsx
+++ b/apps/admin/src/sections/product/subscribe-form.tsx
@@ -8,6 +8,7 @@ import {
AccordionTrigger,
} from "@workspace/ui/components/accordion";
import { Button } from "@workspace/ui/components/button";
+import { Card } from "@workspace/ui/components/card";
import { Checkbox } from "@workspace/ui/components/checkbox";
import {
Form,
@@ -36,7 +37,7 @@ import {
TabsTrigger,
} from "@workspace/ui/components/tabs";
import { Combobox } from "@workspace/ui/composed/combobox";
-import { ArrayInput } from "@workspace/ui/composed/dynamic-Inputs";
+import { ArrayInput } from "@workspace/ui/composed/dynamic-inputs";
import { JSONEditor } from "@workspace/ui/composed/editor/index";
import { EnhancedInput } from "@workspace/ui/composed/enhanced-input";
import { Icon } from "@workspace/ui/composed/icon";
@@ -44,6 +45,8 @@ import {
evaluateWithPrecision,
unitConversion,
} from "@workspace/ui/utils/unit-conversions";
+import { getGroupConfig, getNodeGroupList } from "@workspace/ui/services/admin/group";
+import { useQuery } from "@tanstack/react-query";
import { CreditCard, Server, Settings } from "lucide-react";
import { assign, shake } from "radash";
import { useCallback, useEffect, useRef, useState } from "react";
@@ -72,6 +75,8 @@ const defaultValues = {
language: "",
node_tags: [],
nodes: [],
+ node_group_id: "",
+ node_group_ids: [],
unit_time: "Month",
deduction_ratio: 0,
purchase_with_discount: false,
@@ -117,6 +122,8 @@ export default function SubscribeForm>({
language: z.string().optional(),
node_tags: z.array(z.string()).optional(),
nodes: z.array(z.number()).optional(),
+ node_group_id: z.string().optional(),
+ node_group_ids: z.optional(z.array(z.string()).default([])),
deduction_ratio: z.number().optional(),
allow_deduction: z.boolean().optional(),
reset_cycle: z.number().optional(),
@@ -234,12 +241,22 @@ export default function SubscribeForm>({
);
useEffect(() => {
- form?.reset(
- assign(
- defaultValues,
- shake(initialValues, (value) => value === null) as Record
- )
+ const processedValues = assign(
+ defaultValues,
+ shake(initialValues, (value) => value === null) as Record
);
+
+ // Convert node_group_id from number to string (including 0)
+ if (initialValues?.node_group_id !== undefined) {
+ processedValues.node_group_id = String(initialValues.node_group_id);
+ }
+
+ // Convert node_group_ids from number[] to string[]
+ if (initialValues?.node_group_ids && Array.isArray(initialValues.node_group_ids)) {
+ processedValues.node_group_ids = (initialValues.node_group_ids as any[]).map((id) => String(id));
+ }
+
+ form?.reset(processedValues);
const discount = form.getValues("discount") || [];
if (discount.length > 0) {
debouncedCalculateDiscount(discount, "discount");
@@ -256,15 +273,56 @@ export default function SubscribeForm>({
);
async function handleSubmit(data: { [x: string]: any }) {
+ // Don't process node_group_id - submit as-is
+
const bool = await onSubmit(data as T);
if (bool) setOpen(false);
}
- const { getAllAvailableTags, getNodesByTag, getNodesWithoutTags } = useNode();
+ const { getAllAvailableTags, getNodesByTag, getNodesWithoutTags, getNodesWithoutGroups, nodes } = useNode();
const tagGroups = getAllAvailableTags();
+ // Fetch node groups
+ const { data: nodeGroupsData } = useQuery({
+ queryKey: ["nodeGroups"],
+ queryFn: async () => {
+ const { data } = await getNodeGroupList({ page: 1, size: 1000 });
+ return data.data?.list || [];
+ },
+ });
+
+ // Fetch group config to check if group feature is enabled
+ const { data: groupConfigData } = useQuery({
+ queryKey: ["groupConfig"],
+ queryFn: async () => {
+ const { data } = await getGroupConfig();
+ return data.data;
+ },
+ });
+
+ const isGroupEnabled = groupConfigData?.enabled || false;
+
const unit_time = form.watch("unit_time");
+ const node_group_id = form.watch("node_group_id");
+ const node_group_ids = form.watch("node_group_ids");
+
+ // Watch node_group_id and automatically include it in node_group_ids
+ useEffect(() => {
+ if (node_group_id) {
+ const currentGroupIds = form.getValues("node_group_ids") || [];
+ if (!currentGroupIds.includes(node_group_id)) {
+ form.setValue("node_group_ids", [...currentGroupIds, node_group_id]);
+ }
+ }
+ }, [node_group_id, form]);
+
+ // If node_group_id is empty or 0, automatically set it to the first item in node_group_ids
+ useEffect(() => {
+ if ((!node_group_id || node_group_id === "0") && node_group_ids && node_group_ids.length > 0) {
+ form.setValue("node_group_id", node_group_ids[0]);
+ }
+ }, [node_group_ids, node_group_id, form]);
return (
@@ -932,80 +990,83 @@ export default function SubscribeForm>({
-
(
-
- {t("form.nodeGroup")}
-
-
- {tagGroups.map((tag) => {
- const value = field.value || [];
- const tagId = tag;
- const nodesWithTag = getNodesByTag(tag);
+ {/* Show node_tags field only when group feature is disabled */}
+ {!isGroupEnabled && (
+ (
+
+ {t("form.nodeGroup")}
+
+
+ {tagGroups.map((tag) => {
+ const value = field.value || [];
+ const tagId = tag;
+ const nodesWithTag = getNodesByTag(tag);
- return (
-
-
-
-
- checked
- ? form.setValue(field.name, [
- ...value,
- tagId,
- ] as any)
- : form.setValue(
- field.name,
- value.filter(
- (v: any) => v !== tagId
+ return (
+
+
+
+
+ checked
+ ? form.setValue(field.name, [
+ ...value,
+ tagId,
+ ] as any)
+ : form.setValue(
+ field.name,
+ value.filter(
+ (v: any) => v !== tagId
+ )
)
- )
- }
- />
-
- {tag}
-
- ({nodesWithTag.length})
-
-
-
-
-
-
- {getNodesByTag(tag).map((node) => (
-
-
- {node.name}
+ }
+ />
+
+ {tag}
+
+ ({nodesWithTag.length})
-
- {node.address}:{node.port}
-
-
- {node.protocol}
-
-
- ))}
-
-
-
- );
- })}
-
-
-
-
- )}
- />
+
+
+
+
+
+ {getNodesByTag(tag).map((node) => (
+
+
+ {node.name}
+
+
+ {node.address}:{node.port}
+
+
+ {node.protocol}
+
+
+ ))}
+
+
+
+ );
+ })}
+
+
+
+
+ )}
+ />
+ )}
>({
{t("form.node")}
- {getNodesWithoutTags().map((item) => {
+ {/* When group feature is enabled, show nodes without groups */}
+ {/* When group feature is disabled, show nodes without tags */}
+ {(isGroupEnabled ? getNodesWithoutGroups() : getNodesWithoutTags()).map((item: API.Node) => {
const value = field.value || [];
return (
@@ -1056,10 +1119,281 @@ export default function SubscribeForm>({
})}
+
+ {isGroupEnabled
+ ? t("form.nodesWithoutGroupsDescription", "Nodes without group assignment will be shown here (nodes that belong to groups are managed in the Node Groups section above)")
+ : t("form.nodesDescription", "Select nodes for this subscription")
+ }
+
)}
/>
+
+ {/* Show node_group_ids field only when group feature is enabled */}
+ {isGroupEnabled && (
+ <>
+ {/* When no default node group is set, show simple node group selection */}
+ {!node_group_id ? (
+ (
+
+ {t("form.nodeGroups", "Node Groups")}
+
+
+ {nodeGroupsData?.map((g) => {
+ // Filter nodes that belong to this group
+ const nodesInGroup = (nodes || []).filter((node) => {
+ const nodeGroupIds = (node as any).node_group_ids || [];
+ return nodeGroupIds.includes(g.id);
+ });
+
+ return (
+
+
+ {
+ const currentValue = field.value || [];
+ const currentDefaultGroupId = form.getValues("node_group_id");
+
+ if (checked) {
+ const newValue = [...currentValue, String(g.id)];
+ form.setValue(field.name, newValue);
+
+ // If no default node group is set, set this one as default
+ if (!currentDefaultGroupId) {
+ form.setValue("node_group_id", String(g.id));
+ }
+ } else {
+ form.setValue(
+ field.name,
+ currentValue.filter((v: string) => v !== String(g.id))
+ );
+ }
+ }}
+ />
+
+ {g.name}
+
+ ({nodesInGroup.length} {t("form.nodes", "nodes")})
+
+
+
+
+ {/* Show nodes in this group */}
+ {nodesInGroup.length > 0 && (
+
+
+ {t("form.nodesInGroup", "Nodes in this group:")}
+
+
+ {nodesInGroup.map((node) => (
+
+
+ {node.name}
+
+
+ {node.address}:{node.port}
+
+
+ {node.protocol}
+
+
+ ))}
+
+
+ )}
+
+ );
+ })}
+
+
+
+ {t(
+ "form.nodeGroupsFirstSelectionDescription",
+ "Select node groups for this product. The first selected group will be set as the default node group."
+ )}
+
+
+
+ )}
+ />
+ ) : (
+ <>
+ {/* Default Node Group Selection - shown when default is set */}
+ {
+ // Find the selected node group
+ const selectedNodeGroup = nodeGroupsData?.find((g) => String(g.id) === field.value);
+ // Filter nodes that belong to this group
+ const nodesInGroup = selectedNodeGroup ? (nodes || []).filter((node) => {
+ const nodeGroupIds = (node as any).node_group_ids || [];
+ return nodeGroupIds.includes(selectedNodeGroup.id);
+ }) : [];
+
+ return (
+
+ {t("form.defaultNodeGroup", "Default Node Group")}
+
+
+ {
+ form.setValue(field.name, value || "");
+ }}
+ options={[
+ { label: t("form.noDefaultNodeGroup", "No Default Node Group"), value: "" },
+ ...(nodeGroupsData?.map((g) => ({
+ label: g.name,
+ value: String(g.id),
+ })) || []),
+ ]}
+ />
+
+
+ {t(
+ "form.defaultNodeGroupDescription",
+ "The default node group for this product."
+ )}
+
+ {/* Show nodes in the selected default node group */}
+ {nodesInGroup.length > 0 && (
+ <>
+
+ {t("form.nodesInGroup", "Nodes in this group:")}
+
+
+ {nodesInGroup.map((node) => (
+
+
+ {node.name}
+
+
+ {node.address}:{node.port}
+
+
+ {node.protocol}
+
+
+ ))}
+
+ >
+ )}
+
+
+
+ );
+ }}
+ />
+
+ {/* Backup Node Groups Selection - filter out default node group */}
+ (
+
+ {t("form.backupNodeGroups", "Backup Node Groups")}
+
+
+ {nodeGroupsData
+ ?.filter((g) => String(g.id) !== node_group_id)
+ ?.map((g) => {
+ // Filter nodes that belong to this group
+ const nodesInGroup = (nodes || []).filter((node) => {
+ const nodeGroupIds = (node as any).node_group_ids || [];
+ return nodeGroupIds.includes(g.id);
+ });
+
+ return (
+
+
+ {
+ const currentValue = field.value || [];
+ if (checked) {
+ form.setValue(field.name, [...currentValue, String(g.id)]);
+ } else {
+ form.setValue(
+ field.name,
+ currentValue.filter((v: string) => v !== String(g.id))
+ );
+ }
+ }}
+ />
+
+ {g.name}
+
+ ({nodesInGroup.length} {t("form.nodes", "nodes")})
+
+
+
+
+ {/* Show nodes in this group */}
+ {nodesInGroup.length > 0 && (
+
+
+ {t("form.nodesInGroup", "Nodes in this group:")}
+
+
+ {nodesInGroup.map((node) => (
+
+
+ {node.name}
+
+
+ {node.address}:{node.port}
+
+
+ {node.protocol}
+
+
+ ))}
+
+
+ )}
+
+ );
+ })}
+
+
+
+ {t(
+ "form.backupNodeGroupsDescription",
+ "Select additional backup node groups."
+ )}
+
+
+
+ )}
+ />
+ >
+ )}
+ >
+ )}
diff --git a/apps/admin/src/sections/product/subscribe-table.tsx b/apps/admin/src/sections/product/subscribe-table.tsx
index de846e8..a3f1a96 100644
--- a/apps/admin/src/sections/product/subscribe-table.tsx
+++ b/apps/admin/src/sections/product/subscribe-table.tsx
@@ -16,6 +16,8 @@ import {
subscribeSort,
updateSubscribe,
} from "@workspace/ui/services/admin/subscribe";
+import { getNodeGroupList } from "@workspace/ui/services/admin/group";
+import { useQuery } from "@tanstack/react-query";
import { useRef, useState } from "react";
import { useTranslation } from "react-i18next";
import { toast } from "sonner";
@@ -28,8 +30,29 @@ export default function SubscribeTable() {
const [loading, setLoading] = useState(false);
const ref = useRef(null);
const { fetchSubscribes } = useSubscribe();
+
+ // Fetch node groups for filtering
+ const { data: nodeGroupsData } = useQuery({
+ queryKey: ["nodeGroups"],
+ queryFn: async () => {
+ const { data } = await getNodeGroupList({ page: 1, size: 1000 });
+ return data.data?.list || [];
+ },
+ });
+
+ // Fetch group config to check if group feature is enabled
+ const { data: groupConfigData } = useQuery({
+ queryKey: ["groupConfig"],
+ queryFn: async () => {
+ const { data } = await (await import("@workspace/ui/services/admin/group")).getGroupConfig();
+ return data.data;
+ },
+ });
+
+ const isGroupEnabled = groupConfigData?.enabled || false;
+
return (
-
+
action={ref}
actions={{
render: (row) => [
@@ -40,10 +63,16 @@ export default function SubscribeTable() {
onSubmit={async (values) => {
setLoading(true);
try {
- await updateSubscribe({
+ const updateBody: any = {
...row,
...values,
- } as API.UpdateSubscribeRequest);
+ };
+ // Add node_group_ids if it exists in values
+ const vals = values as any;
+ if (vals.node_group_ids) {
+ updateBody.node_group_ids = vals.node_group_ids.map((id: string | number) => Number(id));
+ }
+ await updateSubscribe(updateBody as API.UpdateSubscribeRequest);
toast.success(t("updateSuccess"));
ref.current?.refresh();
fetchSubscribes();
@@ -243,6 +272,26 @@ export default function SubscribeTable() {
{row.getValue("sold")}
),
},
+ ...(isGroupEnabled
+ ? [
+ {
+ id: "node_group",
+ header: t("defaultNodeGroup", "Default Node Group"),
+ cell: ({ row }: { row: any }) => {
+ const nodeGroupId = row.original.node_group_id;
+ const nodeGroup = nodeGroupsData?.find((g) => g.id === nodeGroupId);
+
+ return (
+
+ {nodeGroup ? (
+ {nodeGroup.name}
+ ) : null}
+
+ );
+ },
+ },
+ ]
+ : []),
]}
header={{
toolbar: (
@@ -251,11 +300,17 @@ export default function SubscribeTable() {
onSubmit={async (values) => {
setLoading(true);
try {
- await createSubscribe({
+ const createBody: any = {
...values,
show: false,
sell: false,
- });
+ };
+ // Add node_group_ids if it exists in values
+ const vals = values as any;
+ if (vals.node_group_ids) {
+ createBody.node_group_ids = vals.node_group_ids.map((id: string | number) => Number(id));
+ }
+ await createSubscribe(createBody);
toast.success(t("createSuccess"));
ref.current?.refresh();
fetchSubscribes();
@@ -312,12 +367,30 @@ export default function SubscribeTable() {
{
key: "search",
},
+ ...(isGroupEnabled
+ ? [
+ {
+ key: "node_group_id",
+ placeholder: t("nodeGroups", "Node Groups"),
+ options: [
+ { label: t("all", "All"), value: "" },
+ ...(nodeGroupsData?.map((item) => ({
+ label: item.name,
+ value: String(item.id),
+ })) || []),
+ ],
+ },
+ ]
+ : []),
]}
request={async (pagination, filters) => {
- const { data } = await getSubscribeList({
+ const params = {
...pagination,
...filters,
- });
+ node_group_id: filters?.node_group_id ? Number(filters.node_group_id) : undefined,
+ } as any;
+
+ const { data } = await getSubscribeList(params);
return {
list: data.data?.list || [],
total: data.data?.total || 0,
diff --git a/apps/admin/src/sections/user/edit-user-group-dialog.tsx b/apps/admin/src/sections/user/edit-user-group-dialog.tsx
new file mode 100644
index 0000000..c1722f5
--- /dev/null
+++ b/apps/admin/src/sections/user/edit-user-group-dialog.tsx
@@ -0,0 +1,199 @@
+"use client";
+
+import { Button } from "@workspace/ui/components/button";
+import {
+ Dialog,
+ DialogContent,
+ DialogDescription,
+ DialogFooter,
+ DialogHeader,
+ DialogTitle,
+ DialogTrigger,
+} from "@workspace/ui/components/dialog";
+import {
+ Form,
+ FormControl,
+ FormField,
+ FormItem,
+ FormLabel,
+ FormMessage,
+} from "@workspace/ui/components/form";
+import {
+ Select,
+ SelectContent,
+ SelectItem,
+ SelectTrigger,
+ SelectValue,
+} from "@workspace/ui/components/select";
+import {
+ getUserGroupList,
+} from "@workspace/ui/services/admin/group";
+import { zodResolver } from "@hookform/resolvers/zod";
+import { useForm } from "react-hook-form";
+import { useTranslation } from "react-i18next";
+import { z } from "zod";
+import { useQuery } from "@tanstack/react-query";
+import React, { useEffect } from "react";
+
+const editUserGroupSchema = z.object({
+ user_group_id: z.number().min(0),
+ group_locked: z.boolean(),
+});
+
+type EditUserGroupFormValues = z.infer;
+
+interface EditUserGroupDialogProps {
+ userId: number;
+ userSubscribeId?: number;
+ currentGroupId?: number | undefined;
+ currentLocked?: boolean | undefined;
+ currentGroupIds?: number[] | null | undefined;
+ trigger: React.ReactNode;
+ onSubmit?: (values: EditUserGroupFormValues) => Promise;
+}
+
+export default function EditUserGroupDialog({
+ userId: _userId,
+ userSubscribeId: _userSubscribeId,
+ currentGroupId,
+ currentLocked,
+ currentGroupIds,
+ trigger,
+ onSubmit,
+}: EditUserGroupDialogProps) {
+ const { t } = useTranslation("user");
+ const [open, setOpen] = React.useState(false);
+
+ // Fetch user groups list
+ const { data: groupsData } = useQuery({
+ enabled: open,
+ queryKey: ["getUserGroupList"],
+ queryFn: async () => {
+ const { data } = await getUserGroupList({
+ page: 1,
+ size: 100,
+ });
+ return data.data?.list || [];
+ },
+ });
+
+ const form = useForm({
+ resolver: zodResolver(editUserGroupSchema),
+ defaultValues: {
+ user_group_id: 0,
+ group_locked: false,
+ },
+ });
+
+ // Reset form when dialog closes
+ useEffect(() => {
+ if (!open) {
+ form.reset();
+ }
+ }, [open, form]);
+
+ // Set form values when dialog opens
+ useEffect(() => {
+ if (open) {
+ // Support both usage scenarios:
+ // 1. User list page: currentGroupId (single number)
+ // 2. Subscribe detail page: currentGroupIds (array)
+ const groupId = currentGroupId || (currentGroupIds?.[0]) || 0;
+
+ form.reset({
+ user_group_id: groupId,
+ group_locked: currentLocked || false,
+ });
+ }
+ }, [open, currentGroupId, currentGroupIds, currentLocked, form]);
+
+ const handleSubmit = async (values: EditUserGroupFormValues) => {
+ if (onSubmit) {
+ const success = await onSubmit(values);
+ if (success) {
+ setOpen(false);
+ }
+ }
+ };
+
+ return (
+
+ {trigger}
+
+
+ {t("editUserGroup", "Edit User Group")}
+
+ {t(
+ "editUserGroupDescription",
+ "Edit user group assignment and lock status"
+ )}
+
+
+
+
+
+ (
+
+ {t("userGroup", "User Group")}
+ 0 ? String(field.value) : undefined}
+ onValueChange={(value) => field.onChange(parseInt(value))}
+ >
+
+
+
+
+
+
+ {groupsData?.map((group: API.UserGroup) => (
+
+ {group.name}
+
+ ))}
+
+
+
+
+ )}
+ />
+
+ (
+
+
+
{t("lockGroup", "Lock Group")}
+
+ {t(
+ "lockGroupDescription",
+ "Prevent automatic grouping from changing this user's group"
+ )}
+
+
+
+ field.onChange(e.target.checked)}
+ className="h-4 w-4"
+ />
+
+
+ )}
+ />
+
+
+
+ {t("save", "Save")}
+
+
+
+
+
+
+ );
+}
diff --git a/apps/admin/src/sections/user/index.tsx b/apps/admin/src/sections/user/index.tsx
index 0b50a80..a8343f6 100644
--- a/apps/admin/src/sections/user/index.tsx
+++ b/apps/admin/src/sections/user/index.tsx
@@ -2,6 +2,13 @@ import { useQuery } from "@tanstack/react-query";
import { Link, useSearch } from "@tanstack/react-router";
import { Badge } from "@workspace/ui/components/badge";
import { Button } from "@workspace/ui/components/button";
+import {
+ Dialog,
+ DialogContent,
+ DialogHeader,
+ DialogTitle,
+ DialogTrigger,
+} from "@workspace/ui/components/dialog";
import {
DropdownMenu,
DropdownMenuContent,
@@ -35,6 +42,10 @@ import {
getUserList,
updateUserBasicInfo,
} from "@workspace/ui/services/admin/user";
+import {
+ // getUserGroupList,
+ previewUserNodes,
+} from "@workspace/ui/services/admin/group";
import { useRef, useState } from "react";
import { useTranslation } from "react-i18next";
import { toast } from "sonner";
@@ -47,6 +58,7 @@ import { AuthMethodsForm } from "./user-profile/auth-methods-form";
import { BasicInfoForm } from "./user-profile/basic-info-form";
import { NotifySettingsForm } from "./user-profile/notify-settings-form";
import UserSubscription from "./user-subscription";
+// import EditUserGroupDialog from "./edit-user-group-dialog";
export default function User() {
const { t } = useTranslation("user");
@@ -56,12 +68,21 @@ export default function User() {
const { subscribes } = useSubscribe();
+ // const { data: userGroupsData } = useQuery({
+ // queryKey: ["userGroups"],
+ // queryFn: async () => {
+ // const { data } = await getUserGroupList({ page: 1, size: 1000 });
+ // return data.data?.list || [];
+ // },
+ // });
+
const initialFilters = {
search: sp.search || undefined,
user_id: sp.user_id || undefined,
subscribe_id: sp.subscribe_id || undefined,
user_subscribe_id: sp.user_subscribe_id || undefined,
short_code: sp.short_code || undefined,
+ // user_group_id: sp.user_group_id || undefined,
};
return (
@@ -75,6 +96,7 @@ export default function User() {
userId={row.id}
/>,
,
+ ,
(
@@ -174,10 +197,12 @@ export default function User() {
),
},
{
+ id: "id",
accessorKey: "id",
header: "ID",
},
{
+ id: "deleted_at",
accessorKey: "deleted_at",
header: t("isDeleted", "Deleted"),
cell: ({ row }) => {
@@ -190,6 +215,7 @@ export default function User() {
},
},
{
+ id: "auth_methods",
accessorKey: "auth_methods",
header: t("userName", "Username"),
cell: ({ row }) => {
@@ -208,6 +234,7 @@ export default function User() {
},
},
{
+ id: "balance",
accessorKey: "balance",
header: t("balance", "Balance"),
cell: ({ row }) => (
@@ -215,6 +242,7 @@ export default function User() {
),
},
{
+ id: "gift_amount",
accessorKey: "gift_amount",
header: t("giftAmount", "Gift Amount"),
cell: ({ row }) => (
@@ -222,6 +250,7 @@ export default function User() {
),
},
{
+ id: "commission",
accessorKey: "commission",
header: t("commission", "Commission"),
cell: ({ row }) => (
@@ -229,16 +258,19 @@ export default function User() {
),
},
{
+ id: "refer_code",
accessorKey: "refer_code",
header: t("inviteCode", "Invite Code"),
cell: ({ row }) => row.getValue("refer_code") || "--",
},
{
+ id: "referer_id",
accessorKey: "referer_id",
header: t("referer", "Referer"),
cell: ({ row }) => ,
},
{
+ id: "created_at",
accessorKey: "created_at",
header: t("createdAt", "Created At"),
cell: ({ row }) => formatDate(row.getValue("created_at")),
@@ -276,10 +308,13 @@ export default function User() {
{
key: "subscribe_id",
placeholder: t("subscription", "Subscription"),
- options: subscribes?.map((item) => ({
- label: item.name!,
- value: String(item.id!),
- })),
+ options: [
+ { label: t("all", "All"), value: "" },
+ ...(subscribes?.map((item) => ({
+ label: item.name!,
+ value: String(item.id!),
+ })) || []),
+ ],
},
{
key: "search",
@@ -401,3 +436,80 @@ function SubscriptionSheet({ userId }: { userId: number }) {
);
}
+
+function PreviewNodesDialog({ userId }: { userId: number }) {
+ const { t } = useTranslation("user");
+ const [open, setOpen] = useState(false);
+ const { data: previewData, isLoading } = useQuery({
+ enabled: open,
+ queryKey: ["previewUserNodes", userId],
+ queryFn: async () => {
+ const { data } = await previewUserNodes({ user_id: userId });
+ return data.data;
+ },
+ });
+
+ return (
+
+
+ {t("previewNodes", "Preview Nodes")}
+
+
+
+
+ {t("previewNodes", "Preview Nodes")} · ID: {userId}
+
+
+ {isLoading ? (
+
+ {t("loading", "Loading...")}
+
+ ) : previewData ? (
+
+
+
+ {t("availableNodes", "Available Nodes")}:
+ {" "}
+ {previewData.node_groups?.reduce((sum, group) => sum + (group.nodes?.length || 0), 0) || 0}
+
+ {previewData.node_groups && previewData.node_groups.length > 0 ? (
+
+ {previewData.node_groups.map((group) => (
+
+
+ {group.name || (group.id === 0 ? t("publicNodes", "Public Nodes") : `${t("nodeGroup", "Node Group")} ${group.id}`)}
+
+ {group.nodes && group.nodes.length > 0 ? (
+
+
+
+ ID
+ {t("name", "Name")}
+ {t("address", "Address")}
+
+
+
+ {group.nodes.map((node) => (
+
+ {node.id}
+ {node.name}
+ {node.address}:{node.port}
+
+ ))}
+
+
+ ) : null}
+
+ ))}
+
+ ) : (
+
+ {t("noNodesAvailable", "No nodes available")}
+
+ )}
+
+ ) : null}
+
+
+ );
+}
diff --git a/apps/admin/src/sections/user/user-detail.tsx b/apps/admin/src/sections/user/user-detail.tsx
index f830fc2..6e4b023 100644
--- a/apps/admin/src/sections/user/user-detail.tsx
+++ b/apps/admin/src/sections/user/user-detail.tsx
@@ -16,6 +16,8 @@ import { formatBytes } from "@workspace/ui/utils/formatting";
import { useTranslation } from "react-i18next";
import { Display } from "@/components/display";
import { formatDate } from "@/utils/common";
+// import EditUserGroupDialog from "./edit-user-group-dialog";
+// import { getUserGroupList } from "@workspace/ui/services/admin/group";
export function UserSubscribeDetail({
id,
@@ -37,10 +39,33 @@ export function UserSubscribeDetail({
},
});
+ // Fetch user groups for display
+ // const { data: groupsData } = useQuery({
+ // enabled: id !== 0 && enabled,
+ // queryKey: ["getUserGroupList"],
+ // queryFn: async () => {
+ // const { data } = await getUserGroupList({
+ // page: 1,
+ // size: 100,
+ // });
+ // return data.data?.list || [];
+ // },
+ // });
+
if (!id) return "--";
const usedTraffic = data ? data.upload + data.download : 0;
const totalTraffic = data?.traffic || 0;
+ const remainingTraffic = totalTraffic > 0 ? totalTraffic - usedTraffic : 0;
+
+ // Get user group info from data.user
+ // const userGroupId = typeof data?.user?.user_group_id === 'number' ? data?.user?.user_group_id : 0;
+ // const groupLocked = data?.user?.group_locked || false;
+ // const groupIds = userGroupId > 0 ? [userGroupId] : [];
+
+ // const groupNames = userGroupId > 0
+ // ? groupsData?.find((g: API.UserGroup) => g.id === userGroupId)?.name || "--"
+ // : "--";
const subscribeContent = (
@@ -76,6 +101,16 @@ export function UserSubscribeDetail({
: "--"}
+
+ {t("remainingTraffic")}
+
+ {data
+ ? totalTraffic === 0
+ ? t("unlimited")
+ : formatBytes(remainingTraffic)
+ : "--"}
+
+
{t("startTime")}
@@ -88,6 +123,35 @@ export function UserSubscribeDetail({
{data?.expire_time ? formatDate(data.expire_time) : "--"}
+ {/*
+ {t("userGroup")}
+
+ {groupNames || "--"}
+ {data?.id && (
+
+ {t("edit", "Edit")}
+
+ }
+ onSubmit={async () => {
+ window.location.reload();
+ return true;
+ }}
+ />
+ )}
+
+
+
+ {t("groupLocked")}
+
+ {groupLocked ? t("yes", "Yes") : t("no", "No")}
+
+ */}
diff --git a/apps/admin/src/sections/user/user-subscription/index.tsx b/apps/admin/src/sections/user/user-subscription/index.tsx
index a132924..b9a2101 100644
--- a/apps/admin/src/sections/user/user-subscription/index.tsx
+++ b/apps/admin/src/sections/user/user-subscription/index.tsx
@@ -143,6 +143,19 @@ export default function UserSubscription({ userId }: { userId: number }) {
),
},
+ {
+ id: "remaining_traffic",
+ header: t("remainingTraffic", "Remaining Traffic"),
+ cell: ({ row }) => {
+ const upload = row.original.upload || 0;
+ const download = row.original.download || 0;
+ const totalTraffic = row.original.traffic || 0;
+ const remainingTraffic = totalTraffic > 0 ? totalTraffic - upload - download : 0;
+ return (
+
+ );
+ },
+ },
{
accessorKey: "speed_limit",
header: t("speedLimit", "Speed Limit"),
@@ -390,7 +403,7 @@ function RowMoreActions({
"This action cannot be undone."
)}
onConfirm={async () => {
- await deleteUserSubscribe({ user_subscribe_id: row.id });
+ await deleteUserSubscribe({ user_subscribe_id: String(row.id) });
toast.success(t("deleteSuccess", "Deleted successfully"));
refresh();
}}
diff --git a/apps/admin/src/stores/node.ts b/apps/admin/src/stores/node.ts
index 8c446cf..6dd629c 100644
--- a/apps/admin/src/stores/node.ts
+++ b/apps/admin/src/stores/node.ts
@@ -25,6 +25,7 @@ interface NodeState {
isServerReferencedByNodes: (serverId: number) => boolean;
getNodesByTag: (tag: string) => API.Node[];
getNodesWithoutTags: () => API.Node[];
+ getNodesWithoutGroups: () => API.Node[];
getNodeTags: () => string[];
getAllAvailableTags: () => string[];
}
@@ -92,6 +93,12 @@ export const useNodeStore = create((set, get) => ({
getNodesWithoutTags: () =>
get().nodes.filter((node) => (node.tags || []).length === 0),
+ getNodesWithoutGroups: () =>
+ get().nodes.filter((node) => {
+ const groupIds = (node as any).node_group_ids;
+ return !groupIds || groupIds.length === 0;
+ }),
+
getNodeTags: () =>
Array.from(
new Set(
@@ -135,6 +142,7 @@ export const useNode = () => {
isServerReferencedByNodes: store.isServerReferencedByNodes,
getNodesByTag: store.getNodesByTag,
getNodesWithoutTags: store.getNodesWithoutTags,
+ getNodesWithoutGroups: store.getNodesWithoutGroups,
getNodeTags: store.getNodeTags,
getAllAvailableTags: store.getAllAvailableTags,
};
diff --git a/apps/admin/src/utils/common.ts b/apps/admin/src/utils/common.ts
index 7b4960c..b18d543 100644
--- a/apps/admin/src/utils/common.ts
+++ b/apps/admin/src/utils/common.ts
@@ -24,8 +24,20 @@ export function differenceInDays(date1: Date, date2: Date): number {
export function formatDate(date?: Date | number, showTime = true) {
if (!date) return;
+
+ // 如果是数字(Unix时间戳),需要判断是秒级还是毫秒级
+ // Unix时间戳(秒级):10位数字,如 1771936457
+ // JavaScript时间戳(毫秒级):13位数字
+ let dateValue = date;
+ if (typeof date === "number") {
+ // 如果小于 10000000000(100亿),认为是秒级时间戳,需要乘以1000
+ if (date < 10000000000) {
+ dateValue = date * 1000;
+ }
+ }
+
const timeZone = localStorage.getItem("timezone") || "UTC";
- return intlFormat(date, {
+ return intlFormat(dateValue, {
year: "numeric",
month: "numeric",
day: "numeric",
diff --git a/packages/ui/src/composed/date-picker.tsx b/packages/ui/src/composed/date-picker.tsx
index 690e8e7..68c9e35 100644
--- a/packages/ui/src/composed/date-picker.tsx
+++ b/packages/ui/src/composed/date-picker.tsx
@@ -53,17 +53,18 @@ export function DatePicker({
)}
variant="outline"
>
- {value ? intlFormat(value) : {placeholder} }
+ {value ? intlFormat(value) : {placeholder} }
{value && (
-
-
+
)}
diff --git a/packages/ui/src/services/admin/group.ts b/packages/ui/src/services/admin/group.ts
new file mode 100644
index 0000000..bc86c26
--- /dev/null
+++ b/packages/ui/src/services/admin/group.ts
@@ -0,0 +1,394 @@
+// @ts-nocheck
+/* eslint-disable */
+import request from "@workspace/ui/lib/request";
+
+/** Get user group list GET /v1/admin/group/user/list */
+export async function getUserGroupList(
+ params: API.GetUserGroupListRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/user/list`,
+ {
+ method: "GET",
+ params: {
+ ...params,
+ },
+ ...(options || {}),
+ }
+ );
+}
+
+/** Create user group POST /v1/admin/group/user */
+export async function createUserGroup(
+ body: API.CreateUserGroupRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/user`,
+ {
+ method: "POST",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ data: body,
+ ...(options || {}),
+ }
+ );
+}
+
+/** Update user group PUT /v1/admin/group/user */
+export async function updateUserGroup(
+ body: API.UpdateUserGroupRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/user`,
+ {
+ method: "PUT",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ data: body,
+ ...(options || {}),
+ }
+ );
+}
+
+/** Delete user group DELETE /v1/admin/group/user */
+export async function deleteUserGroup(
+ body: API.DeleteUserGroupRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/user`,
+ {
+ method: "DELETE",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ data: body,
+ ...(options || {}),
+ }
+ );
+}
+
+/** Get node group list GET /v1/admin/group/node/list */
+export async function getNodeGroupList(
+ params: API.GetNodeGroupListRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/node/list`,
+ {
+ method: "GET",
+ params: {
+ ...params,
+ },
+ ...(options || {}),
+ }
+ );
+}
+
+/** Create node group POST /v1/admin/group/node */
+export async function createNodeGroup(
+ body: API.CreateNodeGroupRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/node`,
+ {
+ method: "POST",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ data: body,
+ ...(options || {}),
+ }
+ );
+}
+
+/** Update node group PUT /v1/admin/group/node */
+export async function updateNodeGroup(
+ body: API.UpdateNodeGroupRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/node`,
+ {
+ method: "PUT",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ data: body,
+ ...(options || {}),
+ }
+ );
+}
+
+/** Delete node group DELETE /v1/admin/group/node */
+export async function deleteNodeGroup(
+ body: API.DeleteNodeGroupRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/node`,
+ {
+ method: "DELETE",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ data: body,
+ ...(options || {}),
+ }
+ );
+}
+
+/** Get subscribe mapping list GET /v1/admin/group/subscribe/mapping */
+export async function getSubscribeMapping(
+ params: API.GetSubscribeMappingRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/subscribe/mapping`,
+ {
+ method: "GET",
+ params: {
+ ...params,
+ },
+ ...(options || {}),
+ }
+ );
+}
+
+/** Update subscribe mapping PUT /v1/admin/group/subscribe/mapping */
+export async function updateSubscribeMapping(
+ body: API.UpdateSubscribeMappingRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/subscribe/mapping`,
+ {
+ method: "PUT",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ data: body,
+ ...(options || {}),
+ }
+ );
+}
+
+/** Get subscribe group mapping GET /v1/admin/group/subscribe/mapping */
+export async function getSubscribeGroupMapping(
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/subscribe/mapping`,
+ {
+ method: "GET",
+ ...(options || {}),
+ }
+ );
+}
+
+/** Get group config GET /v1/admin/group/config */
+export async function getGroupConfig(
+ params?: API.GetGroupConfigRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/config`,
+ {
+ method: "GET",
+ params: params || {},
+ ...(options || {}),
+ }
+ );
+}
+
+/** Update group config PUT /v1/admin/group/config */
+export async function updateGroupConfig(
+ body: API.UpdateGroupConfigRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/config`,
+ {
+ method: "PUT",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ data: body,
+ ...(options || {}),
+ }
+ );
+}
+
+/** Get recalculation status GET /v1/admin/group/recalculation/status */
+export async function getRecalculationStatus(options?: { [key: string]: any }) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/recalculation/status`,
+ {
+ method: "GET",
+ ...(options || {}),
+ }
+ );
+}
+
+/** Recalculate groups POST /v1/admin/group/recalculate */
+export async function recalculateGroup(
+ body: API.RecalculateGroupRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/recalculate`,
+ {
+ method: "POST",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ data: body,
+ ...(options || {}),
+ }
+ );
+}
+
+/** Get group history GET /v1/admin/group/history */
+export async function getGroupHistory(
+ params: API.GetGroupHistoryRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/history`,
+ {
+ method: "GET",
+ params: {
+ ...params,
+ },
+ ...(options || {}),
+ }
+ );
+}
+
+/** Get group history detail GET /v1/admin/group/history/detail */
+export async function getGroupHistoryDetail(
+ params: { id: number },
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/history/detail`,
+ {
+ method: "GET",
+ params: {
+ id: params.id,
+ },
+ ...(options || {}),
+ }
+ );
+}
+
+/** Export group result GET /v1/admin/group/export */
+export async function exportGroupResult(
+ params?: API.ExportGroupResultRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/export`,
+ {
+ method: "GET",
+ params: params || {},
+ responseType: 'blob',
+ ...(options || {}),
+ }
+ );
+}
+
+/** Preview user nodes GET /v1/admin/group/preview */
+export async function previewUserNodes(
+ params: API.PreviewUserNodesRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/preview`,
+ {
+ method: "GET",
+ params: {
+ user_id: params.user_id,
+ },
+ ...(options || {}),
+ }
+ );
+}
+
+/** Migrate users to another group POST /v1/admin/group/migrate */
+export async function migrateUsersToGroup(
+ body: API.MigrateUsersRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/migrate`,
+ {
+ method: "POST",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ data: body,
+ ...(options || {}),
+ }
+ );
+}
+
+/** Update user user group PUT /v1/admin/user/user_group */
+export async function updateUserUserGroup(
+ body: API.UpdateUserUserGroupRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/user/user_group`,
+ {
+ method: "PUT",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ data: body,
+ ...(options || {}),
+ }
+ );
+}
+
+/** Reset all groups POST /v1/admin/group/reset */
+export async function resetGroups(
+ body: API.ResetGroupsRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/reset`,
+ {
+ method: "POST",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ data: body,
+ ...(options || {}),
+ }
+ );
+}
+
+/** Bind node group to user groups POST /v1/admin/group/user/bind-node-groups */
+export async function bindNodeGroups(
+ body: API.BindNodeGroupsRequest,
+ options?: { [key: string]: any }
+) {
+ return request(
+ `${import.meta.env.VITE_API_PREFIX || ""}/v1/admin/group/user/bind-node-groups`,
+ {
+ method: "POST",
+ headers: {
+ "Content-Type": "application/json",
+ },
+ data: body,
+ ...(options || {}),
+ }
+ );
+}
diff --git a/packages/ui/src/services/admin/index.ts b/packages/ui/src/services/admin/index.ts
index bf43e11..feb9e69 100644
--- a/packages/ui/src/services/admin/index.ts
+++ b/packages/ui/src/services/admin/index.ts
@@ -9,6 +9,7 @@ import * as authMethod from "./authMethod";
import * as console from "./console";
import * as coupon from "./coupon";
import * as document from "./document";
+import * as group from "./group";
import * as log from "./log";
import * as marketing from "./marketing";
import * as order from "./order";
@@ -27,6 +28,7 @@ export default {
console,
coupon,
document,
+ group,
log,
marketing,
order,
diff --git a/packages/ui/src/services/admin/typings.d.ts b/packages/ui/src/services/admin/typings.d.ts
index 4414245..d835996 100644
--- a/packages/ui/src/services/admin/typings.d.ts
+++ b/packages/ui/src/services/admin/typings.d.ts
@@ -103,6 +103,7 @@ declare namespace API {
longitude: string;
created_at: number;
download: number;
+ port: number;
};
type AuthConfig = {
@@ -534,7 +535,7 @@ declare namespace API {
};
type DeleteUserSubscribeRequest = {
- user_subscribe_id: number;
+ user_subscribe_id: string;
};
type DeviceAuthticateConfig = {
@@ -702,12 +703,14 @@ declare namespace API {
page: number;
size: number;
search?: string;
+ node_group_id?: number;
};
type FilterNodeListRequest = {
page: number;
size: number;
search?: string;
+ node_group_id?: number;
};
type FilterNodeListResponse = {
@@ -1147,6 +1150,7 @@ declare namespace API {
size: number;
language?: string;
search?: string;
+ node_group_id?: number;
};
type GetSubscribeListRequest = {
@@ -1154,6 +1158,7 @@ declare namespace API {
size: number;
language?: string;
search?: string;
+ node_group_id?: number;
};
type GetSubscribeListResponse = {
@@ -1210,6 +1215,7 @@ declare namespace API {
unscoped?: boolean;
subscribe_id?: number;
user_subscribe_id?: number;
+ user_group_id?: number;
};
type GetUserListRequest = {
@@ -1440,6 +1446,8 @@ declare namespace API {
protocol: string;
enabled: boolean;
sort?: number;
+ node_group_id?: number;
+ node_group_ids?: number[];
created_at: number;
updated_at: number;
};
@@ -1921,11 +1929,11 @@ declare namespace API {
};
type ResetUserSubscribeTokenRequest = {
- user_subscribe_id: number;
+ user_subscribe_id: any;
};
type ResetUserSubscribeTrafficRequest = {
- user_subscribe_id: number;
+ user_subscribe_id: string;
};
type Response = {
@@ -2106,6 +2114,8 @@ declare namespace API {
quota: number;
nodes: number[];
node_tags: string[];
+ node_group_ids?: number[];
+ node_group_id?: number;
show: boolean;
sell: boolean;
sort: number;
@@ -2171,6 +2181,8 @@ declare namespace API {
quota?: number;
nodes?: number[];
node_tags?: string[];
+ node_group_ids?: number[];
+ node_group_id?: number;
show?: boolean;
sell?: boolean;
sort?: number;
@@ -2244,7 +2256,7 @@ declare namespace API {
};
type ToggleUserSubscribeStatusRequest = {
- user_subscribe_id: number;
+ user_subscribe_id: any;
};
type TosConfig = {
@@ -2473,7 +2485,7 @@ declare namespace API {
};
type UpdateUserSubscribeRequest = {
- user_subscribe_id: number;
+ user_subscribe_id: string;
subscribe_id: number;
traffic: number;
expired_at: number;
@@ -2498,6 +2510,8 @@ declare namespace API {
enable_login_notify: boolean;
enable_subscribe_notify: boolean;
enable_trade_notify: boolean;
+ user_group_id: string;
+ group_locked: boolean;
auth_methods: UserAuthMethod[];
user_devices: UserDevice[];
rules: string[];
@@ -2659,4 +2673,263 @@ declare namespace API {
security: string;
security_config: SecurityConfig;
};
+
+ // ===== Group Management Types =====
+
+ type UserGroup = {
+ id: number;
+ name: string;
+ description: string;
+ sort: number;
+ node_group_id?: number | null;
+ for_calculation?: boolean;
+ created_at: number;
+ updated_at: number;
+ };
+
+ type NodeGroup = {
+ id: number;
+ name: string;
+ description: string;
+ sort: number;
+ for_calculation: boolean;
+ min_traffic_gb?: number;
+ max_traffic_gb?: number;
+ created_at: number;
+ updated_at: number;
+ };
+
+ type Subscribe = {
+ id: number;
+ name: string;
+ unit_price: number;
+ unit_time: number;
+ show?: boolean;
+ sell?: boolean;
+ sort: number;
+ created_at: number;
+ updated_at: number;
+ };
+
+ type SubscribeGroupMapping = {
+ subscribe_id: number;
+ user_group_id: string;
+ created_at: number;
+ updated_at: number;
+ };
+
+ type SubscribeGroupMappingInfo = {
+ id: number;
+ subscribe_id: number;
+ user_group_id: number;
+ subscribe?: Subscribe;
+ user_group?: UserGroup;
+ created_at: number;
+ updated_at: number;
+ };
+
+ type GroupHistory = {
+ id: number;
+ group_mode: string;
+ trigger_type: string;
+ total_users: number;
+ success_count: number;
+ failed_count: number;
+ start_time?: number;
+ end_time?: number;
+ operator?: string;
+ error_log?: string;
+ created_at: number;
+ };
+
+ type GroupHistoryDetailItem = {
+ id: number;
+ history_id: number;
+ user_group_id: string;
+ node_group_id: string;
+ user_count: number;
+ node_count: number;
+ user_data?: string;
+ created_at: number;
+ };
+
+ type GroupHistoryDetail = {
+ id: number;
+ group_mode: string;
+ trigger_type: string;
+ total_users: number;
+ success_count: number;
+ failed_count: number;
+ start_time?: number;
+ end_time?: number;
+ operator?: string;
+ error_log?: string;
+ created_at: number;
+ config_snapshot?: {
+ group_details?: GroupHistoryDetailItem[];
+ config?: Record;
+ };
+ };
+
+ type RecalculationState = {
+ state: string;
+ progress: number;
+ total: number;
+ };
+
+ // ===== Group Request/Response Types =====
+
+ type GetUserGroupListRequest = {
+ page: number;
+ size: number;
+ group_id?: string;
+ };
+
+ type GetUserGroupListResponse = {
+ total: number;
+ list: UserGroup[];
+ };
+
+ type CreateUserGroupRequest = {
+ name: string;
+ description?: string;
+ sort?: number;
+ node_group_id?: number | null;
+ for_calculation?: boolean | null;
+ };
+
+ type UpdateUserGroupRequest = {
+ id: number;
+ name?: string;
+ description?: string;
+ sort?: number;
+ node_group_id?: number | null;
+ for_calculation?: boolean | null;
+ };
+
+ type DeleteUserGroupRequest = {
+ id: number;
+ };
+
+ type BindNodeGroupsRequest = {
+ user_group_ids: number[];
+ node_group_id?: number | null;
+ };
+
+ type GetNodeGroupListRequest = {
+ page: number;
+ size: number;
+ group_id?: string;
+ };
+
+ type GetNodeGroupListResponse = {
+ total: number;
+ list: NodeGroup[];
+ };
+
+ type CreateNodeGroupRequest = {
+ name: string;
+ description?: string;
+ sort?: number;
+ for_calculation?: boolean;
+ min_traffic_gb?: number;
+ max_traffic_gb?: number;
+ };
+
+ type UpdateNodeGroupRequest = {
+ id: number;
+ name?: string;
+ description?: string;
+ sort?: number;
+ for_calculation?: boolean;
+ min_traffic_gb?: number;
+ max_traffic_gb?: number;
+ };
+
+ type DeleteNodeGroupRequest = {
+ id: number;
+ };
+
+ type GetSubscribeMappingRequest = {
+ page: number;
+ size: number;
+ subscribe_id?: number;
+ user_group_id?: number;
+ };
+
+ type GetSubscribeMappingResponse = {
+ total: number;
+ list: SubscribeGroupMappingInfo[];
+ };
+
+ type UpdateSubscribeMappingRequest = {
+ subscribe_id: number;
+ user_group_id: number;
+ };
+
+ type GetGroupConfigResponse = {
+ enabled: boolean;
+ mode: string;
+ };
+
+ type UpdateGroupConfigRequest = {
+ enabled?: boolean;
+ mode?: string;
+ };
+
+ type RecalculateGroupRequest = {
+ mode: string;
+ };
+
+ type GetGroupHistoryRequest = {
+ page: number;
+ size: number;
+ group_mode?: string;
+ trigger_type?: string;
+ };
+
+ type GetGroupHistoryResponse = {
+ total: number;
+ list: GroupHistory[];
+ };
+
+ type GetGroupHistoryDetailRequest = {
+ id: number;
+ };
+
+ type GetGroupHistoryDetailResponse = GroupHistoryDetail;
+
+ type ExportGroupResultRequest = {
+ history_id?: number;
+ };
+ type MigrateUsersRequest = {
+ from_user_group_id: number;
+ to_user_group_id: number;
+ include_locked?: boolean;
+ };
+ type MigrateUsersResponse = {
+ success_count: number;
+ failed_count: number;
+ };
+ type ResetGroupsRequest = {
+ confirm: boolean;
+ };
+
+ type PreviewUserNodesRequest = {
+ user_id: number;
+ };
+
+ type PreviewUserNodesResponse = {
+ user_id: number;
+ node_groups: Array<{
+ id: number;
+ name: string;
+ nodes: Array<{
+ id: number;
+ name: string;
+ address: string;
+ port: number;
+ }>;
+ }>;
+ };
}