mirror of
https://github.com/perfect-panel/ppanel-web.git
synced 2026-02-15 12:51:11 -05:00
♻️ refactor: Replace useQuery with Zustand store for subscription and node data management
This commit is contained in:
parent
bdd53b1551
commit
c6dd0b63f2
12
CHANGELOG.md
12
CHANGELOG.md
@ -1,16 +1,16 @@
|
|||||||
<a name="readme-top"></a>
|
<a name="readme-top"></a>
|
||||||
|
|
||||||
# Changelog
|
# Changelog
|
||||||
|
|
||||||
## [1.4.4](https://github.com/perfect-panel/ppanel-web/compare/v1.4.3...v1.4.4) (2025-09-16)
|
## [1.4.4](https://github.com/perfect-panel/ppanel-web/compare/v1.4.3...v1.4.4) (2025-09-16)
|
||||||
|
|
||||||
|
|
||||||
### 🐛 Bug Fixes
|
### 🐛 Bug Fixes
|
||||||
|
|
||||||
* Add minimum value constraint for count and user limit inputs in CouponForm ([6991b69](https://github.com/perfect-panel/ppanel-web/commit/6991b69))
|
- Add minimum value constraint for count and user limit inputs in CouponForm ([6991b69](https://github.com/perfect-panel/ppanel-web/commit/6991b69))
|
||||||
* Add protocol-related constants, default configurations, field definitions, and validation modes ([d685407](https://github.com/perfect-panel/ppanel-web/commit/d685407))
|
- Add protocol-related constants, default configurations, field definitions, and validation modes ([d685407](https://github.com/perfect-panel/ppanel-web/commit/d685407))
|
||||||
* Added the enabled field in the protocol configuration, updated the related type definition and default configuration ([2b0cf9a](https://github.com/perfect-panel/ppanel-web/commit/2b0cf9a))
|
- Added the enabled field in the protocol configuration, updated the related type definition and default configuration ([2b0cf9a](https://github.com/perfect-panel/ppanel-web/commit/2b0cf9a))
|
||||||
* Filter available protocols to exclude disabled ones in NodeForm ([982d288](https://github.com/perfect-panel/ppanel-web/commit/982d288))
|
- Filter available protocols to exclude disabled ones in NodeForm ([982d288](https://github.com/perfect-panel/ppanel-web/commit/982d288))
|
||||||
* Refactor key generation logic and update dependencies for ML-KEM-768 integration ([b8f630f](https://github.com/perfect-panel/ppanel-web/commit/b8f630f))
|
- Refactor key generation logic and update dependencies for ML-KEM-768 integration ([b8f630f](https://github.com/perfect-panel/ppanel-web/commit/b8f630f))
|
||||||
|
|
||||||
<a name="readme-top"></a>
|
<a name="readme-top"></a>
|
||||||
|
|
||||||
|
|||||||
@ -1,8 +1,7 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { getSubscribeList } from '@/services/admin/subscribe';
|
import { useSubscribe } from '@/store/subscribe';
|
||||||
import { zodResolver } from '@hookform/resolvers/zod';
|
import { zodResolver } from '@hookform/resolvers/zod';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
|
||||||
import { Button } from '@workspace/ui/components/button';
|
import { Button } from '@workspace/ui/components/button';
|
||||||
import {
|
import {
|
||||||
Form,
|
Form,
|
||||||
@ -81,16 +80,7 @@ export default function CouponForm<T extends Record<string, any>>({
|
|||||||
|
|
||||||
const type = form.watch('type');
|
const type = form.watch('type');
|
||||||
|
|
||||||
const { data: subscribe } = useQuery({
|
const { subscribes } = useSubscribe();
|
||||||
queryKey: ['getSubscribeList', 'all'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await getSubscribeList({
|
|
||||||
page: 1,
|
|
||||||
size: 9999,
|
|
||||||
});
|
|
||||||
return data.data?.list as API.Subscribe[];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Sheet open={open} onOpenChange={setOpen}>
|
<Sheet open={open} onOpenChange={setOpen}>
|
||||||
@ -247,9 +237,9 @@ export default function CouponForm<T extends Record<string, any>>({
|
|||||||
onChange={(value) => {
|
onChange={(value) => {
|
||||||
form.setValue(field.name, value);
|
form.setValue(field.name, value);
|
||||||
}}
|
}}
|
||||||
options={subscribe?.map((item: API.Subscribe) => ({
|
options={subscribes?.map((item) => ({
|
||||||
value: item.id,
|
value: item.id!,
|
||||||
label: item.name,
|
label: item.name!,
|
||||||
}))}
|
}))}
|
||||||
/>
|
/>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
|
|||||||
@ -9,9 +9,8 @@ import {
|
|||||||
getCouponList,
|
getCouponList,
|
||||||
updateCoupon,
|
updateCoupon,
|
||||||
} from '@/services/admin/coupon';
|
} from '@/services/admin/coupon';
|
||||||
import { getSubscribeList } from '@/services/admin/subscribe';
|
import { useSubscribe } from '@/store/subscribe';
|
||||||
import { formatDate } from '@/utils/common';
|
import { formatDate } from '@/utils/common';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
|
||||||
import { Badge } from '@workspace/ui/components/badge';
|
import { Badge } from '@workspace/ui/components/badge';
|
||||||
import { Button } from '@workspace/ui/components/button';
|
import { Button } from '@workspace/ui/components/button';
|
||||||
import { Switch } from '@workspace/ui/components/switch';
|
import { Switch } from '@workspace/ui/components/switch';
|
||||||
@ -24,16 +23,7 @@ import CouponForm from './coupon-form';
|
|||||||
export default function Page() {
|
export default function Page() {
|
||||||
const t = useTranslations('coupon');
|
const t = useTranslations('coupon');
|
||||||
const [loading, setLoading] = useState(false);
|
const [loading, setLoading] = useState(false);
|
||||||
const { data } = useQuery({
|
const { subscribes } = useSubscribe();
|
||||||
queryKey: ['getSubscribeList', 'all'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await getSubscribeList({
|
|
||||||
page: 1,
|
|
||||||
size: 9999,
|
|
||||||
});
|
|
||||||
return data.data?.list as API.SubscribeGroup[];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
const ref = useRef<ProTableActions>(null);
|
const ref = useRef<ProTableActions>(null);
|
||||||
return (
|
return (
|
||||||
<ProTable<API.Coupon, { group_id: number; query: string }>
|
<ProTable<API.Coupon, { group_id: number; query: string }>
|
||||||
@ -67,8 +57,8 @@ export default function Page() {
|
|||||||
{
|
{
|
||||||
key: 'subscribe',
|
key: 'subscribe',
|
||||||
placeholder: t('subscribe'),
|
placeholder: t('subscribe'),
|
||||||
options: data?.map((item) => ({
|
options: subscribes?.map((item) => ({
|
||||||
label: item.name,
|
label: item.name!,
|
||||||
value: String(item.id),
|
value: String(item.id),
|
||||||
})),
|
})),
|
||||||
},
|
},
|
||||||
|
|||||||
@ -2,8 +2,7 @@
|
|||||||
|
|
||||||
import { ProTable } from '@/components/pro-table';
|
import { ProTable } from '@/components/pro-table';
|
||||||
import { filterServerTrafficLog } from '@/services/admin/log';
|
import { filterServerTrafficLog } from '@/services/admin/log';
|
||||||
import { filterServerList } from '@/services/admin/server';
|
import { useServer } from '@/store/server';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
|
||||||
import { Button } from '@workspace/ui/components/button';
|
import { Button } from '@workspace/ui/components/button';
|
||||||
import { formatBytes } from '@workspace/ui/utils';
|
import { formatBytes } from '@workspace/ui/utils';
|
||||||
import { useTranslations } from 'next-intl';
|
import { useTranslations } from 'next-intl';
|
||||||
@ -13,20 +12,10 @@ import { useSearchParams } from 'next/navigation';
|
|||||||
export default function ServerTrafficLogPage() {
|
export default function ServerTrafficLogPage() {
|
||||||
const t = useTranslations('log');
|
const t = useTranslations('log');
|
||||||
const sp = useSearchParams();
|
const sp = useSearchParams();
|
||||||
|
const { getServerName } = useServer();
|
||||||
|
|
||||||
const today = new Date().toISOString().split('T')[0];
|
const today = new Date().toISOString().split('T')[0];
|
||||||
|
|
||||||
const { data: servers = [] } = useQuery({
|
|
||||||
queryKey: ['filterServerListAll'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await filterServerList({ page: 1, size: 999999999 });
|
|
||||||
return data?.data?.list || [];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const getServerName = (id?: number) =>
|
|
||||||
id ? (servers.find((s) => s.id === id)?.name ?? `Server ${id}`) : 'Unknown';
|
|
||||||
|
|
||||||
const initialFilters = {
|
const initialFilters = {
|
||||||
date: sp.get('date') || today,
|
date: sp.get('date') || today,
|
||||||
server_id: sp.get('server_id') ? Number(sp.get('server_id')) : undefined,
|
server_id: sp.get('server_id') ? Number(sp.get('server_id')) : undefined,
|
||||||
|
|||||||
@ -3,9 +3,8 @@
|
|||||||
import { UserDetail, UserSubscribeDetail } from '@/app/dashboard/user/user-detail';
|
import { UserDetail, UserSubscribeDetail } from '@/app/dashboard/user/user-detail';
|
||||||
import { ProTable } from '@/components/pro-table';
|
import { ProTable } from '@/components/pro-table';
|
||||||
import { filterTrafficLogDetails } from '@/services/admin/log';
|
import { filterTrafficLogDetails } from '@/services/admin/log';
|
||||||
import { filterServerList } from '@/services/admin/server';
|
import { useServer } from '@/store/server';
|
||||||
import { formatDate } from '@/utils/common';
|
import { formatDate } from '@/utils/common';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
|
||||||
import { formatBytes } from '@workspace/ui/utils';
|
import { formatBytes } from '@workspace/ui/utils';
|
||||||
import { useTranslations } from 'next-intl';
|
import { useTranslations } from 'next-intl';
|
||||||
import { useSearchParams } from 'next/navigation';
|
import { useSearchParams } from 'next/navigation';
|
||||||
@ -13,20 +12,10 @@ import { useSearchParams } from 'next/navigation';
|
|||||||
export default function TrafficDetailsPage() {
|
export default function TrafficDetailsPage() {
|
||||||
const t = useTranslations('log');
|
const t = useTranslations('log');
|
||||||
const sp = useSearchParams();
|
const sp = useSearchParams();
|
||||||
|
const { getServerName } = useServer();
|
||||||
|
|
||||||
const today = new Date().toISOString().split('T')[0];
|
const today = new Date().toISOString().split('T')[0];
|
||||||
|
|
||||||
const { data: servers = [] } = useQuery({
|
|
||||||
queryKey: ['filterServerListAll'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await filterServerList({ page: 1, size: 999999999 });
|
|
||||||
return data?.data?.list || [];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const getServerName = (id?: number) =>
|
|
||||||
id ? (servers.find((s) => s.id === id)?.name ?? `Server ${id}`) : 'Unknown';
|
|
||||||
|
|
||||||
const initialFilters = {
|
const initialFilters = {
|
||||||
date: sp.get('date') || today,
|
date: sp.get('date') || today,
|
||||||
server_id: sp.get('server_id') ? Number(sp.get('server_id')) : undefined,
|
server_id: sp.get('server_id') ? Number(sp.get('server_id')) : undefined,
|
||||||
|
|||||||
@ -2,9 +2,8 @@
|
|||||||
|
|
||||||
import { Display } from '@/components/display';
|
import { Display } from '@/components/display';
|
||||||
import { createQuotaTask, queryQuotaTaskPreCount } from '@/services/admin/marketing';
|
import { createQuotaTask, queryQuotaTaskPreCount } from '@/services/admin/marketing';
|
||||||
import { getSubscribeList } from '@/services/admin/subscribe';
|
import { useSubscribe } from '@/store/subscribe';
|
||||||
import { zodResolver } from '@hookform/resolvers/zod';
|
import { zodResolver } from '@hookform/resolvers/zod';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
|
||||||
import { Button } from '@workspace/ui/components/button';
|
import { Button } from '@workspace/ui/components/button';
|
||||||
import {
|
import {
|
||||||
Form,
|
Form,
|
||||||
@ -73,17 +72,7 @@ export default function QuotaBroadcastForm() {
|
|||||||
const [isSubmitting, setIsSubmitting] = useState(false);
|
const [isSubmitting, setIsSubmitting] = useState(false);
|
||||||
const [open, setOpen] = useState(false);
|
const [open, setOpen] = useState(false);
|
||||||
|
|
||||||
// Get subscribe list
|
const { subscribes } = useSubscribe();
|
||||||
const { data: subscribeList } = useQuery({
|
|
||||||
queryKey: ['getSubscribeList', 'all'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await getSubscribeList({
|
|
||||||
page: 1,
|
|
||||||
size: 999999999,
|
|
||||||
});
|
|
||||||
return data.data?.list as API.SubscribeItem[];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
// Calculate recipient count
|
// Calculate recipient count
|
||||||
const calculateRecipients = async () => {
|
const calculateRecipients = async () => {
|
||||||
@ -217,21 +206,19 @@ export default function QuotaBroadcastForm() {
|
|||||||
value={field.value || []}
|
value={field.value || []}
|
||||||
onChange={field.onChange}
|
onChange={field.onChange}
|
||||||
placeholder={t('pleaseSelectSubscribers')}
|
placeholder={t('pleaseSelectSubscribers')}
|
||||||
options={
|
options={subscribes?.map((subscribe) => ({
|
||||||
subscribeList?.map((subscribe) => ({
|
value: subscribe.id!,
|
||||||
value: subscribe.id!,
|
label: subscribe.name!,
|
||||||
label: subscribe.name!,
|
children: (
|
||||||
children: (
|
<div>
|
||||||
<div>
|
<div>{subscribe.name}</div>
|
||||||
<div>{subscribe.name}</div>
|
<div className='text-muted-foreground text-xs'>
|
||||||
<div className='text-muted-foreground text-xs'>
|
<Display type='traffic' value={subscribe.traffic || 0} /> /{' '}
|
||||||
<Display type='traffic' value={subscribe.traffic || 0} /> /{' '}
|
<Display type='currency' value={subscribe.unit_price || 0} />
|
||||||
<Display type='currency' value={subscribe.unit_price || 0} />
|
|
||||||
</div>
|
|
||||||
</div>
|
</div>
|
||||||
),
|
</div>
|
||||||
})) || []
|
),
|
||||||
}
|
}))}
|
||||||
/>
|
/>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
<FormMessage />
|
<FormMessage />
|
||||||
|
|||||||
@ -3,9 +3,8 @@
|
|||||||
import { Display } from '@/components/display';
|
import { Display } from '@/components/display';
|
||||||
import { ProTable } from '@/components/pro-table';
|
import { ProTable } from '@/components/pro-table';
|
||||||
import { queryQuotaTaskList } from '@/services/admin/marketing';
|
import { queryQuotaTaskList } from '@/services/admin/marketing';
|
||||||
import { getSubscribeList } from '@/services/admin/subscribe';
|
import { useSubscribe } from '@/store/subscribe';
|
||||||
import { formatDate } from '@/utils/common';
|
import { formatDate } from '@/utils/common';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
|
||||||
import { Badge } from '@workspace/ui/components/badge';
|
import { Badge } from '@workspace/ui/components/badge';
|
||||||
import { ScrollArea } from '@workspace/ui/components/scroll-area';
|
import { ScrollArea } from '@workspace/ui/components/scroll-area';
|
||||||
import {
|
import {
|
||||||
@ -23,21 +22,9 @@ export default function QuotaTaskManager() {
|
|||||||
const t = useTranslations('marketing');
|
const t = useTranslations('marketing');
|
||||||
const [open, setOpen] = useState(false);
|
const [open, setOpen] = useState(false);
|
||||||
|
|
||||||
// Get subscribe list to show subscription names
|
const { subscribes } = useSubscribe();
|
||||||
const { data: subscribeList } = useQuery({
|
|
||||||
queryKey: ['getSubscribeList', 'all'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await getSubscribeList({
|
|
||||||
page: 1,
|
|
||||||
size: 999999999,
|
|
||||||
});
|
|
||||||
return data.data?.list as API.SubscribeItem[];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
// Create a map for quick lookup of subscription names
|
|
||||||
const subscribeMap =
|
const subscribeMap =
|
||||||
subscribeList?.reduce(
|
subscribes?.reduce(
|
||||||
(acc, subscribe) => {
|
(acc, subscribe) => {
|
||||||
acc[subscribe.id!] = subscribe.name!;
|
acc[subscribe.id!] = subscribe.name!;
|
||||||
return acc;
|
return acc;
|
||||||
|
|||||||
@ -1,8 +1,8 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { filterServerList, queryNodeTag } from '@/services/admin/server';
|
import { useNode } from '@/store/node';
|
||||||
|
import { useServer } from '@/store/server';
|
||||||
import { zodResolver } from '@hookform/resolvers/zod';
|
import { zodResolver } from '@hookform/resolvers/zod';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
|
||||||
import { Button } from '@workspace/ui/components/button';
|
import { Button } from '@workspace/ui/components/button';
|
||||||
import {
|
import {
|
||||||
Form,
|
Form,
|
||||||
@ -40,8 +40,6 @@ export type ProtocolName =
|
|||||||
| 'tuic'
|
| 'tuic'
|
||||||
| 'anytls';
|
| 'anytls';
|
||||||
|
|
||||||
type ServerRow = API.Server;
|
|
||||||
|
|
||||||
const buildSchema = (t: ReturnType<typeof useTranslations>) =>
|
const buildSchema = (t: ReturnType<typeof useTranslations>) =>
|
||||||
z.object({
|
z.object({
|
||||||
name: z.string().trim().min(1, t('errors.nameRequired')),
|
name: z.string().trim().min(1, t('errors.nameRequired')),
|
||||||
@ -103,38 +101,12 @@ export default function NodeForm(props: {
|
|||||||
|
|
||||||
const serverId = form.watch('server_id');
|
const serverId = form.watch('server_id');
|
||||||
|
|
||||||
const { data } = useQuery({
|
const { servers, getAvailableProtocols } = useServer();
|
||||||
enabled: open,
|
const { tags } = useNode();
|
||||||
queryKey: ['filterServerListAll'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await filterServerList({ page: 1, size: 999999999 });
|
|
||||||
return data?.data?.list || [];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
const servers: ServerRow[] = data as ServerRow[];
|
|
||||||
|
|
||||||
const { data: tagsData } = useQuery({
|
const existingTags: string[] = tags || [];
|
||||||
enabled: open,
|
|
||||||
queryKey: ['queryNodeTag'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await queryNodeTag();
|
|
||||||
return data?.data?.tags || [];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
const existingTags: string[] = tagsData as string[];
|
|
||||||
|
|
||||||
const currentServer = useMemo(() => servers?.find((s) => s.id === serverId), [servers, serverId]);
|
const availableProtocols = getAvailableProtocols(serverId);
|
||||||
|
|
||||||
const availableProtocols = useMemo(() => {
|
|
||||||
if (!currentServer?.protocols) return [];
|
|
||||||
|
|
||||||
return currentServer.protocols
|
|
||||||
.filter((p) => p.enable !== false)
|
|
||||||
.map((p) => ({
|
|
||||||
protocol: p.type,
|
|
||||||
port: p.port,
|
|
||||||
}));
|
|
||||||
}, [currentServer]);
|
|
||||||
|
|
||||||
useEffect(() => {
|
useEffect(() => {
|
||||||
if (initialValues) {
|
if (initialValues) {
|
||||||
@ -176,12 +148,11 @@ export default function NodeForm(props: {
|
|||||||
fieldsToFill.push('address');
|
fieldsToFill.push('address');
|
||||||
}
|
}
|
||||||
|
|
||||||
const protocols =
|
const protocols = getAvailableProtocols(id);
|
||||||
(selectedServer.protocols as Array<{ type: ProtocolName; port?: number }>) || [];
|
|
||||||
const firstProtocol = protocols[0];
|
const firstProtocol = protocols[0];
|
||||||
|
|
||||||
if (firstProtocol && (!currentValues.protocol || autoFilledFields.has('protocol'))) {
|
if (firstProtocol && (!currentValues.protocol || autoFilledFields.has('protocol'))) {
|
||||||
form.setValue('protocol', firstProtocol.type, { shouldDirty: false });
|
form.setValue('protocol', firstProtocol.protocol, { shouldDirty: false });
|
||||||
fieldsToFill.push('protocol');
|
fieldsToFill.push('protocol');
|
||||||
|
|
||||||
if (!currentValues.port || currentValues.port === 0 || autoFilledFields.has('port')) {
|
if (!currentValues.port || currentValues.port === 0 || autoFilledFields.has('port')) {
|
||||||
@ -203,7 +174,7 @@ export default function NodeForm(props: {
|
|||||||
const protocol = (nextProto || '') as ProtocolName | '';
|
const protocol = (nextProto || '') as ProtocolName | '';
|
||||||
form.setValue('protocol', protocol);
|
form.setValue('protocol', protocol);
|
||||||
|
|
||||||
if (!protocol || !currentServer) {
|
if (!protocol || !serverId) {
|
||||||
removeAutoFilledField('protocol');
|
removeAutoFilledField('protocol');
|
||||||
return;
|
return;
|
||||||
}
|
}
|
||||||
@ -214,9 +185,7 @@ export default function NodeForm(props: {
|
|||||||
removeAutoFilledField('protocol');
|
removeAutoFilledField('protocol');
|
||||||
|
|
||||||
if (!currentValues.port || currentValues.port === 0 || isPortAutoFilled) {
|
if (!currentValues.port || currentValues.port === 0 || isPortAutoFilled) {
|
||||||
const protocolData = (
|
const protocolData = availableProtocols.find((p) => p.protocol === protocol);
|
||||||
currentServer.protocols as Array<{ type: ProtocolName; port?: number }>
|
|
||||||
)?.find((p) => p.type === protocol);
|
|
||||||
|
|
||||||
if (protocolData) {
|
if (protocolData) {
|
||||||
const port = protocolData.port || 0;
|
const port = protocolData.port || 0;
|
||||||
|
|||||||
@ -5,12 +5,12 @@ import {
|
|||||||
createNode,
|
createNode,
|
||||||
deleteNode,
|
deleteNode,
|
||||||
filterNodeList,
|
filterNodeList,
|
||||||
filterServerList,
|
|
||||||
resetSortWithNode,
|
resetSortWithNode,
|
||||||
toggleNodeStatus,
|
toggleNodeStatus,
|
||||||
updateNode,
|
updateNode,
|
||||||
} from '@/services/admin/server';
|
} from '@/services/admin/server';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
import { useNode } from '@/store/node';
|
||||||
|
import { useServer } from '@/store/server';
|
||||||
import { Badge } from '@workspace/ui/components/badge';
|
import { Badge } from '@workspace/ui/components/badge';
|
||||||
import { Button } from '@workspace/ui/components/button';
|
import { Button } from '@workspace/ui/components/button';
|
||||||
import { Switch } from '@workspace/ui/components/switch';
|
import { Switch } from '@workspace/ui/components/switch';
|
||||||
@ -25,24 +25,9 @@ export default function NodesPage() {
|
|||||||
const ref = useRef<ProTableActions>(null);
|
const ref = useRef<ProTableActions>(null);
|
||||||
const [loading, setLoading] = useState(false);
|
const [loading, setLoading] = useState(false);
|
||||||
|
|
||||||
const { data: servers = [] } = useQuery({
|
// Use our zustand store for server data
|
||||||
queryKey: ['filterServerListAll'],
|
const { getServerName, getServerAddress, getProtocolPort } = useServer();
|
||||||
queryFn: async () => {
|
const { fetchNodes } = useNode();
|
||||||
const { data } = await filterServerList({ page: 1, size: 999999999 });
|
|
||||||
return data?.data?.list || [];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const getServerName = (id?: number) =>
|
|
||||||
id ? (servers.find((s) => s.id === id)?.name ?? `#${id}`) : '—';
|
|
||||||
const getServerOriginAddr = (id?: number) =>
|
|
||||||
id ? (servers.find((s) => s.id === id)?.address ?? '—') : '—';
|
|
||||||
const getProtocolOriginPort = (id?: number, proto?: string) => {
|
|
||||||
if (!id || !proto) return '—';
|
|
||||||
const hit = servers.find((s) => s.id === id)?.protocols?.find((p) => (p as any).type === proto);
|
|
||||||
const p = (hit as any)?.port as number | undefined;
|
|
||||||
return typeof p === 'number' ? String(p) : '—';
|
|
||||||
};
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<ProTable<API.Node, { search: string }>
|
<ProTable<API.Node, { search: string }>
|
||||||
@ -69,6 +54,7 @@ export default function NodesPage() {
|
|||||||
await createNode(body);
|
await createNode(body);
|
||||||
toast.success(t('created'));
|
toast.success(t('created'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchNodes();
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
return true;
|
return true;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
@ -90,6 +76,7 @@ export default function NodesPage() {
|
|||||||
await toggleNodeStatus({ id: row.original.id, enable: v });
|
await toggleNodeStatus({ id: row.original.id, enable: v });
|
||||||
toast.success(v ? t('enabled_on') : t('enabled_off'));
|
toast.success(v ? t('enabled_on') : t('enabled_off'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchNodes();
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
),
|
),
|
||||||
@ -106,14 +93,14 @@ export default function NodesPage() {
|
|||||||
id: 'server_id',
|
id: 'server_id',
|
||||||
header: t('server'),
|
header: t('server'),
|
||||||
cell: ({ row }) => (
|
cell: ({ row }) => (
|
||||||
<div className='flex flex-wrap gap-2'>
|
<div className='space-y-1'>
|
||||||
<Badge variant='outline'>
|
<Badge variant='outline'>
|
||||||
{getServerName(row.original.server_id)} ·{' '}
|
{getServerName(row.original.server_id)} : {getServerAddress(row.original.server_id)}
|
||||||
{getServerOriginAddr(row.original.server_id)}
|
|
||||||
</Badge>
|
</Badge>
|
||||||
|
<br />
|
||||||
<Badge variant='outline'>
|
<Badge variant='outline'>
|
||||||
{row.original.protocol || '—'} ·{' '}
|
{row.original.protocol || '—'} :{' '}
|
||||||
{getProtocolOriginPort(row.original.server_id, row.original.protocol)}
|
{getProtocolPort(row.original.server_id, row.original.protocol)}
|
||||||
</Badge>
|
</Badge>
|
||||||
</div>
|
</div>
|
||||||
),
|
),
|
||||||
@ -163,6 +150,7 @@ export default function NodesPage() {
|
|||||||
await updateNode(body);
|
await updateNode(body);
|
||||||
toast.success(t('updated'));
|
toast.success(t('updated'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchNodes();
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
return true;
|
return true;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
@ -180,6 +168,7 @@ export default function NodesPage() {
|
|||||||
await deleteNode({ id: row.id } as any);
|
await deleteNode({ id: row.id } as any);
|
||||||
toast.success(t('deleted'));
|
toast.success(t('deleted'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchNodes();
|
||||||
}}
|
}}
|
||||||
cancelText={t('cancel')}
|
cancelText={t('cancel')}
|
||||||
confirmText={t('confirm')}
|
confirmText={t('confirm')}
|
||||||
@ -195,6 +184,7 @@ export default function NodesPage() {
|
|||||||
});
|
});
|
||||||
toast.success(t('copied'));
|
toast.success(t('copied'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchNodes();
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
{t('copy')}
|
{t('copy')}
|
||||||
@ -211,6 +201,7 @@ export default function NodesPage() {
|
|||||||
await Promise.all(rows.map((r) => deleteNode({ id: r.id } as any)));
|
await Promise.all(rows.map((r) => deleteNode({ id: r.id } as any)));
|
||||||
toast.success(t('deleted'));
|
toast.success(t('deleted'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchNodes();
|
||||||
}}
|
}}
|
||||||
cancelText={t('cancel')}
|
cancelText={t('cancel')}
|
||||||
confirmText={t('confirm')}
|
confirmText={t('confirm')}
|
||||||
|
|||||||
@ -1,6 +1,5 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { useQuery } from '@tanstack/react-query';
|
|
||||||
import { useTranslations } from 'next-intl';
|
import { useTranslations } from 'next-intl';
|
||||||
import { useSearchParams } from 'next/navigation';
|
import { useSearchParams } from 'next/navigation';
|
||||||
import { useRef } from 'react';
|
import { useRef } from 'react';
|
||||||
@ -8,7 +7,7 @@ import { useRef } from 'react';
|
|||||||
import { Display } from '@/components/display';
|
import { Display } from '@/components/display';
|
||||||
import { ProTable, ProTableActions } from '@/components/pro-table';
|
import { ProTable, ProTableActions } from '@/components/pro-table';
|
||||||
import { getOrderList, updateOrderStatus } from '@/services/admin/order';
|
import { getOrderList, updateOrderStatus } from '@/services/admin/order';
|
||||||
import { getSubscribeList } from '@/services/admin/subscribe';
|
import { useSubscribe } from '@/store/subscribe';
|
||||||
import { formatDate } from '@/utils/common';
|
import { formatDate } from '@/utils/common';
|
||||||
import { Badge } from '@workspace/ui/components/badge';
|
import { Badge } from '@workspace/ui/components/badge';
|
||||||
import { Button } from '@workspace/ui/components/button';
|
import { Button } from '@workspace/ui/components/button';
|
||||||
@ -32,16 +31,7 @@ export default function Page() {
|
|||||||
|
|
||||||
const ref = useRef<ProTableActions>(null);
|
const ref = useRef<ProTableActions>(null);
|
||||||
|
|
||||||
const { data: subscribeList } = useQuery({
|
const { subscribes, getSubscribeName } = useSubscribe();
|
||||||
queryKey: ['getSubscribeList', 'all'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await getSubscribeList({
|
|
||||||
page: 1,
|
|
||||||
size: 999999999,
|
|
||||||
});
|
|
||||||
return data.data?.list as API.SubscribeGroup[];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const initialFilters = {
|
const initialFilters = {
|
||||||
search: sp.get('search') || undefined,
|
search: sp.get('search') || undefined,
|
||||||
@ -68,9 +58,7 @@ export default function Page() {
|
|||||||
accessorKey: 'subscribe_id',
|
accessorKey: 'subscribe_id',
|
||||||
header: t('subscribe'),
|
header: t('subscribe'),
|
||||||
cell: ({ row }) => {
|
cell: ({ row }) => {
|
||||||
const name = subscribeList?.find(
|
const name = getSubscribeName(row.getValue('subscribe_id'));
|
||||||
(item) => item.id === row.getValue('subscribe_id'),
|
|
||||||
)?.name;
|
|
||||||
const quantity = row.original.quantity;
|
const quantity = row.original.quantity;
|
||||||
return name ? `${name} × ${quantity}` : '';
|
return name ? `${name} × ${quantity}` : '';
|
||||||
},
|
},
|
||||||
@ -186,8 +174,8 @@ export default function Page() {
|
|||||||
{
|
{
|
||||||
key: 'subscribe_id',
|
key: 'subscribe_id',
|
||||||
placeholder: `${t('subscribe')}`,
|
placeholder: `${t('subscribe')}`,
|
||||||
options: subscribeList?.map((item) => ({
|
options: subscribes?.map((item) => ({
|
||||||
label: item.name,
|
label: item.name!,
|
||||||
value: String(item.id),
|
value: String(item.id),
|
||||||
})),
|
})),
|
||||||
},
|
},
|
||||||
|
|||||||
@ -1,9 +1,5 @@
|
|||||||
import { getTranslations } from 'next-intl/server';
|
|
||||||
|
|
||||||
import SubscribeTable from './subscribe-table';
|
import SubscribeTable from './subscribe-table';
|
||||||
|
|
||||||
export default async function Page() {
|
export default async function Page() {
|
||||||
const t = await getTranslations('product');
|
|
||||||
|
|
||||||
return <SubscribeTable />;
|
return <SubscribeTable />;
|
||||||
}
|
}
|
||||||
|
|||||||
@ -1,8 +1,7 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { filterNodeList, queryNodeTag } from '@/services/admin/server';
|
import { useNode } from '@/store/node';
|
||||||
import { zodResolver } from '@hookform/resolvers/zod';
|
import { zodResolver } from '@hookform/resolvers/zod';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
|
||||||
import {
|
import {
|
||||||
Accordion,
|
Accordion,
|
||||||
AccordionContent,
|
AccordionContent,
|
||||||
@ -229,35 +228,9 @@ export default function SubscribeForm<T extends Record<string, any>>({
|
|||||||
if (bool) setOpen(false);
|
if (bool) setOpen(false);
|
||||||
}
|
}
|
||||||
|
|
||||||
const { data: nodes } = useQuery({
|
const { nodes, getAllAvailableTags, getNodesByTag, getNodesWithoutTags } = useNode();
|
||||||
queryKey: ['filterNodeListAll'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await filterNodeList({ page: 1, size: 999999999 });
|
|
||||||
return (data.data?.list || []) as API.Node[];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const { data: allTagsData } = useQuery({
|
const tagGroups = getAllAvailableTags();
|
||||||
queryKey: ['queryNodeTag'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await queryNodeTag();
|
|
||||||
return data?.data?.tags || [];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const nodeExtractedTags = Array.from(
|
|
||||||
new Set(
|
|
||||||
((nodes as API.Node[]) || [])
|
|
||||||
.flatMap((n) => (Array.isArray(n.tags) ? n.tags : []))
|
|
||||||
.filter(Boolean),
|
|
||||||
),
|
|
||||||
) as string[];
|
|
||||||
|
|
||||||
const allAvailableTags = (allTagsData as string[]) || [];
|
|
||||||
|
|
||||||
const tagGroups = Array.from(new Set([...allAvailableTags, ...nodeExtractedTags])).filter(
|
|
||||||
Boolean,
|
|
||||||
);
|
|
||||||
|
|
||||||
const unit_time = form.watch('unit_time');
|
const unit_time = form.watch('unit_time');
|
||||||
|
|
||||||
@ -806,10 +779,7 @@ export default function SubscribeForm<T extends Record<string, any>>({
|
|||||||
{tagGroups.map((tag) => {
|
{tagGroups.map((tag) => {
|
||||||
const value = field.value || [];
|
const value = field.value || [];
|
||||||
const tagId = tag;
|
const tagId = tag;
|
||||||
const nodesWithTag =
|
const nodesWithTag = getNodesByTag(tag);
|
||||||
(nodes as API.Node[])?.filter((n) =>
|
|
||||||
(n.tags || []).includes(tag),
|
|
||||||
) || [];
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<AccordionItem key={tag} value={String(tag)}>
|
<AccordionItem key={tag} value={String(tag)}>
|
||||||
@ -836,22 +806,20 @@ export default function SubscribeForm<T extends Record<string, any>>({
|
|||||||
</AccordionTrigger>
|
</AccordionTrigger>
|
||||||
<AccordionContent>
|
<AccordionContent>
|
||||||
<ul className='space-y-1'>
|
<ul className='space-y-1'>
|
||||||
{(nodes as API.Node[])
|
{getNodesByTag(tag).map((node) => (
|
||||||
?.filter((n) => (n.tags || []).includes(tag))
|
<li
|
||||||
?.map((node) => (
|
key={node.id}
|
||||||
<li
|
className='flex items-center justify-between gap-3'
|
||||||
key={node.id}
|
>
|
||||||
className='flex items-center justify-between gap-3'
|
<span className='flex-1'>{node.name}</span>
|
||||||
>
|
<span className='flex-1'>
|
||||||
<span className='flex-1'>{node.name}</span>
|
{node.address}:{node.port}
|
||||||
<span className='flex-1'>
|
</span>
|
||||||
{node.address}:{node.port}
|
<span className='flex-1 text-right'>
|
||||||
</span>
|
{node.protocol}
|
||||||
<span className='flex-1 text-right'>
|
</span>
|
||||||
{node.protocol}
|
</li>
|
||||||
</span>
|
))}
|
||||||
</li>
|
|
||||||
))}
|
|
||||||
</ul>
|
</ul>
|
||||||
</AccordionContent>
|
</AccordionContent>
|
||||||
</AccordionItem>
|
</AccordionItem>
|
||||||
@ -872,34 +840,32 @@ export default function SubscribeForm<T extends Record<string, any>>({
|
|||||||
<FormLabel>{t('form.node')}</FormLabel>
|
<FormLabel>{t('form.node')}</FormLabel>
|
||||||
<FormControl>
|
<FormControl>
|
||||||
<div className='flex flex-col gap-2'>
|
<div className='flex flex-col gap-2'>
|
||||||
{(nodes as API.Node[])
|
{getNodesWithoutTags().map((item) => {
|
||||||
?.filter((item) => (item.tags || []).length === 0)
|
const value = field.value || [];
|
||||||
?.map((item) => {
|
|
||||||
const value = field.value || [];
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<div className='flex items-center gap-2' key={item.id}>
|
<div className='flex items-center gap-2' key={item.id}>
|
||||||
<Checkbox
|
<Checkbox
|
||||||
checked={value.includes(item.id!)}
|
checked={value.includes(item.id!)}
|
||||||
onCheckedChange={(checked) => {
|
onCheckedChange={(checked) => {
|
||||||
return checked
|
return checked
|
||||||
? form.setValue(field.name, [...value, item.id])
|
? form.setValue(field.name, [...value, item.id])
|
||||||
: form.setValue(
|
: form.setValue(
|
||||||
field.name,
|
field.name,
|
||||||
value.filter((value: number) => value !== item.id),
|
value.filter((value: number) => value !== item.id),
|
||||||
);
|
);
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
<Label className='flex w-full items-center justify-between gap-3'>
|
<Label className='flex w-full items-center justify-between gap-3'>
|
||||||
<span className='flex-1'>{item.name}</span>
|
<span className='flex-1'>{item.name}</span>
|
||||||
<span className='flex-1'>
|
<span className='flex-1'>
|
||||||
{item.address}:{item.port}
|
{item.address}:{item.port}
|
||||||
</span>
|
</span>
|
||||||
<span className='flex-1 text-right'>{item.protocol}</span>
|
<span className='flex-1 text-right'>{item.protocol}</span>
|
||||||
</Label>
|
</Label>
|
||||||
</div>
|
</div>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
</div>
|
</div>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
<FormMessage />
|
<FormMessage />
|
||||||
|
|||||||
@ -10,6 +10,7 @@ import {
|
|||||||
subscribeSort,
|
subscribeSort,
|
||||||
updateSubscribe,
|
updateSubscribe,
|
||||||
} from '@/services/admin/subscribe';
|
} from '@/services/admin/subscribe';
|
||||||
|
import { useSubscribe } from '@/store/subscribe';
|
||||||
import { Badge } from '@workspace/ui/components/badge';
|
import { Badge } from '@workspace/ui/components/badge';
|
||||||
import { Button } from '@workspace/ui/components/button';
|
import { Button } from '@workspace/ui/components/button';
|
||||||
import { Switch } from '@workspace/ui/components/switch';
|
import { Switch } from '@workspace/ui/components/switch';
|
||||||
@ -23,6 +24,7 @@ export default function SubscribeTable() {
|
|||||||
const t = useTranslations('product');
|
const t = useTranslations('product');
|
||||||
const [loading, setLoading] = useState(false);
|
const [loading, setLoading] = useState(false);
|
||||||
const ref = useRef<ProTableActions>(null);
|
const ref = useRef<ProTableActions>(null);
|
||||||
|
const { fetchSubscribes } = useSubscribe();
|
||||||
return (
|
return (
|
||||||
<ProTable<API.SubscribeItem, { group_id: number; query: string }>
|
<ProTable<API.SubscribeItem, { group_id: number; query: string }>
|
||||||
action={ref}
|
action={ref}
|
||||||
@ -42,6 +44,7 @@ export default function SubscribeTable() {
|
|||||||
});
|
});
|
||||||
toast.success(t('createSuccess'));
|
toast.success(t('createSuccess'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchSubscribes();
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
|
|
||||||
return true;
|
return true;
|
||||||
@ -83,6 +86,7 @@ export default function SubscribeTable() {
|
|||||||
show: checked,
|
show: checked,
|
||||||
} as API.UpdateSubscribeRequest);
|
} as API.UpdateSubscribeRequest);
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchSubscribes();
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
@ -101,6 +105,7 @@ export default function SubscribeTable() {
|
|||||||
sell: checked,
|
sell: checked,
|
||||||
} as API.UpdateSubscribeRequest);
|
} as API.UpdateSubscribeRequest);
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchSubscribes();
|
||||||
}}
|
}}
|
||||||
/>
|
/>
|
||||||
);
|
);
|
||||||
@ -186,6 +191,7 @@ export default function SubscribeTable() {
|
|||||||
} as API.UpdateSubscribeRequest);
|
} as API.UpdateSubscribeRequest);
|
||||||
toast.success(t('updateSuccess'));
|
toast.success(t('updateSuccess'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchSubscribes();
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
return true;
|
return true;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@ -206,6 +212,7 @@ export default function SubscribeTable() {
|
|||||||
});
|
});
|
||||||
toast.success(t('deleteSuccess'));
|
toast.success(t('deleteSuccess'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchSubscribes();
|
||||||
}}
|
}}
|
||||||
cancelText={t('cancel')}
|
cancelText={t('cancel')}
|
||||||
confirmText={t('confirm')}
|
confirmText={t('confirm')}
|
||||||
@ -224,6 +231,7 @@ export default function SubscribeTable() {
|
|||||||
} as API.CreateSubscribeRequest);
|
} as API.CreateSubscribeRequest);
|
||||||
toast.success(t('copySuccess'));
|
toast.success(t('copySuccess'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchSubscribes();
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
return true;
|
return true;
|
||||||
} catch (error) {
|
} catch (error) {
|
||||||
@ -248,6 +256,7 @@ export default function SubscribeTable() {
|
|||||||
|
|
||||||
toast.success(t('deleteSuccess'));
|
toast.success(t('deleteSuccess'));
|
||||||
ref.current?.reset();
|
ref.current?.reset();
|
||||||
|
fetchSubscribes();
|
||||||
}}
|
}}
|
||||||
cancelText={t('cancel')}
|
cancelText={t('cancel')}
|
||||||
confirmText={t('confirm')}
|
confirmText={t('confirm')}
|
||||||
|
|||||||
@ -91,6 +91,7 @@ export function getProtocolDefaultConfig(proto: ProtocolType) {
|
|||||||
case 'mieru':
|
case 'mieru':
|
||||||
return {
|
return {
|
||||||
type: 'mieru',
|
type: 'mieru',
|
||||||
|
enable: false,
|
||||||
port: null,
|
port: null,
|
||||||
multiplex: 'none',
|
multiplex: 'none',
|
||||||
transport: 'tcp',
|
transport: 'tcp',
|
||||||
@ -98,6 +99,7 @@ export function getProtocolDefaultConfig(proto: ProtocolType) {
|
|||||||
case 'anytls':
|
case 'anytls':
|
||||||
return {
|
return {
|
||||||
type: 'anytls',
|
type: 'anytls',
|
||||||
|
enable: false,
|
||||||
port: null,
|
port: null,
|
||||||
security: 'tls',
|
security: 'tls',
|
||||||
padding_scheme: null,
|
padding_scheme: null,
|
||||||
|
|||||||
@ -18,7 +18,7 @@ export {
|
|||||||
} from './constants';
|
} from './constants';
|
||||||
|
|
||||||
// Re-export all types
|
// Re-export all types
|
||||||
export type { FieldConfig, ProtocolType, ServerFormValues } from './types';
|
export type { FieldConfig, ProtocolType } from './types';
|
||||||
|
|
||||||
// Re-export all schemas
|
// Re-export all schemas
|
||||||
export { formSchema, protocolApiScheme } from './schemas';
|
export { formSchema, protocolApiScheme } from './schemas';
|
||||||
|
|||||||
@ -1,6 +1,4 @@
|
|||||||
import { z } from 'zod';
|
|
||||||
import { protocols } from './constants';
|
import { protocols } from './constants';
|
||||||
import { formSchema } from './schemas';
|
|
||||||
|
|
||||||
export type FieldConfig = {
|
export type FieldConfig = {
|
||||||
name: string;
|
name: string;
|
||||||
@ -22,6 +20,4 @@ export type FieldConfig = {
|
|||||||
gridSpan?: 1 | 2;
|
gridSpan?: 1 | 2;
|
||||||
};
|
};
|
||||||
|
|
||||||
export type ServerFormValues = z.infer<typeof formSchema>;
|
|
||||||
|
|
||||||
export type ProtocolType = (typeof protocols)[number];
|
export type ProtocolType = (typeof protocols)[number];
|
||||||
|
|||||||
@ -1,5 +1,5 @@
|
|||||||
'use client';
|
'use client';
|
||||||
// Online users detail moved to separate component
|
|
||||||
import { ProTable, ProTableActions } from '@/components/pro-table';
|
import { ProTable, ProTableActions } from '@/components/pro-table';
|
||||||
import {
|
import {
|
||||||
createServer,
|
createServer,
|
||||||
@ -10,6 +10,8 @@ import {
|
|||||||
resetSortWithServer,
|
resetSortWithServer,
|
||||||
updateServer,
|
updateServer,
|
||||||
} from '@/services/admin/server';
|
} from '@/services/admin/server';
|
||||||
|
import { useNode } from '@/store/node';
|
||||||
|
import { useServer } from '@/store/server';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
import { useQuery } from '@tanstack/react-query';
|
||||||
import { Badge } from '@workspace/ui/components/badge';
|
import { Badge } from '@workspace/ui/components/badge';
|
||||||
import { Button } from '@workspace/ui/components/button';
|
import { Button } from '@workspace/ui/components/button';
|
||||||
@ -23,18 +25,6 @@ import OnlineUsersCell from './online-users-cell';
|
|||||||
import ServerConfig from './server-config';
|
import ServerConfig from './server-config';
|
||||||
import ServerForm from './server-form';
|
import ServerForm from './server-form';
|
||||||
|
|
||||||
type ProtocolName = 'shadowsocks' | 'vmess' | 'vless' | 'trojan' | 'hysteria2' | 'tuic' | 'anytls';
|
|
||||||
|
|
||||||
const PROTOCOL_COLORS: Record<ProtocolName, string> = {
|
|
||||||
shadowsocks: 'bg-green-500',
|
|
||||||
vmess: 'bg-rose-500',
|
|
||||||
vless: 'bg-blue-500',
|
|
||||||
trojan: 'bg-yellow-500',
|
|
||||||
hysteria2: 'bg-purple-500',
|
|
||||||
tuic: 'bg-cyan-500',
|
|
||||||
anytls: 'bg-gray-500',
|
|
||||||
};
|
|
||||||
|
|
||||||
function PctBar({ value }: { value: number }) {
|
function PctBar({ value }: { value: number }) {
|
||||||
const v = value.toFixed(2);
|
const v = value.toFixed(2);
|
||||||
return (
|
return (
|
||||||
@ -69,6 +59,8 @@ function RegionIpCell({
|
|||||||
|
|
||||||
export default function ServersPage() {
|
export default function ServersPage() {
|
||||||
const t = useTranslations('servers');
|
const t = useTranslations('servers');
|
||||||
|
const { isServerReferencedByNodes } = useNode();
|
||||||
|
const { fetchServers } = useServer();
|
||||||
|
|
||||||
const [loading, setLoading] = useState(false);
|
const [loading, setLoading] = useState(false);
|
||||||
const [migrating, setMigrating] = useState(false);
|
const [migrating, setMigrating] = useState(false);
|
||||||
@ -129,6 +121,7 @@ export default function ServersPage() {
|
|||||||
await createServer(values as unknown as API.CreateServerRequest);
|
await createServer(values as unknown as API.CreateServerRequest);
|
||||||
toast.success(t('created'));
|
toast.success(t('created'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchServers();
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
return true;
|
return true;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
@ -163,23 +156,16 @@ export default function ServersPage() {
|
|||||||
accessorKey: 'protocols',
|
accessorKey: 'protocols',
|
||||||
header: t('protocols'),
|
header: t('protocols'),
|
||||||
cell: ({ row }) => {
|
cell: ({ row }) => {
|
||||||
const list = (row.original.protocols || []) as API.Protocol[];
|
const list = row.original.protocols.filter(
|
||||||
if (!list.length) return t('noData');
|
(p) => p.enable !== false,
|
||||||
|
) as API.Protocol[];
|
||||||
|
if (!list.length) return '—';
|
||||||
return (
|
return (
|
||||||
<div className='flex flex-wrap gap-1'>
|
<div className='flex flex-wrap gap-1'>
|
||||||
{list.map((p, idx) => {
|
{list.map((p, idx) => {
|
||||||
const proto = ((p as any)?.type || '') as ProtocolName | '';
|
|
||||||
if (!proto) return null;
|
|
||||||
const color = PROTOCOL_COLORS[proto as ProtocolName];
|
|
||||||
const port = (p as any)?.port as number | undefined;
|
|
||||||
const label = `${proto}${port ? ` (${port})` : ''}`;
|
|
||||||
return (
|
return (
|
||||||
<Badge
|
<Badge key={idx} variant='outline'>
|
||||||
key={idx}
|
{p.type} ({p.port})
|
||||||
variant='outline'
|
|
||||||
className={cn('text-primary-foreground', color)}
|
|
||||||
>
|
|
||||||
{label}
|
|
||||||
</Badge>
|
</Badge>
|
||||||
);
|
);
|
||||||
})}
|
})}
|
||||||
@ -272,6 +258,7 @@ export default function ServersPage() {
|
|||||||
});
|
});
|
||||||
toast.success(t('updated'));
|
toast.success(t('updated'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchServers();
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
return true;
|
return true;
|
||||||
} catch (e) {
|
} catch (e) {
|
||||||
@ -282,13 +269,18 @@ export default function ServersPage() {
|
|||||||
/>,
|
/>,
|
||||||
<ConfirmButton
|
<ConfirmButton
|
||||||
key='delete'
|
key='delete'
|
||||||
trigger={<Button variant='destructive'>{t('delete')}</Button>}
|
trigger={
|
||||||
|
<Button variant='destructive' disabled={isServerReferencedByNodes(row.id)}>
|
||||||
|
{t('delete')}
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
title={t('confirmDeleteTitle')}
|
title={t('confirmDeleteTitle')}
|
||||||
description={t('confirmDeleteDesc')}
|
description={t('confirmDeleteDesc')}
|
||||||
onConfirm={async () => {
|
onConfirm={async () => {
|
||||||
await deleteServer({ id: row.id } as any);
|
await deleteServer({ id: row.id } as any);
|
||||||
toast.success(t('deleted'));
|
toast.success(t('deleted'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchServers();
|
||||||
}}
|
}}
|
||||||
cancelText={t('cancel')}
|
cancelText={t('cancel')}
|
||||||
confirmText={t('confirm')}
|
confirmText={t('confirm')}
|
||||||
@ -311,6 +303,7 @@ export default function ServersPage() {
|
|||||||
await createServer(body);
|
await createServer(body);
|
||||||
toast.success(t('copied'));
|
toast.success(t('copied'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchServers();
|
||||||
setLoading(false);
|
setLoading(false);
|
||||||
}}
|
}}
|
||||||
>
|
>
|
||||||
@ -318,16 +311,22 @@ export default function ServersPage() {
|
|||||||
</Button>,
|
</Button>,
|
||||||
],
|
],
|
||||||
batchRender(rows) {
|
batchRender(rows) {
|
||||||
|
const hasReferencedServers = rows.some((row) => isServerReferencedByNodes(row.id));
|
||||||
return [
|
return [
|
||||||
<ConfirmButton
|
<ConfirmButton
|
||||||
key='delete'
|
key='delete'
|
||||||
trigger={<Button variant='destructive'>{t('delete')}</Button>}
|
trigger={
|
||||||
|
<Button variant='destructive' disabled={hasReferencedServers}>
|
||||||
|
{t('delete')}
|
||||||
|
</Button>
|
||||||
|
}
|
||||||
title={t('confirmDeleteTitle')}
|
title={t('confirmDeleteTitle')}
|
||||||
description={t('confirmDeleteDesc')}
|
description={t('confirmDeleteDesc')}
|
||||||
onConfirm={async () => {
|
onConfirm={async () => {
|
||||||
await Promise.all(rows.map((r) => deleteServer({ id: r.id })));
|
await Promise.all(rows.map((r) => deleteServer({ id: r.id })));
|
||||||
toast.success(t('deleted'));
|
toast.success(t('deleted'));
|
||||||
ref.current?.refresh();
|
ref.current?.refresh();
|
||||||
|
fetchServers();
|
||||||
}}
|
}}
|
||||||
cancelText={t('cancel')}
|
cancelText={t('cancel')}
|
||||||
confirmText={t('confirm')}
|
confirmText={t('confirm')}
|
||||||
|
|||||||
@ -1,5 +1,6 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
|
import { useNode } from '@/store/node';
|
||||||
import { zodResolver } from '@hookform/resolvers/zod';
|
import { zodResolver } from '@hookform/resolvers/zod';
|
||||||
import {
|
import {
|
||||||
Accordion,
|
Accordion,
|
||||||
@ -48,7 +49,6 @@ import {
|
|||||||
getProtocolDefaultConfig,
|
getProtocolDefaultConfig,
|
||||||
PROTOCOL_FIELDS,
|
PROTOCOL_FIELDS,
|
||||||
protocols as PROTOCOLS,
|
protocols as PROTOCOLS,
|
||||||
ServerFormValues,
|
|
||||||
} from './form-schema';
|
} from './form-schema';
|
||||||
|
|
||||||
function DynamicField({
|
function DynamicField({
|
||||||
@ -321,14 +321,16 @@ export default function ServerForm(props: {
|
|||||||
trigger: string;
|
trigger: string;
|
||||||
title: string;
|
title: string;
|
||||||
loading?: boolean;
|
loading?: boolean;
|
||||||
initialValues?: Partial<ServerFormValues>;
|
initialValues?: Partial<API.Server>;
|
||||||
onSubmit: (values: ServerFormValues) => Promise<boolean> | boolean;
|
onSubmit: (values: Partial<API.Server>) => Promise<boolean> | boolean;
|
||||||
}) {
|
}) {
|
||||||
const { trigger, title, loading, initialValues, onSubmit } = props;
|
const { trigger, title, loading, initialValues, onSubmit } = props;
|
||||||
const t = useTranslations('servers');
|
const t = useTranslations('servers');
|
||||||
const [open, setOpen] = useState(false);
|
const [open, setOpen] = useState(false);
|
||||||
const [accordionValue, setAccordionValue] = useState<string>();
|
const [accordionValue, setAccordionValue] = useState<string>();
|
||||||
|
|
||||||
|
const { isProtocolUsedInNodes } = useNode();
|
||||||
|
|
||||||
const form = useForm({
|
const form = useForm({
|
||||||
resolver: zodResolver(formSchema),
|
resolver: zodResolver(formSchema),
|
||||||
defaultValues: {
|
defaultValues: {
|
||||||
@ -337,7 +339,7 @@ export default function ServerForm(props: {
|
|||||||
country: '',
|
country: '',
|
||||||
city: '',
|
city: '',
|
||||||
ratio: 1,
|
ratio: 1,
|
||||||
protocols: [],
|
protocols: [] as any[],
|
||||||
...initialValues,
|
...initialValues,
|
||||||
},
|
},
|
||||||
});
|
});
|
||||||
@ -515,6 +517,7 @@ export default function ServerForm(props: {
|
|||||||
PROTOCOLS.findIndex((t) => t === type),
|
PROTOCOLS.findIndex((t) => t === type),
|
||||||
);
|
);
|
||||||
const current = (protocolsValues[i] || {}) as Record<string, any>;
|
const current = (protocolsValues[i] || {}) as Record<string, any>;
|
||||||
|
const isEnabled = current?.enable !== false;
|
||||||
const fields = PROTOCOL_FIELDS[type] || [];
|
const fields = PROTOCOL_FIELDS[type] || [];
|
||||||
return (
|
return (
|
||||||
<AccordionItem key={type} value={type} className='mb-2 rounded-lg border'>
|
<AccordionItem key={type} value={type} className='mb-2 rounded-lg border'>
|
||||||
@ -539,16 +542,20 @@ export default function ServerForm(props: {
|
|||||||
<span
|
<span
|
||||||
className={cn(
|
className={cn(
|
||||||
'text-xs',
|
'text-xs',
|
||||||
current.enable ? 'text-green-500' : 'text-muted-foreground',
|
isEnabled ? 'text-green-500' : 'text-muted-foreground',
|
||||||
)}
|
)}
|
||||||
>
|
>
|
||||||
{current.enable ? t('enabled') : t('disabled')}
|
{isEnabled ? t('enabled') : t('disabled')}
|
||||||
</span>
|
</span>
|
||||||
</div>
|
</div>
|
||||||
</div>
|
</div>
|
||||||
<Switch
|
<Switch
|
||||||
className='mr-2'
|
className='mr-2'
|
||||||
checked={!!current.enable}
|
checked={!!isEnabled}
|
||||||
|
disabled={Boolean(
|
||||||
|
initialValues?.id &&
|
||||||
|
isProtocolUsedInNodes(initialValues?.id || 0, type),
|
||||||
|
)}
|
||||||
onCheckedChange={(checked) => {
|
onCheckedChange={(checked) => {
|
||||||
form.setValue(`protocols.${i}.enable`, checked);
|
form.setValue(`protocols.${i}.enable`, checked);
|
||||||
}}
|
}}
|
||||||
|
|||||||
@ -1,7 +1,7 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { getSubscribeList } from '@/services/admin/subscribe';
|
|
||||||
import { getRegisterConfig, updateRegisterConfig } from '@/services/admin/system';
|
import { getRegisterConfig, updateRegisterConfig } from '@/services/admin/system';
|
||||||
|
import { useSubscribe } from '@/store/subscribe';
|
||||||
import { zodResolver } from '@hookform/resolvers/zod';
|
import { zodResolver } from '@hookform/resolvers/zod';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
import { useQuery } from '@tanstack/react-query';
|
||||||
import { Button } from '@workspace/ui/components/button';
|
import { Button } from '@workspace/ui/components/button';
|
||||||
@ -61,17 +61,7 @@ export default function RegisterConfig() {
|
|||||||
enabled: open,
|
enabled: open,
|
||||||
});
|
});
|
||||||
|
|
||||||
const { data: subscribe } = useQuery({
|
const { subscribes } = useSubscribe();
|
||||||
queryKey: ['getSubscribeList', 'all'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await getSubscribeList({
|
|
||||||
page: 1,
|
|
||||||
size: 9999,
|
|
||||||
});
|
|
||||||
return data.data?.list as API.Subscribe[];
|
|
||||||
},
|
|
||||||
enabled: open,
|
|
||||||
});
|
|
||||||
|
|
||||||
const form = useForm<RegisterFormData>({
|
const form = useForm<RegisterFormData>({
|
||||||
resolver: zodResolver(registerSchema),
|
resolver: zodResolver(registerSchema),
|
||||||
@ -268,12 +258,10 @@ export default function RegisterConfig() {
|
|||||||
field.onChange(value);
|
field.onChange(value);
|
||||||
}
|
}
|
||||||
}}
|
}}
|
||||||
options={
|
options={subscribes?.map((item) => ({
|
||||||
subscribe?.map((item) => ({
|
label: item.name!,
|
||||||
label: item.name,
|
value: item.id!,
|
||||||
value: item.id,
|
}))}
|
||||||
})) || []
|
|
||||||
}
|
|
||||||
className='bg-secondary w-32 rounded-r-none'
|
className='bg-secondary w-32 rounded-r-none'
|
||||||
/>
|
/>
|
||||||
)}
|
)}
|
||||||
|
|||||||
@ -2,7 +2,6 @@
|
|||||||
|
|
||||||
import { Display } from '@/components/display';
|
import { Display } from '@/components/display';
|
||||||
import { ProTable, ProTableActions } from '@/components/pro-table';
|
import { ProTable, ProTableActions } from '@/components/pro-table';
|
||||||
import { getSubscribeList } from '@/services/admin/subscribe';
|
|
||||||
import {
|
import {
|
||||||
createUser,
|
createUser,
|
||||||
deleteUser,
|
deleteUser,
|
||||||
@ -10,6 +9,7 @@ import {
|
|||||||
getUserList,
|
getUserList,
|
||||||
updateUserBasicInfo,
|
updateUserBasicInfo,
|
||||||
} from '@/services/admin/user';
|
} from '@/services/admin/user';
|
||||||
|
import { useSubscribe } from '@/store/subscribe';
|
||||||
import { formatDate } from '@/utils/common';
|
import { formatDate } from '@/utils/common';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
import { useQuery } from '@tanstack/react-query';
|
||||||
import { Badge } from '@workspace/ui/components/badge';
|
import { Badge } from '@workspace/ui/components/badge';
|
||||||
@ -49,16 +49,7 @@ export default function Page() {
|
|||||||
const ref = useRef<ProTableActions>(null);
|
const ref = useRef<ProTableActions>(null);
|
||||||
const sp = useSearchParams();
|
const sp = useSearchParams();
|
||||||
|
|
||||||
const { data: subscribeList } = useQuery({
|
const { subscribes } = useSubscribe();
|
||||||
queryKey: ['getSubscribeList', 'all'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await getSubscribeList({
|
|
||||||
page: 1,
|
|
||||||
size: 9999,
|
|
||||||
});
|
|
||||||
return data.data?.list as API.SubscribeGroup[];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
const initialFilters = {
|
const initialFilters = {
|
||||||
search: sp.get('search') || undefined,
|
search: sp.get('search') || undefined,
|
||||||
@ -194,9 +185,9 @@ export default function Page() {
|
|||||||
{
|
{
|
||||||
key: 'subscribe_id',
|
key: 'subscribe_id',
|
||||||
placeholder: t('subscription'),
|
placeholder: t('subscription'),
|
||||||
options: subscribeList?.map((item) => ({
|
options: subscribes?.map((item) => ({
|
||||||
label: item.name,
|
label: item.name!,
|
||||||
value: String(item.id),
|
value: String(item.id!),
|
||||||
})),
|
})),
|
||||||
},
|
},
|
||||||
{
|
{
|
||||||
|
|||||||
@ -1,8 +1,7 @@
|
|||||||
'use client';
|
'use client';
|
||||||
|
|
||||||
import { getSubscribeList } from '@/services/admin/subscribe';
|
import { useSubscribe } from '@/store/subscribe';
|
||||||
import { zodResolver } from '@hookform/resolvers/zod';
|
import { zodResolver } from '@hookform/resolvers/zod';
|
||||||
import { useQuery } from '@tanstack/react-query';
|
|
||||||
import { Button } from '@workspace/ui/components/button';
|
import { Button } from '@workspace/ui/components/button';
|
||||||
import {
|
import {
|
||||||
Form,
|
Form,
|
||||||
@ -74,16 +73,7 @@ export function SubscriptionForm({ trigger, title, loading, initialData, onSubmi
|
|||||||
}
|
}
|
||||||
};
|
};
|
||||||
|
|
||||||
const { data: subscribe } = useQuery({
|
const { subscribes } = useSubscribe();
|
||||||
queryKey: ['getSubscribeList', 'all'],
|
|
||||||
queryFn: async () => {
|
|
||||||
const { data } = await getSubscribeList({
|
|
||||||
page: 1,
|
|
||||||
size: 9999,
|
|
||||||
});
|
|
||||||
return data.data?.list as API.Subscribe[];
|
|
||||||
},
|
|
||||||
});
|
|
||||||
|
|
||||||
return (
|
return (
|
||||||
<Sheet open={open} onOpenChange={setOpen}>
|
<Sheet open={open} onOpenChange={setOpen}>
|
||||||
@ -117,9 +107,9 @@ export function SubscriptionForm({ trigger, title, loading, initialData, onSubmi
|
|||||||
onChange={(value) => {
|
onChange={(value) => {
|
||||||
form.setValue(field.name, value);
|
form.setValue(field.name, value);
|
||||||
}}
|
}}
|
||||||
options={subscribe?.map((item: API.Subscribe) => ({
|
options={subscribes?.map((item) => ({
|
||||||
value: item.id,
|
value: item.id!,
|
||||||
label: item.name,
|
label: item.name!,
|
||||||
}))}
|
}))}
|
||||||
/>
|
/>
|
||||||
</FormControl>
|
</FormControl>
|
||||||
|
|||||||
@ -1,5 +1,5 @@
|
|||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
|
|
||||||
// API 更新时间:
|
// API 更新时间:
|
||||||
// API 唯一标识:
|
// API 唯一标识:
|
||||||
import * as ads from './ads';
|
import * as ads from './ads';
|
||||||
|
|||||||
@ -1,5 +1,5 @@
|
|||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
|
|
||||||
// API 更新时间:
|
// API 更新时间:
|
||||||
// API 唯一标识:
|
// API 唯一标识:
|
||||||
import * as auth from './auth';
|
import * as auth from './auth';
|
||||||
|
|||||||
113
apps/admin/store/node.ts
Normal file
113
apps/admin/store/node.ts
Normal file
@ -0,0 +1,113 @@
|
|||||||
|
import { filterNodeList, queryNodeTag } from '@/services/admin/server';
|
||||||
|
import { create } from 'zustand';
|
||||||
|
|
||||||
|
interface NodeState {
|
||||||
|
// Data
|
||||||
|
nodes: API.Node[];
|
||||||
|
tags: string[];
|
||||||
|
|
||||||
|
// Actions
|
||||||
|
fetchNodes: () => Promise<void>;
|
||||||
|
fetchTags: () => Promise<void>;
|
||||||
|
|
||||||
|
// Getters
|
||||||
|
getNodeById: (nodeId: number) => API.Node | undefined;
|
||||||
|
isProtocolUsedInNodes: (serverId: number, protocolType: string) => boolean;
|
||||||
|
isServerReferencedByNodes: (serverId: number) => boolean;
|
||||||
|
getNodesByTag: (tag: string) => API.Node[];
|
||||||
|
getNodesWithoutTags: () => API.Node[];
|
||||||
|
getNodeTags: () => string[];
|
||||||
|
getAllAvailableTags: () => string[];
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useNodeStore = create<NodeState>((set, get) => ({
|
||||||
|
// Initial state
|
||||||
|
nodes: [],
|
||||||
|
tags: [],
|
||||||
|
|
||||||
|
// Actions
|
||||||
|
fetchNodes: async () => {
|
||||||
|
try {
|
||||||
|
const { data } = await filterNodeList({ page: 1, size: 999999999 });
|
||||||
|
set({ nodes: data?.data?.list || [] });
|
||||||
|
} catch (error) {
|
||||||
|
// Handle error silently
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
fetchTags: async () => {
|
||||||
|
try {
|
||||||
|
const { data } = await queryNodeTag();
|
||||||
|
set({ tags: data?.data?.tags || [] });
|
||||||
|
} catch (error) {
|
||||||
|
// Handle error silently
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// Getters
|
||||||
|
getNodeById: (nodeId: number) => {
|
||||||
|
return get().nodes.find((n) => n.id === nodeId);
|
||||||
|
},
|
||||||
|
|
||||||
|
isProtocolUsedInNodes: (serverId: number, protocolType: string) => {
|
||||||
|
return get().nodes.some(
|
||||||
|
(node) => node.server_id === serverId && node.protocol === protocolType,
|
||||||
|
);
|
||||||
|
},
|
||||||
|
|
||||||
|
isServerReferencedByNodes: (serverId: number) => {
|
||||||
|
return get().nodes.some((node) => node.server_id === serverId);
|
||||||
|
},
|
||||||
|
|
||||||
|
getNodesByTag: (tag: string) => {
|
||||||
|
return get().nodes.filter((node) => (node.tags || []).includes(tag));
|
||||||
|
},
|
||||||
|
|
||||||
|
getNodesWithoutTags: () => {
|
||||||
|
return get().nodes.filter((node) => (node.tags || []).length === 0);
|
||||||
|
},
|
||||||
|
|
||||||
|
getNodeTags: () => {
|
||||||
|
return Array.from(
|
||||||
|
new Set(
|
||||||
|
get()
|
||||||
|
.nodes.flatMap((node) => (Array.isArray(node.tags) ? node.tags : []))
|
||||||
|
.filter(Boolean),
|
||||||
|
),
|
||||||
|
) as string[];
|
||||||
|
},
|
||||||
|
|
||||||
|
getAllAvailableTags: () => {
|
||||||
|
const nodeExtractedTags = get().getNodeTags();
|
||||||
|
const allApiTags = get().tags;
|
||||||
|
return Array.from(new Set([...allApiTags, ...nodeExtractedTags])).filter(Boolean);
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
export const useNode = () => {
|
||||||
|
const store = useNodeStore();
|
||||||
|
|
||||||
|
// Auto-fetch nodes and tags
|
||||||
|
if (store.nodes.length === 0) {
|
||||||
|
store.fetchNodes();
|
||||||
|
}
|
||||||
|
if (store.tags.length === 0) {
|
||||||
|
store.fetchTags();
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
nodes: store.nodes,
|
||||||
|
tags: store.tags,
|
||||||
|
fetchNodes: store.fetchNodes,
|
||||||
|
fetchTags: store.fetchTags,
|
||||||
|
getNodeById: store.getNodeById,
|
||||||
|
isProtocolUsedInNodes: store.isProtocolUsedInNodes,
|
||||||
|
isServerReferencedByNodes: store.isServerReferencedByNodes,
|
||||||
|
getNodesByTag: store.getNodesByTag,
|
||||||
|
getNodesWithoutTags: store.getNodesWithoutTags,
|
||||||
|
getNodeTags: store.getNodeTags,
|
||||||
|
getAllAvailableTags: store.getAllAvailableTags,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export default useNodeStore;
|
||||||
94
apps/admin/store/server.ts
Normal file
94
apps/admin/store/server.ts
Normal file
@ -0,0 +1,94 @@
|
|||||||
|
import { filterServerList } from '@/services/admin/server';
|
||||||
|
import { create } from 'zustand';
|
||||||
|
|
||||||
|
interface ServerState {
|
||||||
|
// Data
|
||||||
|
servers: API.Server[];
|
||||||
|
|
||||||
|
// Actions
|
||||||
|
fetchServers: () => Promise<void>;
|
||||||
|
|
||||||
|
// Getters
|
||||||
|
getServerById: (serverId: number) => API.Server | undefined;
|
||||||
|
getServerName: (serverId?: number) => string;
|
||||||
|
getServerAddress: (serverId?: number) => string;
|
||||||
|
getServerEnabledProtocols: (serverId: number) => API.Protocol[];
|
||||||
|
getProtocolPort: (serverId?: number, protocol?: string) => string;
|
||||||
|
getAvailableProtocols: (serverId?: number) => Array<{ protocol: string; port: number }>;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useServerStore = create<ServerState>((set, get) => ({
|
||||||
|
// Initial state
|
||||||
|
servers: [],
|
||||||
|
|
||||||
|
// Actions
|
||||||
|
fetchServers: async () => {
|
||||||
|
try {
|
||||||
|
const { data } = await filterServerList({ page: 1, size: 999999999 });
|
||||||
|
set({ servers: data?.data?.list || [] });
|
||||||
|
} catch (error) {
|
||||||
|
// Handle error silently
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// Getters
|
||||||
|
getServerById: (serverId: number) => {
|
||||||
|
return get().servers.find((s) => s.id === serverId);
|
||||||
|
},
|
||||||
|
|
||||||
|
getServerName: (serverId?: number) => {
|
||||||
|
if (!serverId) return '—';
|
||||||
|
const server = get().servers.find((s) => s.id === serverId);
|
||||||
|
return server?.name ?? `#${serverId}`;
|
||||||
|
},
|
||||||
|
|
||||||
|
getServerAddress: (serverId?: number) => {
|
||||||
|
if (!serverId) return '—';
|
||||||
|
const server = get().servers.find((s) => s.id === serverId);
|
||||||
|
return server?.address ?? '—';
|
||||||
|
},
|
||||||
|
|
||||||
|
getServerEnabledProtocols: (serverId: number) => {
|
||||||
|
const server = get().servers.find((s) => s.id === serverId);
|
||||||
|
return server?.protocols?.filter((p) => p.enable !== false) || [];
|
||||||
|
},
|
||||||
|
|
||||||
|
getProtocolPort: (serverId?: number, protocol?: string) => {
|
||||||
|
if (!serverId || !protocol) return '—';
|
||||||
|
const enabledProtocols = get().getServerEnabledProtocols(serverId);
|
||||||
|
const protocolConfig = enabledProtocols.find((p) => p.type === protocol);
|
||||||
|
return protocolConfig?.port ? String(protocolConfig.port) : '—';
|
||||||
|
},
|
||||||
|
|
||||||
|
getAvailableProtocols: (serverId?: number) => {
|
||||||
|
if (!serverId) return [];
|
||||||
|
return get()
|
||||||
|
.getServerEnabledProtocols(serverId)
|
||||||
|
.map((p) => ({
|
||||||
|
protocol: p.type,
|
||||||
|
port: p.port,
|
||||||
|
}));
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
export const useServer = () => {
|
||||||
|
const store = useServerStore();
|
||||||
|
|
||||||
|
// Auto-fetch servers
|
||||||
|
if (store.servers.length === 0) {
|
||||||
|
store.fetchServers();
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
servers: store.servers,
|
||||||
|
fetchServers: store.fetchServers,
|
||||||
|
getServerById: store.getServerById,
|
||||||
|
getServerName: store.getServerName,
|
||||||
|
getServerAddress: store.getServerAddress,
|
||||||
|
getServerEnabledProtocols: store.getServerEnabledProtocols,
|
||||||
|
getProtocolPort: store.getProtocolPort,
|
||||||
|
getAvailableProtocols: store.getAvailableProtocols,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export default useServerStore;
|
||||||
58
apps/admin/store/subscribe.ts
Normal file
58
apps/admin/store/subscribe.ts
Normal file
@ -0,0 +1,58 @@
|
|||||||
|
import { getSubscribeList } from '@/services/admin/subscribe';
|
||||||
|
import { create } from 'zustand';
|
||||||
|
|
||||||
|
interface SubscribeState {
|
||||||
|
// Data
|
||||||
|
subscribes: API.SubscribeItem[];
|
||||||
|
|
||||||
|
// Actions
|
||||||
|
fetchSubscribes: () => Promise<void>;
|
||||||
|
|
||||||
|
// Getters
|
||||||
|
getSubscribeName: (subscribeId?: number) => string;
|
||||||
|
getSubscribeById: (subscribeId: number) => API.SubscribeItem | undefined;
|
||||||
|
}
|
||||||
|
|
||||||
|
export const useSubscribeStore = create<SubscribeState>((set, get) => ({
|
||||||
|
// Initial state
|
||||||
|
subscribes: [],
|
||||||
|
|
||||||
|
// Actions
|
||||||
|
fetchSubscribes: async () => {
|
||||||
|
try {
|
||||||
|
const { data } = await getSubscribeList({ page: 1, size: 999999999 });
|
||||||
|
set({ subscribes: data?.data?.list || [] });
|
||||||
|
} catch (error) {
|
||||||
|
// Handle error silently
|
||||||
|
}
|
||||||
|
},
|
||||||
|
|
||||||
|
// Getters
|
||||||
|
getSubscribeName: (subscribeId?: number) => {
|
||||||
|
if (!subscribeId) return 'Unknown';
|
||||||
|
const subscribe = get().subscribes.find((s) => s.id === subscribeId);
|
||||||
|
return subscribe?.name ?? `Subscribe ${subscribeId}`;
|
||||||
|
},
|
||||||
|
|
||||||
|
getSubscribeById: (subscribeId: number) => {
|
||||||
|
return get().subscribes.find((s) => s.id === subscribeId);
|
||||||
|
},
|
||||||
|
}));
|
||||||
|
|
||||||
|
export const useSubscribe = () => {
|
||||||
|
const store = useSubscribeStore();
|
||||||
|
|
||||||
|
// Auto-fetch subscribes
|
||||||
|
if (store.subscribes.length === 0) {
|
||||||
|
store.fetchSubscribes();
|
||||||
|
}
|
||||||
|
|
||||||
|
return {
|
||||||
|
subscribes: store.subscribes,
|
||||||
|
fetchSubscribes: store.fetchSubscribes,
|
||||||
|
getSubscribeName: store.getSubscribeName,
|
||||||
|
getSubscribeById: store.getSubscribeById,
|
||||||
|
};
|
||||||
|
};
|
||||||
|
|
||||||
|
export default useSubscribeStore;
|
||||||
@ -1,5 +1,5 @@
|
|||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
|
|
||||||
// API 更新时间:
|
// API 更新时间:
|
||||||
// API 唯一标识:
|
// API 唯一标识:
|
||||||
import * as auth from './auth';
|
import * as auth from './auth';
|
||||||
|
|||||||
@ -1,5 +1,5 @@
|
|||||||
// @ts-ignore
|
// @ts-ignore
|
||||||
|
|
||||||
// API 更新时间:
|
// API 更新时间:
|
||||||
// API 唯一标识:
|
// API 唯一标识:
|
||||||
import * as announcement from './announcement';
|
import * as announcement from './announcement';
|
||||||
|
|||||||
Loading…
x
Reference in New Issue
Block a user