Files
v2ray-proxy/src/parser/vless/mod.rs
2023-10-23 19:13:16 +03:30

265 lines
10 KiB
Rust

use querystring;
mod models;
use crate::parser::config_models::*;
use std::process::exit;
pub fn create_outbound_object(data: models::VlessData) -> Outbound {
return Outbound {
protocol: String::from("vless"),
tag: String::from("proxy"),
streamSettings: StreamSettings {
network: data.query.r#type.clone(),
security: data.query.security.clone(),
tlsSettings: if data.query.security == String::from("tls") {
Some(TlsSettings {
rejectUnknownSni: None,
enableSessionResumption: None,
minVersion: None,
maxVersion: None,
cipherSuites: None,
disableSystemRoot: None,
preferServerCipherSuites: None,
fingerprint: Some(String::from("")),
serverName: Some(data.query.sni.clone()),
allowInsecure: Some(false),
})
} else {
None
},
wsSettings: if data.query.r#type == String::from("ws") {
Some(WsSettings {
path: Some(String::from("")),
acceptProxyProtocol: None,
})
} else {
None
},
tcpSettings: if data.query.r#type == String::from("tcp") {
Some(TCPSettings {
header: Some(NonHeaderObject {
r#type: String::from("none"),
}),
acceptProxyProtocol: None,
})
} else {
None
},
realitySettings: if data.query.security == String::from("reality") {
Some(RealitySettings {
publicKey: data.query.pbk,
serverName: data.query.sni.clone(),
shortId: data.query.sid,
spiderX: String::from(""),
fingerprint: data.query.fp,
})
} else {
None
},
grpcSettings: if data.query.r#type == String::from("grpc") {
Some(GRPCSettings {
multiMode: false,
serviceName: data.query.service_name,
})
} else {
None
},
quicSettings: if data.query.r#type == String::from("quic") {
Some(QuicSettings {
header: Some(NonHeaderObject {
r#type: String::from("none"),
}),
security: String::from("none"),
key: String::from(""),
})
} else {
None
},
},
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: if data.query.encryption.len() > 0 {
data.query.encryption
} else {
String::from("none")
},
level: 0,
}],
}],
}),
};
}
pub fn get_vless_data(uri: &str) -> models::VlessData {
let data = uri.split_once("vless://").unwrap().1;
let query_and_name = uri.split_once("?").unwrap().1;
let query = query_and_name
.split_once("#")
.unwrap_or((query_and_name, ""))
.0;
let parsed_query = parse_vless_query(query);
let parsed_address = parse_vless_address(data.split_once("?").unwrap().0);
return models::VlessData {
query: parsed_query,
address_data: parsed_address,
};
}
fn parse_vless_address(raw_data: &str) -> models::VlessAddress {
let (uuid, raw_address): (String, &str) = match raw_data.split_once("@") {
None => {
println!("Wrong vless format, no `@` found in the address");
exit(0);
}
Some(data) => (String::from(data.0), data.1),
};
let (address, port): (String, u16) = match raw_address.split_once(":") {
None => {
println!("Wrong vless format, no `:` found in the address");
exit(0);
}
Some(data) => (
String::from(data.0),
data.1
.parse::<u16>()
.expect("Wrong vless format, port is not a number"),
),
};
return models::VlessAddress {
uuid,
address,
port,
};
}
fn parse_vless_query(raw_query: &str) -> models::VlessQuery {
let query: Vec<(&str, &str)> = querystring::querify(raw_query);
let a = models::VlessQuery {
path: get_parameter_value(&query, "path"),
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"),
};
return a;
}
fn get_parameter_value(query: &Vec<(&str, &str)>, param: &str) -> String {
return query
.iter()
.find(|q| String::from(q.0) == String::from(param))
.unwrap_or(&("", ""))
.1
.to_string();
}
#[cfg(test)]
mod tests {
use super::*;
#[test]
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, "xtls-rprx-vision");
assert_eq!(data.query.security, "reality");
assert_eq!(data.query.r#type, "tcp");
}
#[test]
fn parse_vless_query_data() {
let query = "security=reality&sni=bench.sh&fp=chrome&pbk=7xhH4b_VkliBxGulljcyPOH-bYUA2dl-XAdZAsfhk04&sid=6ba85179e30d4fc2&type=tcp&flow=xtls-rprx-vision&path=/";
let parsed_query = parse_vless_query(query);
assert_eq!(parsed_query.sni, "bench.sh");
assert_eq!(parsed_query.security, "reality");
assert_eq!(parsed_query.fp, "chrome");
assert_eq!(
parsed_query.pbk,
"7xhH4b_VkliBxGulljcyPOH-bYUA2dl-XAdZAsfhk04"
);
assert_eq!(parsed_query.sid, "6ba85179e30d4fc2");
assert_eq!(parsed_query.r#type, "tcp");
assert_eq!(parsed_query.flow, "xtls-rprx-vision");
assert_eq!(parsed_query.path, "/");
assert_eq!(parsed_query.encryption, "");
assert_eq!(parsed_query.header_type, "");
assert_eq!(parsed_query.host, "");
assert_eq!(parsed_query.seed, "");
assert_eq!(parsed_query.quic_security, "");
assert_eq!(parsed_query.key, "");
assert_eq!(parsed_query.mode, "");
assert_eq!(parsed_query.service_name, "");
assert_eq!(parsed_query.slpn, "");
assert_eq!(parsed_query.spx, "");
}
#[test]
fn parse_vless_query_with_defaults() {
let query = "";
let parsed_query = parse_vless_query(query);
assert_eq!(parsed_query.sni, "");
assert_eq!(parsed_query.security, "");
assert_eq!(parsed_query.fp, "");
assert_eq!(parsed_query.pbk, "");
assert_eq!(parsed_query.sid, "");
assert_eq!(parsed_query.r#type, "");
assert_eq!(parsed_query.r#flow, "");
assert_eq!(parsed_query.path, "");
assert_eq!(parsed_query.encryption, "");
assert_eq!(parsed_query.header_type, "");
assert_eq!(parsed_query.host, "");
assert_eq!(parsed_query.seed, "");
assert_eq!(parsed_query.quic_security, "");
assert_eq!(parsed_query.key, "");
assert_eq!(parsed_query.mode, "");
assert_eq!(parsed_query.service_name, "");
assert_eq!(parsed_query.slpn, "");
assert_eq!(parsed_query.spx, "");
}
#[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");
assert_eq!(parsed.port, 3012);
assert_eq!(parsed.uuid, "uu0id");
}
#[test]
fn create_outbound_for_tcp_reality() {
let v = "vless://3d2c2r05-y739-51e3-bd86-3f3f4950c183@tr.deet23ngdell.com:1818?security=reality&encryption=none&pbk=7xhH8b_VkliBxgulljcyPOH-bYoA2dl-XAdZAsfhk04&headerType=none&fp=chrome&type=tcp&flow=xtls-rprx-vision&sni=bench.sh&sid=6bt85979e30d4fc2#%F0%9F%87%B9%F0%9F%87%B7+H";
let data = get_vless_data(v);
let outbound_object = create_outbound_object(data);
let serialized = serde_json::to_string(&outbound_object).unwrap();
assert_eq!(
serialized,
r#"{"settings":{"vnext":[{"address":"tr.deet23ngdell.com","port":1818,"users":[{"id":"3d2c2r05-y739-51e3-bd86-3f3f4950c183","encryption":"none","flow":"xtls-rprx-vision","level":0}]}]},"streamSettings":{"network":"tcp","security":"reality","tlsSettings":null,"wsSettings":null,"tcpSettings":{"header":{"type":"none"},"acceptProxyProtocol":null},"realitySettings":{"fingerprint":"chrome","serverName":"bench.sh","publicKey":"7xhH8b_VkliBxgulljcyPOH-bYoA2dl-XAdZAsfhk04","shortId":"6bt85979e30d4fc2","spiderX":""},"grpcSettings":null,"quicSettings":null},"protocol":"vless","tag":"proxy"}"#
);
}
}