mirror of
https://github.com/MHSanaei/3x-ui.git
synced 2026-06-05 20:54:14 +00:00
Some checks are pending
CI / go-test (push) Waiting to run
CI / govulncheck (push) Waiting to run
CI / frontend (push) Waiting to run
CodeQL Advanced / Analyze (go) (push) Waiting to run
CodeQL Advanced / Analyze (actions) (push) Waiting to run
CodeQL Advanced / Analyze (javascript-typescript) (push) Waiting to run
Release 3X-UI / build (386) (push) Waiting to run
Release 3X-UI / build (amd64) (push) Waiting to run
Release 3X-UI / build (arm64) (push) Waiting to run
Release 3X-UI / build (armv5) (push) Waiting to run
Release 3X-UI / build (armv6) (push) Waiting to run
Release 3X-UI / build (armv7) (push) Waiting to run
Release 3X-UI / build (s390x) (push) Waiting to run
Release 3X-UI / Build for Windows (push) Waiting to run
Replace the HTTP-mode outbound test that spun up a SOCKS inbound and ran an httptrace'd request from the Go client with a probe-only xray config: burstObservatory probes the target outbound directly and the result is read from xray's /debug/vars metrics endpoint. The probe lives inside xray, so the measured delay and failure reasons reflect what xray itself sees over the real proxy chain. Drops the DNS/Connect/TLS/TTFB breakdown (and statusCode) since the observatory snapshot only exposes total delay; the frontend popover is updated accordingly.
124 lines
3.7 KiB
TypeScript
124 lines
3.7 KiB
TypeScript
import { z } from 'zod';
|
|
import { DnsObjectSchema } from './dns';
|
|
import {
|
|
BalancerObjectSchema,
|
|
BalancerStrategySettingsSchema,
|
|
BalancerStrategyTypeSchema,
|
|
RuleObjectSchema,
|
|
} from './routing';
|
|
|
|
export const XraySettingsValueSchema = z.object({
|
|
inbounds: z.array(z.unknown()).optional(),
|
|
outbounds: z
|
|
.array(
|
|
z.object({
|
|
tag: z.string().optional(),
|
|
protocol: z.string().optional(),
|
|
settings: z.unknown().optional(),
|
|
streamSettings: z.unknown().optional(),
|
|
}).loose(),
|
|
)
|
|
.optional(),
|
|
routing: z.object({
|
|
rules: z.array(RuleObjectSchema).optional(),
|
|
balancers: z.array(BalancerObjectSchema).optional(),
|
|
domainStrategy: z.string().optional(),
|
|
}).loose().optional(),
|
|
dns: DnsObjectSchema.optional(),
|
|
log: z.record(z.string(), z.unknown()).optional(),
|
|
policy: z.object({
|
|
system: z.record(z.string(), z.boolean()).optional(),
|
|
}).loose().optional(),
|
|
observatory: z.unknown().optional(),
|
|
burstObservatory: z.unknown().optional(),
|
|
fakedns: z.unknown().optional(),
|
|
}).loose();
|
|
|
|
export const XrayConfigPayloadSchema = z.object({
|
|
xraySetting: XraySettingsValueSchema,
|
|
inboundTags: z.array(z.string()).optional(),
|
|
clientReverseTags: z.array(z.string()).optional(),
|
|
outboundTestUrl: z.string().optional(),
|
|
}).loose();
|
|
|
|
export const OutboundTrafficRowSchema = z.object({
|
|
tag: z.string(),
|
|
up: z.number(),
|
|
down: z.number(),
|
|
});
|
|
|
|
export const OutboundTrafficListSchema = z.array(OutboundTrafficRowSchema);
|
|
|
|
export const OutboundTestResultSchema = z.object({
|
|
success: z.boolean(),
|
|
delay: z.number().optional(),
|
|
error: z.string().optional(),
|
|
mode: z.string().optional(),
|
|
endpoints: z
|
|
.array(
|
|
z.object({
|
|
address: z.string(),
|
|
delay: z.number().optional(),
|
|
success: z.boolean(),
|
|
error: z.string().optional(),
|
|
}).loose(),
|
|
)
|
|
.optional(),
|
|
}).loose();
|
|
|
|
export const CustomGeoFormSchema = z.object({
|
|
type: z.enum(['geosite', 'geoip']),
|
|
alias: z.string().regex(/^[a-z0-9_-]+$/, 'pages.index.customGeoValidationAlias'),
|
|
url: z
|
|
.string()
|
|
.trim()
|
|
.refine(
|
|
(u) => {
|
|
if (!/^https?:\/\//i.test(u)) return false;
|
|
try {
|
|
const parsed = new URL(u);
|
|
return parsed.protocol === 'http:' || parsed.protocol === 'https:';
|
|
} catch {
|
|
return false;
|
|
}
|
|
},
|
|
{ message: 'pages.index.customGeoValidationUrl' },
|
|
),
|
|
});
|
|
|
|
export const RuleFormSchema = z.object({
|
|
domain: z.string(),
|
|
ip: z.string(),
|
|
port: z.string(),
|
|
sourcePort: z.string(),
|
|
vlessRoute: z.string(),
|
|
network: z.string(),
|
|
sourceIP: z.string(),
|
|
user: z.string(),
|
|
inboundTag: z.array(z.string()),
|
|
protocol: z.array(z.string()),
|
|
attrs: z.array(z.tuple([z.string(), z.string()])),
|
|
outboundTag: z.string(),
|
|
balancerTag: z.string(),
|
|
});
|
|
|
|
export const BalancerFormSchema = z.object({
|
|
tag: z.string().trim().min(1, 'pages.xray.balancerTagRequired'),
|
|
strategy: BalancerStrategyTypeSchema.default('random'),
|
|
selector: z.array(z.string()).min(1, 'pages.xray.balancerSelectorRequired'),
|
|
fallbackTag: z.string().default(''),
|
|
settings: BalancerStrategySettingsSchema.optional(),
|
|
});
|
|
|
|
export const OutboundTagSchema = z
|
|
.string()
|
|
.trim()
|
|
.min(1, 'pages.xray.outboundTagRequired');
|
|
|
|
export type BalancerFormValues = z.infer<typeof BalancerFormSchema>;
|
|
export type RuleFormValues = z.infer<typeof RuleFormSchema>;
|
|
export type CustomGeoFormValues = z.infer<typeof CustomGeoFormSchema>;
|
|
export type XraySettingsValue = z.infer<typeof XraySettingsValueSchema>;
|
|
export type XrayConfigPayload = z.infer<typeof XrayConfigPayloadSchema>;
|
|
export type OutboundTrafficRow = z.infer<typeof OutboundTrafficRowSchema>;
|
|
export type OutboundTestResult = z.infer<typeof OutboundTestResultSchema>;
|