use querystring; use serde::{Deserialize, Serialize}; use std::process::exit; #[derive(Serialize, Deserialize)] struct Person { name: String, age: u8, phones: Vec, } 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, } #[derive(Serialize, Deserialize)] struct VlessOutboundSettings { vnext: Vec, } #[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::() .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); } }