Files
v2ray-proxy/src/parser/vless/mod.rs

294 lines
11 KiB
Rust
Raw Normal View History

2023-10-16 19:04:37 +03:30
use querystring;
2023-10-17 17:27:24 +03:30
mod models;
2023-11-07 16:36:30 +03:30
use crate::config_models::*;
2025-07-10 20:43:48 +03:30
use http::Uri;
2023-10-10 19:28:53 +03:30
use std::process::exit;
2023-10-20 17:14:56 +03:30
pub fn create_outbound_object(data: models::VlessData) -> Outbound {
let network_type = data.query.r#type.clone().unwrap_or(String::from(""));
2023-10-14 11:30:03 +03:30
return Outbound {
protocol: String::from("vless"),
tag: String::from("proxy"),
2023-10-19 10:04:12 +03:30
streamSettings: StreamSettings {
2023-10-19 11:53:28 +03:30
network: data.query.r#type.clone(),
2023-10-19 10:04:12 +03:30
security: data.query.security.clone(),
tlsSettings: if network_type == String::from("tls") {
2023-10-19 10:04:12 +03:30
Some(TlsSettings {
alpn: data.query.alpn.map(|alpn| vec![alpn]),
2023-10-19 10:04:12 +03:30
rejectUnknownSni: None,
enableSessionResumption: None,
minVersion: None,
maxVersion: None,
cipherSuites: None,
disableSystemRoot: None,
preferServerCipherSuites: None,
fingerprint: data.query.fp.clone(),
serverName: data.query.sni.clone(),
2023-10-19 10:04:12 +03:30
allowInsecure: Some(false),
})
} else {
None
},
wsSettings: if network_type == String::from("ws") {
2023-10-19 11:53:28 +03:30
Some(WsSettings {
Host: data.query.host,
path: data.query.path,
2023-10-19 11:53:28 +03:30
acceptProxyProtocol: None,
})
} else {
None
},
tcpSettings: if network_type == String::from("tcp") {
2023-10-19 11:57:44 +03:30
Some(TCPSettings {
2025-07-23 21:00:07 +03:30
header: Some(TCPHeader {
r#type: Some(data.query.header_type.unwrap_or(String::from("none"))),
2023-10-19 11:57:44 +03:30
}),
acceptProxyProtocol: None,
})
} else {
None
},
realitySettings: if network_type == String::from("reality") {
2023-10-19 15:26:41 +03:30
Some(RealitySettings {
publicKey: data.query.pbk,
serverName: data.query.sni.clone(),
shortId: data.query.sid,
spiderX: Some(String::from("")),
fingerprint: data.query.fp.clone(),
2023-10-19 15:26:41 +03:30
})
} else {
None
},
grpcSettings: if network_type == String::from("grpc") {
2023-10-21 11:42:12 +03:30
Some(GRPCSettings {
authority: data.query.authority,
multiMode: Some(false),
2023-10-21 11:42:12 +03:30
serviceName: data.query.service_name,
})
} else {
None
},
quicSettings: if network_type == String::from("quic") {
2023-10-23 19:13:16 +03:30
Some(QuicSettings {
header: Some(NonHeaderObject {
r#type: Some(String::from("none")),
2023-10-23 19:13:16 +03:30
}),
security: Some(String::from("none")),
key: Some(String::from("")),
2023-10-23 19:13:16 +03:30
})
} else {
None
},
2023-10-19 10:04:12 +03:30
},
2023-10-14 11:30:03 +03:30
settings: OutboundSettings::Vless(VlessOutboundSettings {
vnext: vec![VlessServerObject {
port: data.address_data.port,
address: data.address_data.address,
users: vec![VlessUser {
id: data.address_data.uuid,
flow: data.query.flow,
encryption: data.query.encryption.unwrap_or(String::from("none")),
level: Some(0),
}],
}],
2023-10-14 11:30:03 +03:30
}),
};
}
2023-10-17 17:27:24 +03:30
pub fn get_vless_data(uri: &str) -> models::VlessData {
2023-10-10 16:21:20 +03:30
let data = uri.split_once("vless://").unwrap().1;
2023-10-10 19:00:17 +03:30
let query_and_name = uri.split_once("?").unwrap().1;
let query = query_and_name
.split_once("#")
.unwrap_or((query_and_name, ""))
.0;
2023-10-11 17:24:15 +03:30
let parsed_query = parse_vless_query(query);
let parsed_address = parse_vless_address(data.split_once("?").unwrap().0);
2023-10-17 17:27:24 +03:30
return models::VlessData {
query: parsed_query,
address_data: parsed_address,
};
2023-10-10 16:21:20 +03:30
}
2023-10-17 17:27:24 +03:30
fn parse_vless_address(raw_data: &str) -> models::VlessAddress {
let (uuid, raw_address): (String, &str) = match raw_data.split_once("@") {
2023-10-10 19:28:53 +03:30
None => {
2023-10-11 20:56:21 +03:30
println!("Wrong vless format, no `@` found in the address");
2023-10-10 19:28:53 +03:30
exit(0);
}
Some(data) => (String::from(data.0), data.1),
};
2025-06-27 02:27:26 +03:30
let address_wo_slash = raw_address.strip_suffix("/").unwrap_or(raw_address);
2025-07-10 20:43:48 +03:30
let parsed = address_wo_slash.parse::<Uri>().unwrap();
2023-10-17 17:27:24 +03:30
return models::VlessAddress {
uuid,
2025-07-10 20:43:48 +03:30
address: parsed.host().unwrap().to_string(),
port: parsed.port().unwrap().as_u16(),
};
2023-10-10 19:28:53 +03:30
}
2023-10-17 17:27:24 +03:30
fn parse_vless_query(raw_query: &str) -> models::VlessQuery {
2023-10-11 17:36:53 +03:30
let query: Vec<(&str, &str)> = querystring::querify(raw_query);
2023-10-11 17:17:34 +03:30
2023-10-17 17:27:24 +03:30
let a = models::VlessQuery {
alpn: get_parameter_value(&query, "alpn").and_then(|s| {
urlencoding::decode(&s)
.ok()
.map(|decoded| decoded.into_owned())
}),
path: get_parameter_value(&query, "path").and_then(|s| {
urlencoding::decode(&s)
.ok()
.map(|decoded| decoded.into_owned())
}),
authority: get_parameter_value(&query, "authority").and_then(|s| {
urlencoding::decode(&s)
.ok()
.map(|decoded| decoded.into_owned())
}),
2023-10-11 17:36:53 +03:30
pbk: get_parameter_value(&query, "pbk"),
security: get_parameter_value(&query, "security"),
sid: get_parameter_value(&query, "sid"),
flow: get_parameter_value(&query, "flow"),
sni: get_parameter_value(&query, "sni"),
fp: get_parameter_value(&query, "fp"),
r#type: get_parameter_value(&query, "type"),
encryption: get_parameter_value(&query, "encryption"),
header_type: get_parameter_value(&query, "headerType"),
host: get_parameter_value(&query, "host"),
seed: get_parameter_value(&query, "seed"),
quic_security: get_parameter_value(&query, "quicSecurity"),
key: get_parameter_value(&query, "key"),
mode: get_parameter_value(&query, "mode"),
service_name: get_parameter_value(&query, "serviceName"),
slpn: get_parameter_value(&query, "slpn"),
spx: get_parameter_value(&query, "spx"),
2023-10-10 16:21:20 +03:30
};
return a;
}
fn get_parameter_value(query: &Vec<(&str, &str)>, param: &str) -> Option<String> {
2023-10-11 17:36:53 +03:30
return query
.iter()
.find(|q| String::from(q.0) == String::from(param))
.map(|q| q.1.to_string());
2023-10-11 17:36:53 +03:30
}
2023-10-10 16:21:20 +03:30
#[cfg(test)]
mod tests {
use super::*;
2025-07-23 21:00:07 +03:30
#[test]
fn vless_tcp_header_test() {
let v = "vless://1010501a-ca9a-479c-84d0-1308d97789b5@104.21.25.109:443?security=reality&alpn=http%2F1.1&encryption=none&pbk=testpub&host=mehr14-n.gowow31220.workers.dev&headerType=http&fp=chrome&spx=spideex&type=tcp&flow=xtls-rprx-vision&sni=mehr14-iran-mehr14-iran-mehr14-iran-mehr14-iran-mehr14-iran.gowow31220.workers.dev&sid=testshort#%E2%AD%90%EF%B8%8F%20Telegram%20%3D%20%40z_v2ray";
let data = create_outbound_object(get_vless_data(v));
assert_eq!(
data.streamSettings
.tcpSettings
.unwrap()
.header
.unwrap()
.r#type,
Some(String::from("http"))
2025-07-23 21:00:07 +03:30
)
}
2023-10-10 16:21:20 +03:30
#[test]
2025-06-27 02:27:26 +03:30
fn vless_test_2() {
let v = "vless://2dc56709-sdfs-sdfs-2234-128904@nwarne.fast-ip.com:80/?type=ws&encryption=none&host=Shuposipet.com&path=%2Fde%3Fed%3D1048#[test]@test";
let data = get_vless_data(v);
assert_eq!(data.address_data.address, "nwarne.fast-ip.com");
assert_eq!(data.address_data.uuid, "2dc56709-sdfs-sdfs-2234-128904");
assert_eq!(data.address_data.port, 80);
assert_eq!(data.query.encryption.unwrap(), "none");
assert_eq!(data.query.r#type.unwrap(), "ws");
2025-06-27 02:27:26 +03:30
}
2025-06-27 02:38:53 +03:30
#[test]
2023-10-10 16:21:20 +03:30
fn vless_test() {
let v = "vless://4d2c3e35-749d-52e3-bdb6-3f3f4950c183@tre.test.one:2053?security=reality&type=tcp&flow=xtls-rprx-vision#test-name";
let data = get_vless_data(v);
assert_eq!(data.address_data.address, "tre.test.one");
assert_eq!(
data.address_data.uuid,
"4d2c3e35-749d-52e3-bdb6-3f3f4950c183"
);
assert_eq!(data.address_data.port, 2053);
assert_eq!(data.query.flow.unwrap(), "xtls-rprx-vision");
assert_eq!(data.query.security.unwrap(), "reality");
assert_eq!(data.query.r#type.unwrap(), "tcp");
2023-10-10 16:21:20 +03:30
}
2023-10-10 16:21:20 +03:30
#[test]
2023-10-11 17:24:15 +03:30
fn parse_vless_query_data() {
2025-07-26 11:58:01 +03:30
let query = "security=reality&sni=bench.sh&fp=chrome&pbk=7xhH4b_VkliBxGulljcyPOH-bYUA2dl-XAdZAsfhk04&sid=6ba85179e30d4fc2&type=tcp&flow=xtls-rprx-vision&alpn=http%2F1.1&path=/";
2023-10-11 17:24:15 +03:30
let parsed_query = parse_vless_query(query);
assert_eq!(parsed_query.sni.unwrap(), "bench.sh");
assert_eq!(parsed_query.security.unwrap(), "reality");
assert_eq!(parsed_query.fp.unwrap(), "chrome");
2023-10-10 16:21:20 +03:30
assert_eq!(
parsed_query.pbk.unwrap(),
2023-10-10 16:21:20 +03:30
"7xhH4b_VkliBxGulljcyPOH-bYUA2dl-XAdZAsfhk04"
);
assert_eq!(parsed_query.sid.unwrap(), "6ba85179e30d4fc2");
assert_eq!(parsed_query.r#type.unwrap(), "tcp");
assert_eq!(parsed_query.flow.unwrap(), "xtls-rprx-vision");
assert_eq!(parsed_query.alpn.unwrap(), "http/1.1");
assert_eq!(parsed_query.path.unwrap(), "/");
assert_eq!(parsed_query.encryption, None);
assert_eq!(parsed_query.header_type, None);
assert_eq!(parsed_query.host, None);
assert_eq!(parsed_query.seed, None);
assert_eq!(parsed_query.quic_security, None);
assert_eq!(parsed_query.key, None);
assert_eq!(parsed_query.mode, None);
assert_eq!(parsed_query.service_name, None);
assert_eq!(parsed_query.slpn, None);
assert_eq!(parsed_query.spx, None);
2023-10-10 16:21:20 +03:30
}
#[test]
fn parse_vless_query_with_defaults() {
let query = "";
2023-10-11 17:24:15 +03:30
let parsed_query = parse_vless_query(query);
assert_eq!(parsed_query.sni, None);
assert_eq!(parsed_query.security, None);
assert_eq!(parsed_query.fp, None);
assert_eq!(parsed_query.pbk, None);
assert_eq!(parsed_query.sid, None);
assert_eq!(parsed_query.r#type, None);
assert_eq!(parsed_query.r#flow, None);
assert_eq!(parsed_query.path, None);
assert_eq!(parsed_query.encryption, None);
assert_eq!(parsed_query.header_type, None);
assert_eq!(parsed_query.host, None);
assert_eq!(parsed_query.seed, None);
assert_eq!(parsed_query.quic_security, None);
assert_eq!(parsed_query.key, None);
assert_eq!(parsed_query.mode, None);
assert_eq!(parsed_query.service_name, None);
assert_eq!(parsed_query.slpn, None);
assert_eq!(parsed_query.spx, None);
}
2023-10-10 19:28:53 +03:30
#[test]
fn parse_vless_host() {
let raw_host = "uu0id@127.0.0.1:3012";
let parsed = parse_vless_address(raw_host);
assert_eq!(parsed.address, "127.0.0.1");
2023-10-10 19:28:53 +03:30
assert_eq!(parsed.port, 3012);
assert_eq!(parsed.uuid, "uu0id");
}
2023-10-16 19:04:37 +03:30
2025-07-10 20:43:48 +03:30
#[test]
fn parse_vless_ipv6_host() {
let raw_host = "uu0id@[2a06:98c1:3120::1]:443";
let parsed = parse_vless_address(raw_host);
assert_eq!(parsed.port, 443);
assert_eq!(parsed.address, "[2a06:98c1:3120::1]");
assert_eq!(parsed.uuid, "uu0id");
}
2023-10-10 16:21:20 +03:30
}