mirror of
https://github.com/house-of-vanity/v2-uri-parser.git
synced 2025-12-16 15:07:53 +00:00
285 lines
8.5 KiB
Rust
285 lines
8.5 KiB
Rust
use querystring;
|
|
use serde::{Deserialize, Serialize};
|
|
use std::process::exit;
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
struct Person {
|
|
name: String,
|
|
age: u8,
|
|
phones: Vec<String>,
|
|
}
|
|
|
|
struct VlessQuery {
|
|
security: String,
|
|
sni: String,
|
|
fp: String,
|
|
pbk: String,
|
|
sid: String,
|
|
r#type: String,
|
|
flow: String,
|
|
path: String,
|
|
encryption: String,
|
|
header_type: String,
|
|
host: String,
|
|
seed: String,
|
|
quic_security: String,
|
|
r#key: String,
|
|
mode: String,
|
|
service_name: String,
|
|
slpn: String,
|
|
spx: String,
|
|
}
|
|
|
|
struct VlessAddress {
|
|
uuid: String,
|
|
address: String,
|
|
port: u16,
|
|
}
|
|
|
|
pub struct VlessData {
|
|
query: VlessQuery,
|
|
address_data: VlessAddress,
|
|
}
|
|
|
|
// Outbound structs
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
struct VlessUser {
|
|
id: String,
|
|
encryption: String,
|
|
flow: String,
|
|
level: u8,
|
|
}
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
struct VlessServerObject {
|
|
address: String,
|
|
port: u16,
|
|
users: Vec<VlessUser>,
|
|
}
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
struct VlessOutboundSettings {
|
|
vnext: Vec<VlessServerObject>,
|
|
}
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
enum OutboundSettings {
|
|
Vless(VlessOutboundSettings),
|
|
}
|
|
|
|
#[allow(non_snake_case)]
|
|
struct TlsSettings {
|
|
allowInsecure: bool,
|
|
certificates: u8,
|
|
serverName: String,
|
|
// u8 is a dummy type here
|
|
alpn: u8,
|
|
enableSessionResumption: bool,
|
|
disableSystemRoot: bool,
|
|
minVersion: String,
|
|
maxVersion: String,
|
|
cipherSuites: String,
|
|
preferServerCipherSuites: bool,
|
|
fingerprint: String,
|
|
rejectUnknownSni: bool,
|
|
pinnedPeerCertificateChainSha256: u8,
|
|
pinnedPeerCertificatePublicKeySha256: u8,
|
|
}
|
|
|
|
#[allow(non_snake_case)]
|
|
struct StreamSettings {
|
|
network: String,
|
|
security: String,
|
|
tlsSettings: TlsSettings,
|
|
}
|
|
|
|
#[derive(Serialize, Deserialize)]
|
|
pub struct Outbound {
|
|
settings: OutboundSettings,
|
|
protocol: String,
|
|
tag: String,
|
|
}
|
|
|
|
fn create_outbound_object(data: VlessData) -> Outbound {
|
|
return Outbound {
|
|
protocol: String::from("vless"),
|
|
tag: String::from("proxy"),
|
|
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,
|
|
level: 0,
|
|
}],
|
|
}],
|
|
}),
|
|
};
|
|
}
|
|
|
|
pub fn get_vless_data(uri: &str) -> 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 VlessData {
|
|
query: parsed_query,
|
|
address_data: parsed_address,
|
|
};
|
|
}
|
|
|
|
fn parse_vless_address(raw_data: &str) -> 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 VlessAddress {
|
|
uuid,
|
|
address,
|
|
port,
|
|
};
|
|
}
|
|
|
|
fn parse_vless_query(raw_query: &str) -> VlessQuery {
|
|
let query: Vec<(&str, &str)> = querystring::querify(raw_query);
|
|
|
|
let a = 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 log_vless_outbound() {
|
|
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);
|
|
let outbound_object = create_outbound_object(data);
|
|
let serialized = serde_json::to_string(&outbound_object).unwrap();
|
|
println!("serialized = {}", serialized);
|
|
}
|
|
}
|