mirror of
https://github.com/MHSanaei/3x-ui.git
synced 2025-11-29 10:52:54 +00:00
2581 lines
78 KiB
JavaScript
2581 lines
78 KiB
JavaScript
const Protocols = {
|
||
VMESS: 'vmess',
|
||
VLESS: 'vless',
|
||
TROJAN: 'trojan',
|
||
SHADOWSOCKS: 'shadowsocks',
|
||
TUNNEL: 'tunnel',
|
||
SOCKS: 'socks',
|
||
HTTP: 'http',
|
||
WIREGUARD: 'wireguard',
|
||
};
|
||
|
||
const SSMethods = {
|
||
AES_256_GCM: 'aes-256-gcm',
|
||
CHACHA20_POLY1305: 'chacha20-poly1305',
|
||
CHACHA20_IETF_POLY1305: 'chacha20-ietf-poly1305',
|
||
XCHACHA20_IETF_POLY1305: 'xchacha20-ietf-poly1305',
|
||
BLAKE3_AES_128_GCM: '2022-blake3-aes-128-gcm',
|
||
BLAKE3_AES_256_GCM: '2022-blake3-aes-256-gcm',
|
||
BLAKE3_CHACHA20_POLY1305: '2022-blake3-chacha20-poly1305',
|
||
};
|
||
|
||
const TLS_FLOW_CONTROL = {
|
||
VISION: "xtls-rprx-vision",
|
||
VISION_UDP443: "xtls-rprx-vision-udp443",
|
||
};
|
||
|
||
const TLS_VERSION_OPTION = {
|
||
TLS10: "1.0",
|
||
TLS11: "1.1",
|
||
TLS12: "1.2",
|
||
TLS13: "1.3",
|
||
};
|
||
|
||
const TLS_CIPHER_OPTION = {
|
||
AES_128_GCM: "TLS_AES_128_GCM_SHA256",
|
||
AES_256_GCM: "TLS_AES_256_GCM_SHA384",
|
||
CHACHA20_POLY1305: "TLS_CHACHA20_POLY1305_SHA256",
|
||
ECDHE_ECDSA_AES_128_CBC: "TLS_ECDHE_ECDSA_WITH_AES_128_CBC_SHA",
|
||
ECDHE_ECDSA_AES_256_CBC: "TLS_ECDHE_ECDSA_WITH_AES_256_CBC_SHA",
|
||
ECDHE_RSA_AES_128_CBC: "TLS_ECDHE_RSA_WITH_AES_128_CBC_SHA",
|
||
ECDHE_RSA_AES_256_CBC: "TLS_ECDHE_RSA_WITH_AES_256_CBC_SHA",
|
||
ECDHE_ECDSA_AES_128_GCM: "TLS_ECDHE_ECDSA_WITH_AES_128_GCM_SHA256",
|
||
ECDHE_ECDSA_AES_256_GCM: "TLS_ECDHE_ECDSA_WITH_AES_256_GCM_SHA384",
|
||
ECDHE_RSA_AES_128_GCM: "TLS_ECDHE_RSA_WITH_AES_128_GCM_SHA256",
|
||
ECDHE_RSA_AES_256_GCM: "TLS_ECDHE_RSA_WITH_AES_256_GCM_SHA384",
|
||
ECDHE_ECDSA_CHACHA20_POLY1305: "TLS_ECDHE_ECDSA_WITH_CHACHA20_POLY1305_SHA256",
|
||
ECDHE_RSA_CHACHA20_POLY1305: "TLS_ECDHE_RSA_WITH_CHACHA20_POLY1305_SHA256",
|
||
};
|
||
|
||
const UTLS_FINGERPRINT = {
|
||
UTLS_CHROME: "chrome",
|
||
UTLS_FIREFOX: "firefox",
|
||
UTLS_SAFARI: "safari",
|
||
UTLS_IOS: "ios",
|
||
UTLS_android: "android",
|
||
UTLS_EDGE: "edge",
|
||
UTLS_360: "360",
|
||
UTLS_QQ: "qq",
|
||
UTLS_RANDOM: "random",
|
||
UTLS_RANDOMIZED: "randomized",
|
||
UTLS_RONDOMIZEDNOALPN: "randomizednoalpn",
|
||
UTLS_UNSAFE: "unsafe",
|
||
};
|
||
|
||
const ALPN_OPTION = {
|
||
H3: "h3",
|
||
H2: "h2",
|
||
HTTP1: "http/1.1",
|
||
};
|
||
|
||
const SNIFFING_OPTION = {
|
||
HTTP: "http",
|
||
TLS: "tls",
|
||
QUIC: "quic",
|
||
FAKEDNS: "fakedns"
|
||
};
|
||
|
||
const USAGE_OPTION = {
|
||
ENCIPHERMENT: "encipherment",
|
||
VERIFY: "verify",
|
||
ISSUE: "issue",
|
||
};
|
||
|
||
const DOMAIN_STRATEGY_OPTION = {
|
||
AS_IS: "AsIs",
|
||
USE_IP: "UseIP",
|
||
USE_IPV6V4: "UseIPv6v4",
|
||
USE_IPV6: "UseIPv6",
|
||
USE_IPV4V6: "UseIPv4v6",
|
||
USE_IPV4: "UseIPv4",
|
||
FORCE_IP: "ForceIP",
|
||
FORCE_IPV6V4: "ForceIPv6v4",
|
||
FORCE_IPV6: "ForceIPv6",
|
||
FORCE_IPV4V6: "ForceIPv4v6",
|
||
FORCE_IPV4: "ForceIPv4",
|
||
};
|
||
|
||
const TCP_CONGESTION_OPTION = {
|
||
BBR: "bbr",
|
||
CUBIC: "cubic",
|
||
RENO: "reno",
|
||
};
|
||
|
||
const USERS_SECURITY = {
|
||
AES_128_GCM: "aes-128-gcm",
|
||
CHACHA20_POLY1305: "chacha20-poly1305",
|
||
AUTO: "auto",
|
||
NONE: "none",
|
||
ZERO: "zero",
|
||
};
|
||
|
||
const MODE_OPTION = {
|
||
AUTO: "auto",
|
||
PACKET_UP: "packet-up",
|
||
STREAM_UP: "stream-up",
|
||
STREAM_ONE: "stream-one",
|
||
};
|
||
|
||
Object.freeze(Protocols);
|
||
Object.freeze(SSMethods);
|
||
Object.freeze(TLS_FLOW_CONTROL);
|
||
Object.freeze(TLS_VERSION_OPTION);
|
||
Object.freeze(TLS_CIPHER_OPTION);
|
||
Object.freeze(UTLS_FINGERPRINT);
|
||
Object.freeze(ALPN_OPTION);
|
||
Object.freeze(SNIFFING_OPTION);
|
||
Object.freeze(USAGE_OPTION);
|
||
Object.freeze(DOMAIN_STRATEGY_OPTION);
|
||
Object.freeze(TCP_CONGESTION_OPTION);
|
||
Object.freeze(USERS_SECURITY);
|
||
Object.freeze(MODE_OPTION);
|
||
|
||
class XrayCommonClass {
|
||
|
||
static toJsonArray(arr) {
|
||
return arr.map(obj => obj.toJson());
|
||
}
|
||
|
||
static fromJson() {
|
||
return new XrayCommonClass();
|
||
}
|
||
|
||
/**
|
||
* 【最佳实践】中文注释:这是一个智能的、通用的序列化方法。
|
||
* 1. 使用 ...this 创建一个当前对象所有属性的浅拷贝。
|
||
* 2. 遍历拷贝后的对象,删除所有以下划线 "_" 开头的属性。
|
||
* 这些带下划线的属性被约定为仅供前端 UI 逻辑使用(如 _expiryTime),不应提交给后端。
|
||
* 3. 返回一个干净的、只包含持久化数据的对象。
|
||
* 这个方法将被所有子类继承,无需在每个子类中重复实现,保证了代码的健壮性和可维护性。
|
||
*/
|
||
toJson() {
|
||
const obj = { ...this };
|
||
for (const key in obj) {
|
||
if (key.startsWith('_')) {
|
||
delete obj[key];
|
||
}
|
||
}
|
||
return obj;
|
||
}
|
||
|
||
toString(format = true) {
|
||
return format ? JSON.stringify(this.toJson(), null, 2) : JSON.stringify(this.toJson());
|
||
}
|
||
|
||
static toHeaders(v2Headers) {
|
||
let newHeaders = [];
|
||
if (v2Headers) {
|
||
Object.keys(v2Headers).forEach(key => {
|
||
let values = v2Headers[key];
|
||
if (typeof (values) === 'string') {
|
||
newHeaders.push({ name: key, value: values });
|
||
} else {
|
||
for (let i = 0; i < values.length; ++i) {
|
||
newHeaders.push({ name: key, value: values[i] });
|
||
}
|
||
}
|
||
});
|
||
}
|
||
return newHeaders;
|
||
}
|
||
|
||
static toV2Headers(headers, arr = true) {
|
||
let v2Headers = {};
|
||
for (let i = 0; i < headers.length; ++i) {
|
||
let name = headers[i].name;
|
||
let value = headers[i].value;
|
||
if (ObjectUtil.isEmpty(name) || ObjectUtil.isEmpty(value)) {
|
||
continue;
|
||
}
|
||
if (!(name in v2Headers)) {
|
||
v2Headers[name] = arr ? [value] : value;
|
||
} else {
|
||
if (arr) {
|
||
v2Headers[name].push(value);
|
||
} else {
|
||
v2Headers[name] = value;
|
||
}
|
||
}
|
||
}
|
||
return v2Headers;
|
||
}
|
||
}
|
||
|
||
class TcpStreamSettings extends XrayCommonClass {
|
||
constructor(
|
||
acceptProxyProtocol = false,
|
||
type = 'none',
|
||
request = new TcpStreamSettings.TcpRequest(),
|
||
response = new TcpStreamSettings.TcpResponse(),
|
||
) {
|
||
super();
|
||
this.acceptProxyProtocol = acceptProxyProtocol;
|
||
this.type = type;
|
||
this.request = request;
|
||
this.response = response;
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
let header = json.header;
|
||
if (!header) {
|
||
header = {};
|
||
}
|
||
return new TcpStreamSettings(json.acceptProxyProtocol,
|
||
header.type,
|
||
TcpStreamSettings.TcpRequest.fromJson(header.request),
|
||
TcpStreamSettings.TcpResponse.fromJson(header.response),
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
acceptProxyProtocol: this.acceptProxyProtocol,
|
||
header: {
|
||
type: this.type,
|
||
request: this.type === 'http' ? this.request.toJson() : undefined,
|
||
response: this.type === 'http' ? this.response.toJson() : undefined,
|
||
},
|
||
};
|
||
}
|
||
}
|
||
|
||
TcpStreamSettings.TcpRequest = class extends XrayCommonClass {
|
||
constructor(
|
||
version = '1.1',
|
||
method = 'GET',
|
||
path = ['/'],
|
||
headers = [],
|
||
) {
|
||
super();
|
||
this.version = version;
|
||
this.method = method;
|
||
this.path = path.length === 0 ? ['/'] : path;
|
||
this.headers = headers;
|
||
}
|
||
|
||
addPath(path) {
|
||
this.path.push(path);
|
||
}
|
||
|
||
removePath(index) {
|
||
this.path.splice(index, 1);
|
||
}
|
||
|
||
addHeader(name, value) {
|
||
this.headers.push({ name: name, value: value });
|
||
}
|
||
|
||
removeHeader(index) {
|
||
this.headers.splice(index, 1);
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new TcpStreamSettings.TcpRequest(
|
||
json.version,
|
||
json.method,
|
||
json.path,
|
||
XrayCommonClass.toHeaders(json.headers),
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
version: this.version,
|
||
method: this.method,
|
||
path: ObjectUtil.clone(this.path),
|
||
headers: XrayCommonClass.toV2Headers(this.headers),
|
||
};
|
||
}
|
||
};
|
||
|
||
TcpStreamSettings.TcpResponse = class extends XrayCommonClass {
|
||
constructor(
|
||
version = '1.1',
|
||
status = '200',
|
||
reason = 'OK',
|
||
headers = [],
|
||
) {
|
||
super();
|
||
this.version = version;
|
||
this.status = status;
|
||
this.reason = reason;
|
||
this.headers = headers;
|
||
}
|
||
|
||
addHeader(name, value) {
|
||
this.headers.push({ name: name, value: value });
|
||
}
|
||
|
||
removeHeader(index) {
|
||
this.headers.splice(index, 1);
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new TcpStreamSettings.TcpResponse(
|
||
json.version,
|
||
json.status,
|
||
json.reason,
|
||
XrayCommonClass.toHeaders(json.headers),
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
version: this.version,
|
||
status: this.status,
|
||
reason: this.reason,
|
||
headers: XrayCommonClass.toV2Headers(this.headers),
|
||
};
|
||
}
|
||
};
|
||
|
||
class KcpStreamSettings extends XrayCommonClass {
|
||
constructor(
|
||
mtu = 1350,
|
||
tti = 50,
|
||
uplinkCapacity = 5,
|
||
downlinkCapacity = 20,
|
||
congestion = false,
|
||
readBufferSize = 2,
|
||
writeBufferSize = 2,
|
||
type = 'none',
|
||
seed = RandomUtil.randomSeq(10),
|
||
) {
|
||
super();
|
||
this.mtu = mtu;
|
||
this.tti = tti;
|
||
this.upCap = uplinkCapacity;
|
||
this.downCap = downlinkCapacity;
|
||
this.congestion = congestion;
|
||
this.readBuffer = readBufferSize;
|
||
this.writeBuffer = writeBufferSize;
|
||
this.type = type;
|
||
this.seed = seed;
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new KcpStreamSettings(
|
||
json.mtu,
|
||
json.tti,
|
||
json.uplinkCapacity,
|
||
json.downlinkCapacity,
|
||
json.congestion,
|
||
json.readBufferSize,
|
||
json.writeBufferSize,
|
||
ObjectUtil.isEmpty(json.header) ? 'none' : json.header.type,
|
||
json.seed,
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
mtu: this.mtu,
|
||
tti: this.tti,
|
||
uplinkCapacity: this.upCap,
|
||
downlinkCapacity: this.downCap,
|
||
congestion: this.congestion,
|
||
readBufferSize: this.readBuffer,
|
||
writeBufferSize: this.writeBuffer,
|
||
header: {
|
||
type: this.type,
|
||
},
|
||
seed: this.seed,
|
||
};
|
||
}
|
||
}
|
||
|
||
class WsStreamSettings extends XrayCommonClass {
|
||
constructor(
|
||
acceptProxyProtocol = false,
|
||
path = '/',
|
||
host = '',
|
||
headers = [],
|
||
heartbeatPeriod = 0,
|
||
) {
|
||
super();
|
||
this.acceptProxyProtocol = acceptProxyProtocol;
|
||
this.path = path;
|
||
this.host = host;
|
||
this.headers = headers;
|
||
this.heartbeatPeriod = heartbeatPeriod;
|
||
}
|
||
|
||
addHeader(name, value) {
|
||
this.headers.push({ name: name, value: value });
|
||
}
|
||
|
||
removeHeader(index) {
|
||
this.headers.splice(index, 1);
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new WsStreamSettings(
|
||
json.acceptProxyProtocol,
|
||
json.path,
|
||
json.host,
|
||
XrayCommonClass.toHeaders(json.headers),
|
||
json.heartbeatPeriod,
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
acceptProxyProtocol: this.acceptProxyProtocol,
|
||
path: this.path,
|
||
host: this.host,
|
||
headers: XrayCommonClass.toV2Headers(this.headers, false),
|
||
heartbeatPeriod: this.heartbeatPeriod,
|
||
};
|
||
}
|
||
}
|
||
|
||
class GrpcStreamSettings extends XrayCommonClass {
|
||
constructor(
|
||
serviceName = "",
|
||
authority = "",
|
||
multiMode = false,
|
||
) {
|
||
super();
|
||
this.serviceName = serviceName;
|
||
this.authority = authority;
|
||
this.multiMode = multiMode;
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new GrpcStreamSettings(
|
||
json.serviceName,
|
||
json.authority,
|
||
json.multiMode
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
serviceName: this.serviceName,
|
||
authority: this.authority,
|
||
multiMode: this.multiMode,
|
||
}
|
||
}
|
||
}
|
||
|
||
class HTTPUpgradeStreamSettings extends XrayCommonClass {
|
||
constructor(
|
||
acceptProxyProtocol = false,
|
||
path = '/',
|
||
host = '',
|
||
headers = []
|
||
) {
|
||
super();
|
||
this.acceptProxyProtocol = acceptProxyProtocol;
|
||
this.path = path;
|
||
this.host = host;
|
||
this.headers = headers;
|
||
}
|
||
|
||
addHeader(name, value) {
|
||
this.headers.push({ name: name, value: value });
|
||
}
|
||
|
||
removeHeader(index) {
|
||
this.headers.splice(index, 1);
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new HTTPUpgradeStreamSettings(
|
||
json.acceptProxyProtocol,
|
||
json.path,
|
||
json.host,
|
||
XrayCommonClass.toHeaders(json.headers),
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
acceptProxyProtocol: this.acceptProxyProtocol,
|
||
path: this.path,
|
||
host: this.host,
|
||
headers: XrayCommonClass.toV2Headers(this.headers, false),
|
||
};
|
||
}
|
||
}
|
||
|
||
class xHTTPStreamSettings extends XrayCommonClass {
|
||
constructor(
|
||
path = '/',
|
||
host = '',
|
||
headers = [],
|
||
scMaxBufferedPosts = 30,
|
||
scMaxEachPostBytes = "1000000",
|
||
scStreamUpServerSecs = "20-80",
|
||
noSSEHeader = false,
|
||
xPaddingBytes = "100-1000",
|
||
mode = MODE_OPTION.AUTO,
|
||
) {
|
||
super();
|
||
this.path = path;
|
||
this.host = host;
|
||
this.headers = headers;
|
||
this.scMaxBufferedPosts = scMaxBufferedPosts;
|
||
this.scMaxEachPostBytes = scMaxEachPostBytes;
|
||
this.scStreamUpServerSecs = scStreamUpServerSecs;
|
||
this.noSSEHeader = noSSEHeader;
|
||
this.xPaddingBytes = xPaddingBytes;
|
||
this.mode = mode;
|
||
}
|
||
|
||
addHeader(name, value) {
|
||
this.headers.push({ name: name, value: value });
|
||
}
|
||
|
||
removeHeader(index) {
|
||
this.headers.splice(index, 1);
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new xHTTPStreamSettings(
|
||
json.path,
|
||
json.host,
|
||
XrayCommonClass.toHeaders(json.headers),
|
||
json.scMaxBufferedPosts,
|
||
json.scMaxEachPostBytes,
|
||
json.scStreamUpServerSecs,
|
||
json.noSSEHeader,
|
||
json.xPaddingBytes,
|
||
json.mode,
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
path: this.path,
|
||
host: this.host,
|
||
headers: XrayCommonClass.toV2Headers(this.headers, false),
|
||
scMaxBufferedPosts: this.scMaxBufferedPosts,
|
||
scMaxEachPostBytes: this.scMaxEachPostBytes,
|
||
scStreamUpServerSecs: this.scStreamUpServerSecs,
|
||
noSSEHeader: this.noSSEHeader,
|
||
xPaddingBytes: this.xPaddingBytes,
|
||
mode: this.mode,
|
||
};
|
||
}
|
||
}
|
||
|
||
class TlsStreamSettings extends XrayCommonClass {
|
||
constructor(
|
||
serverName = '',
|
||
minVersion = TLS_VERSION_OPTION.TLS12,
|
||
maxVersion = TLS_VERSION_OPTION.TLS13,
|
||
cipherSuites = '',
|
||
rejectUnknownSni = false,
|
||
verifyPeerCertInNames = ['dns.google', 'cloudflare-dns.com'],
|
||
disableSystemRoot = false,
|
||
enableSessionResumption = false,
|
||
certificates = [new TlsStreamSettings.Cert()],
|
||
alpn = [ALPN_OPTION.H2, ALPN_OPTION.HTTP1],
|
||
echServerKeys = '',
|
||
echForceQuery = 'none',
|
||
settings = new TlsStreamSettings.Settings()
|
||
) {
|
||
super();
|
||
this.sni = serverName;
|
||
this.minVersion = minVersion;
|
||
this.maxVersion = maxVersion;
|
||
this.cipherSuites = cipherSuites;
|
||
this.rejectUnknownSni = rejectUnknownSni;
|
||
this.verifyPeerCertInNames = Array.isArray(verifyPeerCertInNames) ? verifyPeerCertInNames.join(",") : verifyPeerCertInNames;
|
||
this.disableSystemRoot = disableSystemRoot;
|
||
this.enableSessionResumption = enableSessionResumption;
|
||
this.certs = certificates;
|
||
this.alpn = alpn;
|
||
this.echServerKeys = echServerKeys;
|
||
this.echForceQuery = echForceQuery;
|
||
this.settings = settings;
|
||
}
|
||
|
||
addCert() {
|
||
this.certs.push(new TlsStreamSettings.Cert());
|
||
}
|
||
|
||
removeCert(index) {
|
||
this.certs.splice(index, 1);
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
let certs;
|
||
let settings;
|
||
if (!ObjectUtil.isEmpty(json.certificates)) {
|
||
certs = json.certificates.map(cert => TlsStreamSettings.Cert.fromJson(cert));
|
||
}
|
||
|
||
if (!ObjectUtil.isEmpty(json.settings)) {
|
||
settings = new TlsStreamSettings.Settings(json.settings.allowInsecure, json.settings.fingerprint, json.settings.echConfigList);
|
||
}
|
||
return new TlsStreamSettings(
|
||
json.serverName,
|
||
json.minVersion,
|
||
json.maxVersion,
|
||
json.cipherSuites,
|
||
json.rejectUnknownSni,
|
||
json.verifyPeerCertInNames,
|
||
json.disableSystemRoot,
|
||
json.enableSessionResumption,
|
||
certs,
|
||
json.alpn,
|
||
json.echServerKeys,
|
||
json.echForceQuery,
|
||
settings,
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
serverName: this.sni,
|
||
minVersion: this.minVersion,
|
||
maxVersion: this.maxVersion,
|
||
cipherSuites: this.cipherSuites,
|
||
rejectUnknownSni: this.rejectUnknownSni,
|
||
verifyPeerCertInNames: this.verifyPeerCertInNames.split(","),
|
||
disableSystemRoot: this.disableSystemRoot,
|
||
enableSessionResumption: this.enableSessionResumption,
|
||
certificates: TlsStreamSettings.toJsonArray(this.certs),
|
||
alpn: this.alpn,
|
||
echServerKeys: this.echServerKeys,
|
||
echForceQuery: this.echForceQuery,
|
||
settings: this.settings,
|
||
};
|
||
}
|
||
}
|
||
|
||
TlsStreamSettings.Cert = class extends XrayCommonClass {
|
||
constructor(
|
||
useFile = true,
|
||
certificateFile = '',
|
||
keyFile = '',
|
||
certificate = '',
|
||
key = '',
|
||
oneTimeLoading = false,
|
||
usage = USAGE_OPTION.ENCIPHERMENT,
|
||
buildChain = false,
|
||
) {
|
||
super();
|
||
this.useFile = useFile;
|
||
this.certFile = certificateFile;
|
||
this.keyFile = keyFile;
|
||
this.cert = Array.isArray(certificate) ? certificate.join('\n') : certificate;
|
||
this.key = Array.isArray(key) ? key.join('\n') : key;
|
||
this.oneTimeLoading = oneTimeLoading;
|
||
this.usage = usage;
|
||
this.buildChain = buildChain
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
if ('certificateFile' in json && 'keyFile' in json) {
|
||
return new TlsStreamSettings.Cert(
|
||
true,
|
||
json.certificateFile,
|
||
json.keyFile, '', '',
|
||
json.oneTimeLoading,
|
||
json.usage,
|
||
json.buildChain,
|
||
);
|
||
} else {
|
||
return new TlsStreamSettings.Cert(
|
||
false, '', '',
|
||
json.certificate.join('\n'),
|
||
json.key.join('\n'),
|
||
json.oneTimeLoading,
|
||
json.usage,
|
||
json.buildChain,
|
||
);
|
||
}
|
||
}
|
||
|
||
toJson() {
|
||
if (this.useFile) {
|
||
return {
|
||
certificateFile: this.certFile,
|
||
keyFile: this.keyFile,
|
||
oneTimeLoading: this.oneTimeLoading,
|
||
usage: this.usage,
|
||
buildChain: this.buildChain,
|
||
};
|
||
} else {
|
||
return {
|
||
certificate: this.cert.split('\n'),
|
||
key: this.key.split('\n'),
|
||
oneTimeLoading: this.oneTimeLoading,
|
||
usage: this.usage,
|
||
buildChain: this.buildChain,
|
||
};
|
||
}
|
||
}
|
||
};
|
||
|
||
TlsStreamSettings.Settings = class extends XrayCommonClass {
|
||
constructor(
|
||
allowInsecure = false,
|
||
fingerprint = UTLS_FINGERPRINT.UTLS_CHROME,
|
||
echConfigList = '',
|
||
) {
|
||
super();
|
||
this.allowInsecure = allowInsecure;
|
||
this.fingerprint = fingerprint;
|
||
this.echConfigList = echConfigList;
|
||
}
|
||
static fromJson(json = {}) {
|
||
return new TlsStreamSettings.Settings(
|
||
json.allowInsecure,
|
||
json.fingerprint,
|
||
json.echConfigList,
|
||
);
|
||
}
|
||
toJson() {
|
||
return {
|
||
allowInsecure: this.allowInsecure,
|
||
fingerprint: this.fingerprint,
|
||
echConfigList: this.echConfigList
|
||
};
|
||
}
|
||
};
|
||
|
||
|
||
class RealityStreamSettings extends XrayCommonClass {
|
||
constructor(
|
||
show = false,
|
||
xver = 0,
|
||
target = 'tesla.com:443',
|
||
serverNames = 'tesla.com,www.tesla.com',
|
||
privateKey = '',
|
||
minClientVer = '',
|
||
maxClientVer = '',
|
||
maxTimediff = 0,
|
||
shortIds = RandomUtil.randomShortIds(),
|
||
mldsa65Seed = '',
|
||
settings = new RealityStreamSettings.Settings()
|
||
) {
|
||
super();
|
||
this.show = show;
|
||
this.xver = xver;
|
||
this.target = target;
|
||
this.serverNames = Array.isArray(serverNames) ? serverNames.join(",") : serverNames;
|
||
this.privateKey = privateKey;
|
||
this.minClientVer = minClientVer;
|
||
this.maxClientVer = maxClientVer;
|
||
this.maxTimediff = maxTimediff;
|
||
this.shortIds = Array.isArray(shortIds) ? shortIds.join(",") : shortIds;
|
||
this.mldsa65Seed = mldsa65Seed;
|
||
this.settings = settings;
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
let settings;
|
||
if (!ObjectUtil.isEmpty(json.settings)) {
|
||
settings = new RealityStreamSettings.Settings(
|
||
json.settings.publicKey,
|
||
json.settings.fingerprint,
|
||
json.settings.serverName,
|
||
json.settings.spiderX,
|
||
json.settings.mldsa65Verify,
|
||
);
|
||
}
|
||
return new RealityStreamSettings(
|
||
json.show,
|
||
json.xver,
|
||
json.target,
|
||
json.serverNames,
|
||
json.privateKey,
|
||
json.minClientVer,
|
||
json.maxClientVer,
|
||
json.maxTimediff,
|
||
json.shortIds,
|
||
json.mldsa65Seed,
|
||
settings,
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
show: this.show,
|
||
xver: this.xver,
|
||
target: this.target,
|
||
serverNames: this.serverNames.split(","),
|
||
privateKey: this.privateKey,
|
||
minClientVer: this.minClientVer,
|
||
maxClientVer: this.maxClientVer,
|
||
maxTimediff: this.maxTimediff,
|
||
shortIds: this.shortIds.split(","),
|
||
mldsa65Seed: this.mldsa65Seed,
|
||
settings: this.settings,
|
||
};
|
||
}
|
||
}
|
||
|
||
RealityStreamSettings.Settings = class extends XrayCommonClass {
|
||
constructor(
|
||
publicKey = '',
|
||
fingerprint = UTLS_FINGERPRINT.UTLS_CHROME,
|
||
serverName = '',
|
||
spiderX = '/',
|
||
mldsa65Verify = ''
|
||
) {
|
||
super();
|
||
this.publicKey = publicKey;
|
||
this.fingerprint = fingerprint;
|
||
this.serverName = serverName;
|
||
this.spiderX = spiderX;
|
||
this.mldsa65Verify = mldsa65Verify;
|
||
}
|
||
static fromJson(json = {}) {
|
||
return new RealityStreamSettings.Settings(
|
||
json.publicKey,
|
||
json.fingerprint,
|
||
json.serverName,
|
||
json.spiderX,
|
||
json.mldsa65Verify
|
||
);
|
||
}
|
||
toJson() {
|
||
return {
|
||
publicKey: this.publicKey,
|
||
fingerprint: this.fingerprint,
|
||
serverName: this.serverName,
|
||
spiderX: this.spiderX,
|
||
mldsa65Verify: this.mldsa65Verify
|
||
};
|
||
}
|
||
};
|
||
|
||
class SockoptStreamSettings extends XrayCommonClass {
|
||
constructor(
|
||
acceptProxyProtocol = false,
|
||
tcpFastOpen = false,
|
||
mark = 0,
|
||
tproxy = "off",
|
||
tcpMptcp = false,
|
||
penetrate = false,
|
||
domainStrategy = DOMAIN_STRATEGY_OPTION.USE_IP,
|
||
tcpMaxSeg = 1440,
|
||
dialerProxy = "",
|
||
tcpKeepAliveInterval = 0,
|
||
tcpKeepAliveIdle = 300,
|
||
tcpUserTimeout = 10000,
|
||
tcpcongestion = TCP_CONGESTION_OPTION.BBR,
|
||
V6Only = false,
|
||
tcpWindowClamp = 600,
|
||
interfaceName = "",
|
||
) {
|
||
super();
|
||
this.acceptProxyProtocol = acceptProxyProtocol;
|
||
this.tcpFastOpen = tcpFastOpen;
|
||
this.mark = mark;
|
||
this.tproxy = tproxy;
|
||
this.tcpMptcp = tcpMptcp;
|
||
this.penetrate = penetrate;
|
||
this.domainStrategy = domainStrategy;
|
||
this.tcpMaxSeg = tcpMaxSeg;
|
||
this.dialerProxy = dialerProxy;
|
||
this.tcpKeepAliveInterval = tcpKeepAliveInterval;
|
||
this.tcpKeepAliveIdle = tcpKeepAliveIdle;
|
||
this.tcpUserTimeout = tcpUserTimeout;
|
||
this.tcpcongestion = tcpcongestion;
|
||
this.V6Only = V6Only;
|
||
this.tcpWindowClamp = tcpWindowClamp;
|
||
this.interfaceName = interfaceName;
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
if (Object.keys(json).length === 0) return undefined;
|
||
return new SockoptStreamSettings(
|
||
json.acceptProxyProtocol,
|
||
json.tcpFastOpen,
|
||
json.mark,
|
||
json.tproxy,
|
||
json.tcpMptcp,
|
||
json.penetrate,
|
||
json.domainStrategy,
|
||
json.tcpMaxSeg,
|
||
json.dialerProxy,
|
||
json.tcpKeepAliveInterval,
|
||
json.tcpKeepAliveIdle,
|
||
json.tcpUserTimeout,
|
||
json.tcpcongestion,
|
||
json.V6Only,
|
||
json.tcpWindowClamp,
|
||
json.interface,
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
acceptProxyProtocol: this.acceptProxyProtocol,
|
||
tcpFastOpen: this.tcpFastOpen,
|
||
mark: this.mark,
|
||
tproxy: this.tproxy,
|
||
tcpMptcp: this.tcpMptcp,
|
||
penetrate: this.penetrate,
|
||
domainStrategy: this.domainStrategy,
|
||
tcpMaxSeg: this.tcpMaxSeg,
|
||
dialerProxy: this.dialerProxy,
|
||
tcpKeepAliveInterval: this.tcpKeepAliveInterval,
|
||
tcpKeepAliveIdle: this.tcpKeepAliveIdle,
|
||
tcpUserTimeout: this.tcpUserTimeout,
|
||
tcpcongestion: this.tcpcongestion,
|
||
V6Only: this.V6Only,
|
||
tcpWindowClamp: this.tcpWindowClamp,
|
||
interface: this.interfaceName,
|
||
};
|
||
}
|
||
}
|
||
|
||
class StreamSettings extends XrayCommonClass {
|
||
constructor(network = 'tcp',
|
||
security = 'none',
|
||
externalProxy = [],
|
||
tlsSettings = new TlsStreamSettings(),
|
||
realitySettings = new RealityStreamSettings(),
|
||
tcpSettings = new TcpStreamSettings(),
|
||
kcpSettings = new KcpStreamSettings(),
|
||
wsSettings = new WsStreamSettings(),
|
||
grpcSettings = new GrpcStreamSettings(),
|
||
httpupgradeSettings = new HTTPUpgradeStreamSettings(),
|
||
xhttpSettings = new xHTTPStreamSettings(),
|
||
sockopt = undefined,
|
||
) {
|
||
super();
|
||
this.network = network;
|
||
this.security = security;
|
||
this.externalProxy = externalProxy;
|
||
this.tls = tlsSettings;
|
||
this.reality = realitySettings;
|
||
this.tcp = tcpSettings;
|
||
this.kcp = kcpSettings;
|
||
this.ws = wsSettings;
|
||
this.grpc = grpcSettings;
|
||
this.httpupgrade = httpupgradeSettings;
|
||
this.xhttp = xhttpSettings;
|
||
this.sockopt = sockopt;
|
||
}
|
||
|
||
get isTls() {
|
||
return this.security === "tls";
|
||
}
|
||
|
||
set isTls(isTls) {
|
||
if (isTls) {
|
||
this.security = 'tls';
|
||
} else {
|
||
this.security = 'none';
|
||
}
|
||
}
|
||
|
||
//for Reality
|
||
get isReality() {
|
||
return this.security === "reality";
|
||
}
|
||
|
||
set isReality(isReality) {
|
||
if (isReality) {
|
||
this.security = 'reality';
|
||
} else {
|
||
this.security = 'none';
|
||
}
|
||
}
|
||
|
||
get sockoptSwitch() {
|
||
return this.sockopt != undefined;
|
||
}
|
||
|
||
set sockoptSwitch(value) {
|
||
this.sockopt = value ? new SockoptStreamSettings() : undefined;
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new StreamSettings(
|
||
json.network,
|
||
json.security,
|
||
json.externalProxy,
|
||
TlsStreamSettings.fromJson(json.tlsSettings),
|
||
RealityStreamSettings.fromJson(json.realitySettings),
|
||
TcpStreamSettings.fromJson(json.tcpSettings),
|
||
KcpStreamSettings.fromJson(json.kcpSettings),
|
||
WsStreamSettings.fromJson(json.wsSettings),
|
||
GrpcStreamSettings.fromJson(json.grpcSettings),
|
||
HTTPUpgradeStreamSettings.fromJson(json.httpupgradeSettings),
|
||
xHTTPStreamSettings.fromJson(json.xhttpSettings),
|
||
SockoptStreamSettings.fromJson(json.sockopt),
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
const network = this.network;
|
||
return {
|
||
network: network,
|
||
security: this.security,
|
||
externalProxy: this.externalProxy,
|
||
tlsSettings: this.isTls ? this.tls.toJson() : undefined,
|
||
realitySettings: this.isReality ? this.reality.toJson() : undefined,
|
||
tcpSettings: network === 'tcp' ? this.tcp.toJson() : undefined,
|
||
kcpSettings: network === 'kcp' ? this.kcp.toJson() : undefined,
|
||
wsSettings: network === 'ws' ? this.ws.toJson() : undefined,
|
||
grpcSettings: network === 'grpc' ? this.grpc.toJson() : undefined,
|
||
httpupgradeSettings: network === 'httpupgrade' ? this.httpupgrade.toJson() : undefined,
|
||
xhttpSettings: network === 'xhttp' ? this.xhttp.toJson() : undefined,
|
||
sockopt: this.sockopt != undefined ? this.sockopt.toJson() : undefined,
|
||
};
|
||
}
|
||
}
|
||
|
||
class Sniffing extends XrayCommonClass {
|
||
constructor(
|
||
enabled = false,
|
||
destOverride = ['http', 'tls', 'quic', 'fakedns'],
|
||
metadataOnly = false,
|
||
routeOnly = false) {
|
||
super();
|
||
this.enabled = enabled;
|
||
this.destOverride = destOverride;
|
||
this.metadataOnly = metadataOnly;
|
||
this.routeOnly = routeOnly;
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
let destOverride = ObjectUtil.clone(json.destOverride);
|
||
if (!ObjectUtil.isEmpty(destOverride) && !ObjectUtil.isArrEmpty(destOverride)) {
|
||
if (ObjectUtil.isEmpty(destOverride[0])) {
|
||
destOverride = ['http', 'tls', 'quic', 'fakedns'];
|
||
}
|
||
}
|
||
return new Sniffing(
|
||
!!json.enabled,
|
||
destOverride,
|
||
json.metadataOnly,
|
||
json.routeOnly,
|
||
);
|
||
}
|
||
}
|
||
|
||
class Inbound extends XrayCommonClass {
|
||
constructor(
|
||
port = RandomUtil.randomInteger(10000, 60000),
|
||
listen = '',
|
||
protocol = Protocols.VLESS,
|
||
settings = null,
|
||
streamSettings = new StreamSettings(),
|
||
tag = '',
|
||
sniffing = new Sniffing(),
|
||
clientStats = '',
|
||
) {
|
||
super();
|
||
this.port = port;
|
||
this.listen = listen;
|
||
this._protocol = protocol;
|
||
this.settings = ObjectUtil.isEmpty(settings) ? Inbound.Settings.getSettings(protocol) : settings;
|
||
this.stream = streamSettings;
|
||
this.tag = tag;
|
||
this.sniffing = sniffing;
|
||
this.clientStats = clientStats;
|
||
}
|
||
getClientStats() {
|
||
return this.clientStats;
|
||
}
|
||
|
||
get clients() {
|
||
switch (this.protocol) {
|
||
case Protocols.VMESS: return this.settings.vmesses;
|
||
case Protocols.VLESS: return this.settings.vlesses;
|
||
case Protocols.TROJAN: return this.settings.trojans;
|
||
case Protocols.SHADOWSOCKS: return this.isSSMultiUser ? this.settings.shadowsockses : null;
|
||
default: return null;
|
||
}
|
||
}
|
||
|
||
get protocol() {
|
||
return this._protocol;
|
||
}
|
||
|
||
set protocol(protocol) {
|
||
this._protocol = protocol;
|
||
this.settings = Inbound.Settings.getSettings(protocol);
|
||
if (protocol === Protocols.TROJAN) {
|
||
this.tls = false;
|
||
}
|
||
}
|
||
|
||
get network() {
|
||
return this.stream.network;
|
||
}
|
||
|
||
set network(network) {
|
||
this.stream.network = network;
|
||
}
|
||
|
||
get isTcp() {
|
||
return this.network === "tcp";
|
||
}
|
||
|
||
get isWs() {
|
||
return this.network === "ws";
|
||
}
|
||
|
||
get isKcp() {
|
||
return this.network === "kcp";
|
||
}
|
||
|
||
get isGrpc() {
|
||
return this.network === "grpc";
|
||
}
|
||
|
||
get isHttpupgrade() {
|
||
return this.network === "httpupgrade";
|
||
}
|
||
|
||
get isXHTTP() {
|
||
return this.network === "xhttp";
|
||
}
|
||
|
||
// Shadowsocks
|
||
get method() {
|
||
switch (this.protocol) {
|
||
case Protocols.SHADOWSOCKS:
|
||
return this.settings.method;
|
||
default:
|
||
return "";
|
||
}
|
||
}
|
||
get isSSMultiUser() {
|
||
return this.method != SSMethods.BLAKE3_CHACHA20_POLY1305;
|
||
}
|
||
get isSS2022() {
|
||
return this.method.substring(0, 4) === "2022";
|
||
}
|
||
|
||
get serverName() {
|
||
if (this.stream.isTls) return this.stream.tls.sni;
|
||
if (this.stream.isReality) return this.stream.reality.serverNames;
|
||
return "";
|
||
}
|
||
|
||
getHeader(obj, name) {
|
||
for (const header of obj.headers) {
|
||
if (header.name.toLowerCase() === name.toLowerCase()) {
|
||
return header.value;
|
||
}
|
||
}
|
||
return "";
|
||
}
|
||
|
||
get host() {
|
||
if (this.isTcp) {
|
||
return this.getHeader(this.stream.tcp.request, 'host');
|
||
} else if (this.isWs) {
|
||
return this.stream.ws.host?.length > 0 ? this.stream.ws.host : this.getHeader(this.stream.ws, 'host');
|
||
} else if (this.isHttpupgrade) {
|
||
return this.stream.httpupgrade.host?.length > 0 ? this.stream.httpupgrade.host : this.getHeader(this.stream.httpupgrade, 'host');
|
||
} else if (this.isXHTTP) {
|
||
return this.stream.xhttp.host?.length > 0 ? this.stream.xhttp.host : this.getHeader(this.stream.xhttp, 'host');
|
||
}
|
||
return null;
|
||
}
|
||
|
||
get path() {
|
||
if (this.isTcp) {
|
||
return this.stream.tcp.request.path[0];
|
||
} else if (this.isWs) {
|
||
return this.stream.ws.path;
|
||
} else if (this.isHttpupgrade) {
|
||
return this.stream.httpupgrade.path;
|
||
} else if (this.isXHTTP) {
|
||
return this.stream.xhttp.path;
|
||
}
|
||
return null;
|
||
}
|
||
|
||
get kcpType() {
|
||
return this.stream.kcp.type;
|
||
}
|
||
|
||
get kcpSeed() {
|
||
return this.stream.kcp.seed;
|
||
}
|
||
|
||
get serviceName() {
|
||
return this.stream.grpc.serviceName;
|
||
}
|
||
|
||
isExpiry(index) {
|
||
let exp = this.clients[index].expiryTime;
|
||
return exp > 0 ? exp < new Date().getTime() : false;
|
||
}
|
||
|
||
canEnableTls() {
|
||
if (![Protocols.VMESS, Protocols.VLESS, Protocols.TROJAN, Protocols.SHADOWSOCKS].includes(this.protocol)) return false;
|
||
return ["tcp", "ws", "http", "grpc", "httpupgrade", "xhttp"].includes(this.network);
|
||
}
|
||
|
||
//this is used for xtls-rprx-vision
|
||
canEnableTlsFlow() {
|
||
if (((this.stream.security === 'tls') || (this.stream.security === 'reality')) && (this.network === "tcp")) {
|
||
return this.protocol === Protocols.VLESS;
|
||
}
|
||
return false;
|
||
}
|
||
|
||
canEnableReality() {
|
||
if (![Protocols.VLESS, Protocols.TROJAN].includes(this.protocol)) return false;
|
||
return ["tcp", "http", "grpc", "xhttp"].includes(this.network);
|
||
}
|
||
|
||
canEnableStream() {
|
||
return [Protocols.VMESS, Protocols.VLESS, Protocols.TROJAN, Protocols.SHADOWSOCKS].includes(this.protocol);
|
||
}
|
||
|
||
reset() {
|
||
this.port = RandomUtil.randomInteger(10000, 60000);
|
||
this.listen = '';
|
||
this.protocol = Protocols.VMESS;
|
||
this.settings = Inbound.Settings.getSettings(Protocols.VMESS);
|
||
this.stream = new StreamSettings();
|
||
this.tag = '';
|
||
this.sniffing = new Sniffing();
|
||
}
|
||
|
||
genVmessLink(address = '', port = this.port, forceTls, remark = '', clientId, security) {
|
||
if (this.protocol !== Protocols.VMESS) {
|
||
return '';
|
||
}
|
||
const tls = forceTls == 'same' ? this.stream.security : forceTls;
|
||
let obj = {
|
||
v: '2',
|
||
ps: remark,
|
||
add: address,
|
||
port: port,
|
||
id: clientId,
|
||
scy: security,
|
||
net: this.stream.network,
|
||
tls: tls,
|
||
};
|
||
const network = this.stream.network;
|
||
if (network === 'tcp') {
|
||
const tcp = this.stream.tcp;
|
||
obj.type = tcp.type;
|
||
if (tcp.type === 'http') {
|
||
const request = tcp.request;
|
||
obj.path = request.path.join(',');
|
||
const host = this.getHeader(request, 'host');
|
||
if (host) obj.host = host;
|
||
}
|
||
} else if (network === 'kcp') {
|
||
const kcp = this.stream.kcp;
|
||
obj.type = kcp.type;
|
||
obj.path = kcp.seed;
|
||
} else if (network === 'ws') {
|
||
const ws = this.stream.ws;
|
||
obj.path = ws.path;
|
||
obj.host = ws.host?.length > 0 ? ws.host : this.getHeader(ws, 'host');
|
||
} else if (network === 'grpc') {
|
||
obj.path = this.stream.grpc.serviceName;
|
||
obj.authority = this.stream.grpc.authority;
|
||
if (this.stream.grpc.multiMode) {
|
||
obj.type = 'multi'
|
||
}
|
||
} else if (network === 'httpupgrade') {
|
||
const httpupgrade = this.stream.httpupgrade;
|
||
obj.path = httpupgrade.path;
|
||
obj.host = httpupgrade.host?.length > 0 ? httpupgrade.host : this.getHeader(httpupgrade, 'host');
|
||
} else if (network === 'xhttp') {
|
||
const xhttp = this.stream.xhttp;
|
||
obj.path = xhttp.path;
|
||
obj.host = xhttp.host?.length > 0 ? xhttp.host : this.getHeader(xhttp, 'host');
|
||
obj.type = xhttp.mode;
|
||
}
|
||
|
||
if (tls === 'tls') {
|
||
if (!ObjectUtil.isEmpty(this.stream.tls.sni)) {
|
||
obj.sni = this.stream.tls.sni;
|
||
}
|
||
if (!ObjectUtil.isEmpty(this.stream.tls.settings.fingerprint)) {
|
||
obj.fp = this.stream.tls.settings.fingerprint;
|
||
}
|
||
if (this.stream.tls.alpn.length > 0) {
|
||
obj.alpn = this.stream.tls.alpn.join(',');
|
||
}
|
||
if (this.stream.tls.settings.allowInsecure) {
|
||
obj.allowInsecure = this.stream.tls.settings.allowInsecure;
|
||
}
|
||
}
|
||
|
||
return 'vmess://' + Base64.encode(JSON.stringify(obj, null, 2));
|
||
}
|
||
|
||
genVLESSLink(address = '', port = this.port, forceTls, remark = '', clientId, flow) {
|
||
const uuid = clientId;
|
||
const type = this.stream.network;
|
||
const security = forceTls == 'same' ? this.stream.security : forceTls;
|
||
const params = new Map();
|
||
params.set("type", this.stream.network);
|
||
params.set("encryption", this.settings.encryption);
|
||
switch (type) {
|
||
case "tcp":
|
||
const tcp = this.stream.tcp;
|
||
if (tcp.type === 'http') {
|
||
const request = tcp.request;
|
||
params.set("path", request.path.join(','));
|
||
const index = request.headers.findIndex(header => header.name.toLowerCase() === 'host');
|
||
if (index >= 0) {
|
||
const host = request.headers[index].value;
|
||
params.set("host", host);
|
||
}
|
||
params.set("headerType", 'http');
|
||
}
|
||
break;
|
||
case "kcp":
|
||
const kcp = this.stream.kcp;
|
||
params.set("headerType", kcp.type);
|
||
params.set("seed", kcp.seed);
|
||
break;
|
||
case "ws":
|
||
const ws = this.stream.ws;
|
||
params.set("path", ws.path);
|
||
params.set("host", ws.host?.length > 0 ? ws.host : this.getHeader(ws, 'host'));
|
||
break;
|
||
case "grpc":
|
||
const grpc = this.stream.grpc;
|
||
params.set("serviceName", grpc.serviceName);
|
||
params.set("authority", grpc.authority);
|
||
if (grpc.multiMode) {
|
||
params.set("mode", "multi");
|
||
}
|
||
break;
|
||
case "httpupgrade":
|
||
const httpupgrade = this.stream.httpupgrade;
|
||
params.set("path", httpupgrade.path);
|
||
params.set("host", httpupgrade.host?.length > 0 ? httpupgrade.host : this.getHeader(httpupgrade, 'host'));
|
||
break;
|
||
case "xhttp":
|
||
const xhttp = this.stream.xhttp;
|
||
params.set("path", xhttp.path);
|
||
params.set("host", xhttp.host?.length > 0 ? xhttp.host : this.getHeader(xhttp, 'host'));
|
||
params.set("mode", xhttp.mode);
|
||
break;
|
||
}
|
||
|
||
if (security === 'tls') {
|
||
params.set("security", "tls");
|
||
if (this.stream.isTls) {
|
||
params.set("fp", this.stream.tls.settings.fingerprint);
|
||
params.set("alpn", this.stream.tls.alpn);
|
||
if (this.stream.tls.settings.allowInsecure) {
|
||
params.set("allowInsecure", "1");
|
||
}
|
||
if (!ObjectUtil.isEmpty(this.stream.tls.sni)) {
|
||
params.set("sni", this.stream.tls.sni);
|
||
}
|
||
if (this.stream.tls.settings.echConfigList?.length > 0) {
|
||
params.set("ech", this.stream.tls.settings.echConfigList);
|
||
}
|
||
if (type == "tcp" && !ObjectUtil.isEmpty(flow)) {
|
||
params.set("flow", flow);
|
||
}
|
||
}
|
||
}
|
||
|
||
else if (security === 'reality') {
|
||
params.set("security", "reality");
|
||
params.set("pbk", this.stream.reality.settings.publicKey);
|
||
params.set("fp", this.stream.reality.settings.fingerprint);
|
||
if (!ObjectUtil.isArrEmpty(this.stream.reality.serverNames)) {
|
||
params.set("sni", this.stream.reality.serverNames.split(",")[0]);
|
||
}
|
||
if (this.stream.reality.shortIds.length > 0) {
|
||
params.set("sid", this.stream.reality.shortIds.split(",")[0]);
|
||
}
|
||
if (!ObjectUtil.isEmpty(this.stream.reality.settings.spiderX)) {
|
||
params.set("spx", this.stream.reality.settings.spiderX);
|
||
}
|
||
if (!ObjectUtil.isEmpty(this.stream.reality.settings.mldsa65Verify)) {
|
||
params.set("pqv", this.stream.reality.settings.mldsa65Verify);
|
||
}
|
||
if (type == 'tcp' && !ObjectUtil.isEmpty(flow)) {
|
||
params.set("flow", flow);
|
||
}
|
||
}
|
||
|
||
else {
|
||
params.set("security", "none");
|
||
}
|
||
|
||
const link = `vless://${uuid}@${address}:${port}`;
|
||
const url = new URL(link);
|
||
for (const [key, value] of params) {
|
||
url.searchParams.set(key, value)
|
||
}
|
||
url.hash = encodeURIComponent(remark);
|
||
return url.toString();
|
||
}
|
||
|
||
genSSLink(address = '', port = this.port, forceTls, remark = '', clientPassword) {
|
||
let settings = this.settings;
|
||
const type = this.stream.network;
|
||
const security = forceTls == 'same' ? this.stream.security : forceTls;
|
||
const params = new Map();
|
||
params.set("type", this.stream.network);
|
||
switch (type) {
|
||
case "tcp":
|
||
const tcp = this.stream.tcp;
|
||
if (tcp.type === 'http') {
|
||
const request = tcp.request;
|
||
params.set("path", request.path.join(','));
|
||
const index = request.headers.findIndex(header => header.name.toLowerCase() === 'host');
|
||
if (index >= 0) {
|
||
const host = request.headers[index].value;
|
||
params.set("host", host);
|
||
}
|
||
params.set("headerType", 'http');
|
||
}
|
||
break;
|
||
case "kcp":
|
||
const kcp = this.stream.kcp;
|
||
params.set("headerType", kcp.type);
|
||
params.set("seed", kcp.seed);
|
||
break;
|
||
case "ws":
|
||
const ws = this.stream.ws;
|
||
params.set("path", ws.path);
|
||
params.set("host", ws.host?.length > 0 ? ws.host : this.getHeader(ws, 'host'));
|
||
break;
|
||
case "grpc":
|
||
const grpc = this.stream.grpc;
|
||
params.set("serviceName", grpc.serviceName);
|
||
params.set("authority", grpc.authority);
|
||
if (grpc.multiMode) {
|
||
params.set("mode", "multi");
|
||
}
|
||
break;
|
||
case "httpupgrade":
|
||
const httpupgrade = this.stream.httpupgrade;
|
||
params.set("path", httpupgrade.path);
|
||
params.set("host", httpupgrade.host?.length > 0 ? httpupgrade.host : this.getHeader(httpupgrade, 'host'));
|
||
break;
|
||
case "xhttp":
|
||
const xhttp = this.stream.xhttp;
|
||
params.set("path", xhttp.path);
|
||
params.set("host", xhttp.host?.length > 0 ? xhttp.host : this.getHeader(xhttp, 'host'));
|
||
params.set("mode", xhttp.mode);
|
||
break;
|
||
}
|
||
|
||
if (security === 'tls') {
|
||
params.set("security", "tls");
|
||
if (this.stream.isTls) {
|
||
params.set("fp", this.stream.tls.settings.fingerprint);
|
||
params.set("alpn", this.stream.tls.alpn);
|
||
if (this.stream.tls.settings.allowInsecure) {
|
||
params.set("allowInsecure", "1");
|
||
}
|
||
if (this.stream.tls.settings.echConfigList?.length > 0) {
|
||
params.set("ech", this.stream.tls.settings.echConfigList);
|
||
}
|
||
if (!ObjectUtil.isEmpty(this.stream.tls.sni)) {
|
||
params.set("sni", this.stream.tls.sni);
|
||
}
|
||
}
|
||
}
|
||
|
||
|
||
let password = new Array();
|
||
if (this.isSS2022) password.push(settings.password);
|
||
if (this.isSSMultiUser) password.push(clientPassword);
|
||
|
||
let link = `ss://${Base64.encode(`${settings.method}:${password.join(':')}`, true)}@${address}:${port}`;
|
||
const url = new URL(link);
|
||
for (const [key, value] of params) {
|
||
url.searchParams.set(key, value)
|
||
}
|
||
url.hash = encodeURIComponent(remark);
|
||
return url.toString();
|
||
}
|
||
|
||
genTrojanLink(address = '', port = this.port, forceTls, remark = '', clientPassword) {
|
||
const security = forceTls == 'same' ? this.stream.security : forceTls;
|
||
const type = this.stream.network;
|
||
const params = new Map();
|
||
params.set("type", this.stream.network);
|
||
switch (type) {
|
||
case "tcp":
|
||
const tcp = this.stream.tcp;
|
||
if (tcp.type === 'http') {
|
||
const request = tcp.request;
|
||
params.set("path", request.path.join(','));
|
||
const index = request.headers.findIndex(header => header.name.toLowerCase() === 'host');
|
||
if (index >= 0) {
|
||
const host = request.headers[index].value;
|
||
params.set("host", host);
|
||
}
|
||
params.set("headerType", 'http');
|
||
}
|
||
break;
|
||
case "kcp":
|
||
const kcp = this.stream.kcp;
|
||
params.set("headerType", kcp.type);
|
||
params.set("seed", kcp.seed);
|
||
break;
|
||
case "ws":
|
||
const ws = this.stream.ws;
|
||
params.set("path", ws.path);
|
||
params.set("host", ws.host?.length > 0 ? ws.host : this.getHeader(ws, 'host'));
|
||
break;
|
||
case "grpc":
|
||
const grpc = this.stream.grpc;
|
||
params.set("serviceName", grpc.serviceName);
|
||
params.set("authority", grpc.authority);
|
||
if (grpc.multiMode) {
|
||
params.set("mode", "multi");
|
||
}
|
||
break;
|
||
case "httpupgrade":
|
||
const httpupgrade = this.stream.httpupgrade;
|
||
params.set("path", httpupgrade.path);
|
||
params.set("host", httpupgrade.host?.length > 0 ? httpupgrade.host : this.getHeader(httpupgrade, 'host'));
|
||
break;
|
||
case "xhttp":
|
||
const xhttp = this.stream.xhttp;
|
||
params.set("path", xhttp.path);
|
||
params.set("host", xhttp.host?.length > 0 ? xhttp.host : this.getHeader(xhttp, 'host'));
|
||
params.set("mode", xhttp.mode);
|
||
break;
|
||
}
|
||
|
||
if (security === 'tls') {
|
||
params.set("security", "tls");
|
||
if (this.stream.isTls) {
|
||
params.set("fp", this.stream.tls.settings.fingerprint);
|
||
params.set("alpn", this.stream.tls.alpn);
|
||
if (this.stream.tls.settings.allowInsecure) {
|
||
params.set("allowInsecure", "1");
|
||
}
|
||
if (this.stream.tls.settings.echConfigList?.length > 0) {
|
||
params.set("ech", this.stream.tls.settings.echConfigList);
|
||
}
|
||
if (!ObjectUtil.isEmpty(this.stream.tls.sni)) {
|
||
params.set("sni", this.stream.tls.sni);
|
||
}
|
||
}
|
||
}
|
||
|
||
else if (security === 'reality') {
|
||
params.set("security", "reality");
|
||
params.set("pbk", this.stream.reality.settings.publicKey);
|
||
params.set("fp", this.stream.reality.settings.fingerprint);
|
||
if (!ObjectUtil.isArrEmpty(this.stream.reality.serverNames)) {
|
||
params.set("sni", this.stream.reality.serverNames.split(",")[0]);
|
||
}
|
||
if (this.stream.reality.shortIds.length > 0) {
|
||
params.set("sid", this.stream.reality.shortIds.split(",")[0]);
|
||
}
|
||
if (!ObjectUtil.isEmpty(this.stream.reality.settings.spiderX)) {
|
||
params.set("spx", this.stream.reality.settings.spiderX);
|
||
}
|
||
if (!ObjectUtil.isEmpty(this.stream.reality.settings.mldsa65Verify)) {
|
||
params.set("pqv", this.stream.reality.settings.mldsa65Verify);
|
||
}
|
||
}
|
||
|
||
else {
|
||
params.set("security", "none");
|
||
}
|
||
|
||
const link = `trojan://${clientPassword}@${address}:${port}`;
|
||
const url = new URL(link);
|
||
for (const [key, value] of params) {
|
||
url.searchParams.set(key, value)
|
||
}
|
||
url.hash = encodeURIComponent(remark);
|
||
return url.toString();
|
||
}
|
||
|
||
getWireguardLink(address, port, remark, peerId) {
|
||
let txt = `[Interface]\n`
|
||
txt += `PrivateKey = ${this.settings.peers[peerId].privateKey}\n`
|
||
txt += `Address = ${this.settings.peers[peerId].allowedIPs[0]}\n`
|
||
txt += `DNS = 1.1.1.1, 1.0.0.1\n`
|
||
if (this.settings.mtu) {
|
||
txt += `MTU = ${this.settings.mtu}\n`
|
||
}
|
||
txt += `\n# ${remark}\n`
|
||
txt += `[Peer]\n`
|
||
txt += `PublicKey = ${this.settings.pubKey}\n`
|
||
txt += `AllowedIPs = 0.0.0.0/0, ::/0\n`
|
||
txt += `Endpoint = ${address}:${port}`
|
||
if (this.settings.peers[peerId].psk) {
|
||
txt += `\nPresharedKey = ${this.settings.peers[peerId].psk}`
|
||
}
|
||
if (this.settings.peers[peerId].keepAlive) {
|
||
txt += `\nPersistentKeepalive = ${this.settings.peers[peerId].keepAlive}\n`
|
||
}
|
||
return txt;
|
||
}
|
||
|
||
genLink(address = '', port = this.port, forceTls = 'same', remark = '', client) {
|
||
switch (this.protocol) {
|
||
case Protocols.VMESS:
|
||
return this.genVmessLink(address, port, forceTls, remark, client.id, client.security);
|
||
case Protocols.VLESS:
|
||
return this.genVLESSLink(address, port, forceTls, remark, client.id, client.flow);
|
||
case Protocols.SHADOWSOCKS:
|
||
return this.genSSLink(address, port, forceTls, remark, this.isSSMultiUser ? client.password : '');
|
||
case Protocols.TROJAN:
|
||
return this.genTrojanLink(address, port, forceTls, remark, client.password);
|
||
default: return '';
|
||
}
|
||
}
|
||
|
||
genAllLinks(remark = '', remarkModel = '-ieo', client) {
|
||
let result = [];
|
||
let email = client ? client.email : '';
|
||
let addr = !ObjectUtil.isEmpty(this.listen) && this.listen !== "0.0.0.0" ? this.listen : location.hostname;
|
||
let port = this.port;
|
||
const separationChar = remarkModel.charAt(0);
|
||
const orderChars = remarkModel.slice(1);
|
||
let orders = {
|
||
'i': remark,
|
||
'e': email,
|
||
'o': '',
|
||
};
|
||
if (ObjectUtil.isArrEmpty(this.stream.externalProxy)) {
|
||
let r = orderChars.split('').map(char => orders[char]).filter(x => x.length > 0).join(separationChar);
|
||
result.push({
|
||
remark: r,
|
||
link: this.genLink(addr, port, 'same', r, client)
|
||
});
|
||
} else {
|
||
this.stream.externalProxy.forEach((ep) => {
|
||
orders['o'] = ep.remark;
|
||
let r = orderChars.split('').map(char => orders[char]).filter(x => x.length > 0).join(separationChar);
|
||
result.push({
|
||
remark: r,
|
||
link: this.genLink(ep.dest, ep.port, ep.forceTls, r, client)
|
||
});
|
||
});
|
||
}
|
||
return result;
|
||
}
|
||
|
||
genInboundLinks(remark = '', remarkModel = '-ieo') {
|
||
let addr = !ObjectUtil.isEmpty(this.listen) && this.listen !== "0.0.0.0" ? this.listen : location.hostname;
|
||
if (this.clients) {
|
||
let links = [];
|
||
this.clients.forEach((client) => {
|
||
this.genAllLinks(remark, remarkModel, client).forEach(l => {
|
||
links.push(l.link);
|
||
})
|
||
});
|
||
return links.join('\r\n');
|
||
} else {
|
||
if (this.protocol == Protocols.SHADOWSOCKS && !this.isSSMultiUser) return this.genSSLink(addr, this.port, 'same', remark);
|
||
if (this.protocol == Protocols.WIREGUARD) {
|
||
let links = [];
|
||
this.settings.peers.forEach((p, index) => {
|
||
links.push(this.getWireguardLink(addr, this.port, remark + remarkModel.charAt(0) + (index + 1), index));
|
||
});
|
||
return links.join('\r\n');
|
||
}
|
||
return '';
|
||
}
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound(
|
||
json.port,
|
||
json.listen,
|
||
json.protocol,
|
||
Inbound.Settings.fromJson(json.protocol, json.settings),
|
||
StreamSettings.fromJson(json.streamSettings),
|
||
json.tag,
|
||
Sniffing.fromJson(json.sniffing),
|
||
json.clientStats
|
||
)
|
||
}
|
||
|
||
toJson() {
|
||
let streamSettings;
|
||
if (this.canEnableStream() || this.stream?.sockopt) {
|
||
streamSettings = this.stream.toJson();
|
||
}
|
||
return {
|
||
port: this.port,
|
||
listen: this.listen,
|
||
protocol: this.protocol,
|
||
settings: this.settings instanceof XrayCommonClass ? this.settings.toJson() : this.settings,
|
||
streamSettings: streamSettings,
|
||
tag: this.tag,
|
||
sniffing: this.sniffing.toJson(),
|
||
clientStats: this.clientStats
|
||
};
|
||
}
|
||
}
|
||
|
||
Inbound.Settings = class extends XrayCommonClass {
|
||
constructor(protocol) {
|
||
super();
|
||
this.protocol = protocol;
|
||
}
|
||
|
||
static getSettings(protocol) {
|
||
switch (protocol) {
|
||
case Protocols.VMESS: return new Inbound.VmessSettings(protocol);
|
||
case Protocols.VLESS: return new Inbound.VLESSSettings(protocol);
|
||
case Protocols.TROJAN: return new Inbound.TrojanSettings(protocol);
|
||
case Protocols.SHADOWSOCKS: return new Inbound.ShadowsocksSettings(protocol);
|
||
case Protocols.TUNNEL: return new Inbound.TunnelSettings(protocol);
|
||
case Protocols.SOCKS: return new Inbound.SocksSettings(protocol);
|
||
case Protocols.HTTP: return new Inbound.HttpSettings(protocol);
|
||
case Protocols.WIREGUARD: return new Inbound.WireguardSettings(protocol);
|
||
default: return null;
|
||
}
|
||
}
|
||
|
||
static fromJson(protocol, json) {
|
||
switch (protocol) {
|
||
case Protocols.VMESS: return Inbound.VmessSettings.fromJson(json);
|
||
case Protocols.VLESS: return Inbound.VLESSSettings.fromJson(json);
|
||
case Protocols.TROJAN: return Inbound.TrojanSettings.fromJson(json);
|
||
case Protocols.SHADOWSOCKS: return Inbound.ShadowsocksSettings.fromJson(json);
|
||
case Protocols.TUNNEL: return Inbound.TunnelSettings.fromJson(json);
|
||
case Protocols.SOCKS: return Inbound.SocksSettings.fromJson(json);
|
||
case Protocols.HTTP: return Inbound.HttpSettings.fromJson(json);
|
||
case Protocols.WIREGUARD: return Inbound.WireguardSettings.fromJson(json);
|
||
default: return null;
|
||
}
|
||
}
|
||
|
||
toJson() {
|
||
return {};
|
||
}
|
||
};
|
||
|
||
Inbound.VmessSettings = class extends Inbound.Settings {
|
||
constructor(protocol,
|
||
vmesses = [new Inbound.VmessSettings.VMESS()]) {
|
||
super(protocol);
|
||
this.vmesses = vmesses;
|
||
}
|
||
|
||
indexOfVmessById(id) {
|
||
return this.vmesses.findIndex(VMESS => VMESS.id === id);
|
||
}
|
||
|
||
addVmess(VMESS) {
|
||
if (this.indexOfVmessById(VMESS.id) >= 0) {
|
||
return false;
|
||
}
|
||
this.vmesses.push(VMESS);
|
||
}
|
||
|
||
delVmess(VMESS) {
|
||
const i = this.indexOfVmessById(VMESS.id);
|
||
if (i >= 0) {
|
||
this.vmesses.splice(i, 1);
|
||
}
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound.VmessSettings(
|
||
Protocols.VMESS,
|
||
json.clients.map(client => Inbound.VmessSettings.VMESS.fromJson(client)),
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
clients: Inbound.VmessSettings.toJsonArray(this.vmesses),
|
||
};
|
||
}
|
||
};
|
||
|
||
Inbound.VmessSettings.VMESS = class extends XrayCommonClass {
|
||
constructor(
|
||
id = RandomUtil.randomUUID(),
|
||
security = USERS_SECURITY.AUTO,
|
||
email = RandomUtil.randomLowerAndNum(8),
|
||
limitIp = 0,
|
||
speedLimit = 0, //
|
||
totalGB = 0,
|
||
expiryTime = 0,
|
||
enable = true,
|
||
tgId = '',
|
||
subId = RandomUtil.randomLowerAndNum(16),
|
||
comment = '',
|
||
reset = 0,
|
||
created_at = undefined,
|
||
updated_at = undefined
|
||
) {
|
||
super();
|
||
this.id = id;
|
||
this.security = security;
|
||
this.email = email;
|
||
this.limitIp = limitIp;
|
||
this.speedLimit = speedLimit;
|
||
this.totalGB = totalGB;
|
||
this.expiryTime = expiryTime;
|
||
this.enable = enable;
|
||
this.tgId = tgId;
|
||
this.subId = subId;
|
||
this.comment = comment;
|
||
this.reset = reset;
|
||
this.created_at = created_at;
|
||
this.updated_at = updated_at;
|
||
}
|
||
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound.VmessSettings.VMESS(
|
||
json.id,
|
||
json.security,
|
||
json.email,
|
||
json.limitIp,
|
||
json.speedLimit ?? 0,
|
||
json.totalGB,
|
||
json.expiryTime,
|
||
json.enable,
|
||
json.tgId,
|
||
json.subId,
|
||
json.comment,
|
||
json.reset,
|
||
json.created_at,
|
||
json.updated_at,
|
||
);
|
||
}
|
||
get _expiryTime() {
|
||
if (this.expiryTime === 0 || this.expiryTime === "") {
|
||
return null;
|
||
}
|
||
if (this.expiryTime < 0) {
|
||
return this.expiryTime / -86400000;
|
||
}
|
||
return moment(this.expiryTime);
|
||
}
|
||
|
||
set _expiryTime(t) {
|
||
if (t == null || t === "") {
|
||
this.expiryTime = 0;
|
||
} else {
|
||
this.expiryTime = t.valueOf();
|
||
}
|
||
}
|
||
get _totalGB() {
|
||
return NumberFormatter.toFixed(this.totalGB / SizeFormatter.ONE_GB, 2);
|
||
}
|
||
|
||
set _totalGB(gb) {
|
||
this.totalGB = NumberFormatter.toFixed(gb * SizeFormatter.ONE_GB, 0);
|
||
}
|
||
|
||
};
|
||
|
||
Inbound.VLESSSettings = class extends Inbound.Settings {
|
||
constructor(
|
||
protocol,
|
||
vlesses = [new Inbound.VLESSSettings.VLESS()],
|
||
decryption = "none",
|
||
encryption = "",
|
||
fallbacks = [],
|
||
) {
|
||
super(protocol);
|
||
this.vlesses = vlesses;
|
||
this.decryption = decryption;
|
||
this.encryption = encryption;
|
||
this.fallbacks = fallbacks;
|
||
this.selectedAuth = "X25519, not Post-Quantum";
|
||
}
|
||
|
||
addFallback() {
|
||
this.fallbacks.push(new Inbound.VLESSSettings.Fallback());
|
||
}
|
||
|
||
delFallback(index) {
|
||
this.fallbacks.splice(index, 1);
|
||
}
|
||
|
||
// decryption should be set to static value
|
||
static fromJson(json = {}) {
|
||
const obj = new Inbound.VLESSSettings(
|
||
Protocols.VLESS,
|
||
(json.clients || []).map(client => Inbound.VLESSSettings.VLESS.fromJson(client)),
|
||
json.decryption,
|
||
json.encryption,
|
||
Inbound.VLESSSettings.Fallback.fromJson(json.fallbacks || [])
|
||
);
|
||
obj.selectedAuth = json.selectedAuth || "X25519, not Post-Quantum";
|
||
return obj;
|
||
}
|
||
|
||
toJson() {
|
||
const json = {
|
||
clients: Inbound.VLESSSettings.toJsonArray(this.vlesses),
|
||
};
|
||
|
||
if (this.decryption) {
|
||
json.decryption = this.decryption;
|
||
}
|
||
|
||
if (this.encryption) {
|
||
json.encryption = this.encryption;
|
||
}
|
||
|
||
if (this.fallbacks && this.fallbacks.length > 0) {
|
||
json.fallbacks = Inbound.VLESSSettings.toJsonArray(this.fallbacks);
|
||
}
|
||
if (this.selectedAuth) {
|
||
json.selectedAuth = this.selectedAuth;
|
||
}
|
||
|
||
return json;
|
||
}
|
||
|
||
|
||
};
|
||
|
||
Inbound.VLESSSettings.VLESS = class extends XrayCommonClass {
|
||
constructor(
|
||
id = RandomUtil.randomUUID(),
|
||
flow = '',
|
||
email = RandomUtil.randomLowerAndNum(8),
|
||
limitIp = 0,
|
||
speedLimit = 0,
|
||
totalGB = 0,
|
||
expiryTime = 0,
|
||
enable = true,
|
||
tgId = '',
|
||
subId = RandomUtil.randomLowerAndNum(16),
|
||
comment = '',
|
||
reset = 0,
|
||
created_at = undefined,
|
||
updated_at = undefined
|
||
) {
|
||
super();
|
||
this.id = id;
|
||
this.flow = flow;
|
||
this.email = email;
|
||
this.limitIp = limitIp;
|
||
this.speedLimit = speedLimit;
|
||
this.totalGB = totalGB;
|
||
this.expiryTime = expiryTime;
|
||
this.enable = enable;
|
||
this.tgId = tgId;
|
||
this.subId = subId;
|
||
this.comment = comment;
|
||
this.reset = reset;
|
||
this.created_at = created_at;
|
||
this.updated_at = updated_at;
|
||
}
|
||
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound.VLESSSettings.VLESS(
|
||
json.id,
|
||
json.flow,
|
||
json.email,
|
||
json.limitIp,
|
||
json.speedLimit ?? 0,
|
||
json.totalGB,
|
||
json.expiryTime,
|
||
json.enable,
|
||
json.tgId,
|
||
json.subId,
|
||
json.comment,
|
||
json.reset,
|
||
json.created_at,
|
||
json.updated_at,
|
||
);
|
||
}
|
||
|
||
get _expiryTime() {
|
||
if (this.expiryTime === 0 || this.expiryTime === "") {
|
||
return null;
|
||
}
|
||
if (this.expiryTime < 0) {
|
||
return this.expiryTime / -86400000;
|
||
}
|
||
return moment(this.expiryTime);
|
||
}
|
||
|
||
set _expiryTime(t) {
|
||
if (t == null || t === "") {
|
||
this.expiryTime = 0;
|
||
} else {
|
||
this.expiryTime = t.valueOf();
|
||
}
|
||
}
|
||
get _totalGB() {
|
||
return NumberFormatter.toFixed(this.totalGB / SizeFormatter.ONE_GB, 2);
|
||
}
|
||
|
||
set _totalGB(gb) {
|
||
this.totalGB = NumberFormatter.toFixed(gb * SizeFormatter.ONE_GB, 0);
|
||
}
|
||
};
|
||
Inbound.VLESSSettings.Fallback = class extends XrayCommonClass {
|
||
constructor(name = "", alpn = '', path = '', dest = '', xver = 0) {
|
||
super();
|
||
this.name = name;
|
||
this.alpn = alpn;
|
||
this.path = path;
|
||
this.dest = dest;
|
||
this.xver = xver;
|
||
}
|
||
|
||
toJson() {
|
||
let xver = this.xver;
|
||
if (!Number.isInteger(xver)) {
|
||
xver = 0;
|
||
}
|
||
return {
|
||
name: this.name,
|
||
alpn: this.alpn,
|
||
path: this.path,
|
||
dest: this.dest,
|
||
xver: xver,
|
||
}
|
||
}
|
||
|
||
static fromJson(json = []) {
|
||
const fallbacks = [];
|
||
for (let fallback of json) {
|
||
fallbacks.push(new Inbound.VLESSSettings.Fallback(
|
||
fallback.name,
|
||
fallback.alpn,
|
||
fallback.path,
|
||
fallback.dest,
|
||
fallback.xver,
|
||
))
|
||
}
|
||
return fallbacks;
|
||
}
|
||
};
|
||
|
||
Inbound.TrojanSettings = class extends Inbound.Settings {
|
||
constructor(protocol,
|
||
trojans = [new Inbound.TrojanSettings.Trojan()],
|
||
fallbacks = [],) {
|
||
super(protocol);
|
||
this.trojans = trojans;
|
||
this.fallbacks = fallbacks;
|
||
}
|
||
|
||
addFallback() {
|
||
this.fallbacks.push(new Inbound.TrojanSettings.Fallback());
|
||
}
|
||
|
||
delFallback(index) {
|
||
this.fallbacks.splice(index, 1);
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound.TrojanSettings(
|
||
Protocols.TROJAN,
|
||
json.clients.map(client => Inbound.TrojanSettings.Trojan.fromJson(client)),
|
||
Inbound.TrojanSettings.Fallback.fromJson(json.fallbacks),);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
clients: Inbound.TrojanSettings.toJsonArray(this.trojans),
|
||
fallbacks: Inbound.TrojanSettings.toJsonArray(this.fallbacks)
|
||
};
|
||
}
|
||
};
|
||
|
||
Inbound.TrojanSettings.Trojan = class extends XrayCommonClass {
|
||
constructor(
|
||
password = RandomUtil.randomSeq(10),
|
||
email = RandomUtil.randomLowerAndNum(8),
|
||
limitIp = 0,
|
||
speedLimit = 0,
|
||
totalGB = 0,
|
||
expiryTime = 0,
|
||
enable = true,
|
||
tgId = '',
|
||
subId = RandomUtil.randomLowerAndNum(16),
|
||
comment = '',
|
||
reset = 0,
|
||
created_at = undefined,
|
||
updated_at = undefined
|
||
) {
|
||
super();
|
||
this.password = password;
|
||
this.email = email;
|
||
this.limitIp = limitIp;
|
||
this.speedLimit = speedLimit;
|
||
this.totalGB = totalGB;
|
||
this.expiryTime = expiryTime;
|
||
this.enable = enable;
|
||
this.tgId = tgId;
|
||
this.subId = subId;
|
||
this.comment = comment;
|
||
this.reset = reset;
|
||
this.created_at = created_at;
|
||
this.updated_at = updated_at;
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
password: this.password,
|
||
email: this.email,
|
||
limitIp: this.limitIp,
|
||
speedLimit: this.speedLimit,
|
||
totalGB: this.totalGB,
|
||
expiryTime: this.expiryTime,
|
||
enable: this.enable,
|
||
tgId: this.tgId,
|
||
subId: this.subId,
|
||
comment: this.comment,
|
||
reset: this.reset,
|
||
created_at: this.created_at,
|
||
updated_at: this.updated_at,
|
||
};
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound.TrojanSettings.Trojan(
|
||
json.password,
|
||
json.email,
|
||
json.limitIp,
|
||
json.speedLimit ?? 0,
|
||
json.totalGB,
|
||
json.expiryTime,
|
||
json.enable,
|
||
json.tgId,
|
||
json.subId,
|
||
json.comment,
|
||
json.reset,
|
||
json.created_at,
|
||
json.updated_at,
|
||
);
|
||
}
|
||
|
||
get _expiryTime() {
|
||
if (this.expiryTime === 0 || this.expiryTime === "") {
|
||
return null;
|
||
}
|
||
if (this.expiryTime < 0) {
|
||
return this.expiryTime / -86400000;
|
||
}
|
||
return moment(this.expiryTime);
|
||
}
|
||
|
||
set _expiryTime(t) {
|
||
if (t == null || t === "") {
|
||
this.expiryTime = 0;
|
||
} else {
|
||
this.expiryTime = t.valueOf();
|
||
}
|
||
}
|
||
get _totalGB() {
|
||
return NumberFormatter.toFixed(this.totalGB / SizeFormatter.ONE_GB, 2);
|
||
}
|
||
|
||
set _totalGB(gb) {
|
||
this.totalGB = NumberFormatter.toFixed(gb * SizeFormatter.ONE_GB, 0);
|
||
}
|
||
|
||
};
|
||
|
||
Inbound.TrojanSettings.Fallback = class extends XrayCommonClass {
|
||
constructor(name = "", alpn = '', path = '', dest = '', xver = 0) {
|
||
super();
|
||
this.name = name;
|
||
this.alpn = alpn;
|
||
this.path = path;
|
||
this.dest = dest;
|
||
this.xver = xver;
|
||
}
|
||
|
||
toJson() {
|
||
let xver = this.xver;
|
||
if (!Number.isInteger(xver)) {
|
||
xver = 0;
|
||
}
|
||
return {
|
||
name: this.name,
|
||
alpn: this.alpn,
|
||
path: this.path,
|
||
dest: this.dest,
|
||
xver: xver,
|
||
}
|
||
}
|
||
|
||
static fromJson(json = []) {
|
||
const fallbacks = [];
|
||
for (let fallback of json) {
|
||
fallbacks.push(new Inbound.TrojanSettings.Fallback(
|
||
fallback.name,
|
||
fallback.alpn,
|
||
fallback.path,
|
||
fallback.dest,
|
||
fallback.xver,
|
||
))
|
||
}
|
||
return fallbacks;
|
||
}
|
||
};
|
||
|
||
Inbound.ShadowsocksSettings = class extends Inbound.Settings {
|
||
constructor(protocol,
|
||
method = SSMethods.BLAKE3_AES_256_GCM,
|
||
password = RandomUtil.randomShadowsocksPassword(),
|
||
network = 'tcp,udp',
|
||
shadowsockses = [new Inbound.ShadowsocksSettings.Shadowsocks()],
|
||
ivCheck = false,
|
||
) {
|
||
super(protocol);
|
||
this.method = method;
|
||
this.password = password;
|
||
this.network = network;
|
||
this.shadowsockses = shadowsockses;
|
||
this.ivCheck = ivCheck;
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound.ShadowsocksSettings(
|
||
Protocols.SHADOWSOCKS,
|
||
json.method,
|
||
json.password,
|
||
json.network,
|
||
json.clients.map(client => Inbound.ShadowsocksSettings.Shadowsocks.fromJson(client)),
|
||
json.ivCheck,
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
method: this.method,
|
||
password: this.password,
|
||
network: this.network,
|
||
clients: Inbound.ShadowsocksSettings.toJsonArray(this.shadowsockses),
|
||
ivCheck: this.ivCheck,
|
||
};
|
||
}
|
||
};
|
||
|
||
Inbound.ShadowsocksSettings.Shadowsocks = class extends XrayCommonClass {
|
||
constructor(
|
||
method = '',
|
||
password = RandomUtil.randomShadowsocksPassword(),
|
||
email = RandomUtil.randomLowerAndNum(8),
|
||
limitIp = 0,
|
||
speedLimit = 0,
|
||
totalGB = 0,
|
||
expiryTime = 0,
|
||
enable = true,
|
||
tgId = '',
|
||
subId = RandomUtil.randomLowerAndNum(16),
|
||
comment = '',
|
||
reset = 0,
|
||
created_at = undefined,
|
||
updated_at = undefined
|
||
) {
|
||
super();
|
||
this.method = method;
|
||
this.password = password;
|
||
this.email = email;
|
||
this.limitIp = limitIp;
|
||
this.speedLimit = speedLimit;
|
||
this.totalGB = totalGB;
|
||
this.expiryTime = expiryTime;
|
||
this.enable = enable;
|
||
this.tgId = tgId;
|
||
this.subId = subId;
|
||
this.comment = comment;
|
||
this.reset = reset;
|
||
this.created_at = created_at;
|
||
this.updated_at = updated_at;
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
method: this.method,
|
||
password: this.password,
|
||
email: this.email,
|
||
limitIp: this.limitIp,
|
||
speedLimit: this.speedLimit,
|
||
totalGB: this.totalGB,
|
||
expiryTime: this.expiryTime,
|
||
enable: this.enable,
|
||
tgId: this.tgId,
|
||
subId: this.subId,
|
||
comment: this.comment,
|
||
reset: this.reset,
|
||
created_at: this.created_at,
|
||
updated_at: this.updated_at,
|
||
};
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound.ShadowsocksSettings.Shadowsocks(
|
||
json.method,
|
||
json.password,
|
||
json.email,
|
||
json.limitIp,
|
||
json.speedLimit ?? 0,
|
||
json.totalGB,
|
||
json.expiryTime,
|
||
json.enable,
|
||
json.tgId,
|
||
json.subId,
|
||
json.comment,
|
||
json.reset,
|
||
json.created_at,
|
||
json.updated_at,
|
||
);
|
||
}
|
||
|
||
get _expiryTime() {
|
||
if (this.expiryTime === 0 || this.expiryTime === "") {
|
||
return null;
|
||
}
|
||
if (this.expiryTime < 0) {
|
||
return this.expiryTime / -86400000;
|
||
}
|
||
return moment(this.expiryTime);
|
||
}
|
||
|
||
set _expiryTime(t) {
|
||
if (t == null || t === "") {
|
||
this.expiryTime = 0;
|
||
} else {
|
||
this.expiryTime = t.valueOf();
|
||
}
|
||
}
|
||
get _totalGB() {
|
||
return NumberFormatter.toFixed(this.totalGB / SizeFormatter.ONE_GB, 2);
|
||
}
|
||
|
||
set _totalGB(gb) {
|
||
this.totalGB = NumberFormatter.toFixed(gb * SizeFormatter.ONE_GB, 0);
|
||
}
|
||
|
||
};
|
||
|
||
Inbound.TunnelSettings = class extends Inbound.Settings {
|
||
constructor(
|
||
protocol,
|
||
address,
|
||
port,
|
||
portMap = [],
|
||
network = 'tcp,udp',
|
||
followRedirect = false
|
||
) {
|
||
super(protocol);
|
||
this.address = address;
|
||
this.port = port;
|
||
this.portMap = portMap;
|
||
this.network = network;
|
||
this.followRedirect = followRedirect;
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound.TunnelSettings(
|
||
Protocols.TUNNEL,
|
||
json.address,
|
||
json.port,
|
||
XrayCommonClass.toHeaders(json.portMap),
|
||
json.network,
|
||
json.followRedirect,
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
address: this.address,
|
||
port: this.port,
|
||
portMap: XrayCommonClass.toV2Headers(this.portMap, false),
|
||
network: this.network,
|
||
followRedirect: this.followRedirect,
|
||
};
|
||
}
|
||
};
|
||
|
||
Inbound.SocksSettings = class extends Inbound.Settings {
|
||
constructor(protocol, auth = 'password', accounts = [new Inbound.SocksSettings.SocksAccount()], udp = false, ip = '127.0.0.1') {
|
||
super(protocol);
|
||
this.auth = auth;
|
||
this.accounts = accounts;
|
||
this.udp = udp;
|
||
this.ip = ip;
|
||
}
|
||
|
||
addAccount(account) {
|
||
this.accounts.push(account);
|
||
}
|
||
|
||
delAccount(index) {
|
||
this.accounts.splice(index, 1);
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
let accounts;
|
||
if (json.auth === 'password') {
|
||
accounts = json.accounts.map(
|
||
account => Inbound.SocksSettings.SocksAccount.fromJson(account)
|
||
)
|
||
}
|
||
return new Inbound.SocksSettings(
|
||
Protocols.SOCKS,
|
||
json.auth,
|
||
accounts,
|
||
json.udp,
|
||
json.ip,
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
auth: this.auth,
|
||
accounts: this.auth === 'password' ? this.accounts.map(account => account.toJson()) : undefined,
|
||
udp: this.udp,
|
||
ip: this.ip,
|
||
};
|
||
}
|
||
};
|
||
Inbound.SocksSettings.SocksAccount = class extends XrayCommonClass {
|
||
constructor(user = RandomUtil.randomSeq(10), pass = RandomUtil.randomSeq(10)) {
|
||
super();
|
||
this.user = user;
|
||
this.pass = pass;
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound.SocksSettings.SocksAccount(json.user, json.pass);
|
||
}
|
||
};
|
||
|
||
Inbound.HttpSettings = class extends Inbound.Settings {
|
||
constructor(
|
||
protocol,
|
||
accounts = [new Inbound.HttpSettings.HttpAccount()],
|
||
allowTransparent = false,
|
||
) {
|
||
super(protocol);
|
||
this.accounts = accounts;
|
||
this.allowTransparent = allowTransparent;
|
||
}
|
||
|
||
addAccount(account) {
|
||
this.accounts.push(account);
|
||
}
|
||
|
||
delAccount(index) {
|
||
this.accounts.splice(index, 1);
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound.HttpSettings(
|
||
Protocols.HTTP,
|
||
json.accounts.map(account => Inbound.HttpSettings.HttpAccount.fromJson(account)),
|
||
json.allowTransparent,
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
accounts: Inbound.HttpSettings.toJsonArray(this.accounts),
|
||
allowTransparent: this.allowTransparent,
|
||
};
|
||
}
|
||
};
|
||
|
||
Inbound.HttpSettings.HttpAccount = class extends XrayCommonClass {
|
||
constructor(user = RandomUtil.randomSeq(10), pass = RandomUtil.randomSeq(10)) {
|
||
super();
|
||
this.user = user;
|
||
this.pass = pass;
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound.HttpSettings.HttpAccount(json.user, json.pass);
|
||
}
|
||
};
|
||
|
||
Inbound.WireguardSettings = class extends XrayCommonClass {
|
||
constructor(
|
||
protocol,
|
||
mtu = 1420,
|
||
secretKey = Wireguard.generateKeypair().privateKey,
|
||
peers = [new Inbound.WireguardSettings.Peer()],
|
||
noKernelTun = false
|
||
) {
|
||
super(protocol);
|
||
this.mtu = mtu;
|
||
this.secretKey = secretKey;
|
||
this.pubKey = secretKey.length > 0 ? Wireguard.generateKeypair(secretKey).publicKey : '';
|
||
this.peers = peers;
|
||
this.noKernelTun = noKernelTun;
|
||
}
|
||
|
||
addPeer() {
|
||
this.peers.push(new Inbound.WireguardSettings.Peer(null, null, '', ['10.0.0.' + (this.peers.length + 2)]));
|
||
}
|
||
|
||
delPeer(index) {
|
||
this.peers.splice(index, 1);
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound.WireguardSettings(
|
||
Protocols.WIREGUARD,
|
||
json.mtu,
|
||
json.secretKey,
|
||
json.peers.map(peer => Inbound.WireguardSettings.Peer.fromJson(peer)),
|
||
json.noKernelTun,
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
return {
|
||
mtu: this.mtu ?? undefined,
|
||
secretKey: this.secretKey,
|
||
peers: Inbound.WireguardSettings.Peer.toJsonArray(this.peers),
|
||
noKernelTun: this.noKernelTun,
|
||
};
|
||
}
|
||
};
|
||
|
||
Inbound.WireguardSettings.Peer = class extends XrayCommonClass {
|
||
constructor(privateKey, publicKey, psk = '', allowedIPs = ['10.0.0.2/32'], keepAlive = 0) {
|
||
super();
|
||
this.privateKey = privateKey
|
||
this.publicKey = publicKey;
|
||
if (!this.publicKey) {
|
||
[this.publicKey, this.privateKey] = Object.values(Wireguard.generateKeypair())
|
||
}
|
||
this.psk = psk;
|
||
allowedIPs.forEach((a, index) => {
|
||
if (a.length > 0 && !a.includes('/')) allowedIPs[index] += '/32';
|
||
})
|
||
this.allowedIPs = allowedIPs;
|
||
this.keepAlive = keepAlive;
|
||
}
|
||
|
||
static fromJson(json = {}) {
|
||
return new Inbound.WireguardSettings.Peer(
|
||
json.privateKey,
|
||
json.publicKey,
|
||
json.preSharedKey,
|
||
json.allowedIPs,
|
||
json.keepAlive
|
||
);
|
||
}
|
||
|
||
toJson() {
|
||
this.allowedIPs.forEach((a, index) => {
|
||
if (a.length > 0 && !a.includes('/')) this.allowedIPs[index] += '/32';
|
||
});
|
||
return {
|
||
privateKey: this.privateKey,
|
||
publicKey: this.publicKey,
|
||
preSharedKey: this.psk.length > 0 ? this.psk : undefined,
|
||
allowedIPs: this.allowedIPs,
|
||
keepAlive: this.keepAlive ?? undefined,
|
||
};
|
||
}
|
||
};
|