use std::process::exit; use querystring; #[derive(PartialEq, Eq)] pub struct VlessQuery { security: String, sni: String, fp: String, pbk: String, sid: String, r#type: String, flow: String, path: String, } pub struct VlessUUIDAndHost { uuid: String, host: String, port: u16, } pub fn get_vless_data(uri: &str) { 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_data(query); println!("{0}", parsed_query.flow); } pub fn parse_vless_uui_and_host(raw_data: &str) -> VlessUUIDAndHost { let (uuid, address): (String, &str) = match raw_data.split_once("@") { None => { println!("Wrong vless format, no `@` in the authentication"); exit(0); } Some(data) => (String::from(data.0), data.1), }; let (host, port): (String, u16) = match 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("Port is not a number"), ), }; return VlessUUIDAndHost { uuid, host, port }; } pub fn parse_vless_query_data(raw_query: &str) -> VlessQuery { let query = querystring::querify(raw_query); let a = VlessQuery { path: query .iter() .find(|q| String::from(q.0) == String::from("path")) .unwrap_or(&("", "")) .1 .to_string(), pbk: query .iter() .find(|q| String::from(q.0) == String::from("pbk")) .unwrap_or(&("", "")) .1 .to_string(), security: query .iter() .find(|q| String::from(q.0) == String::from("security")) .unwrap_or(&("", "")) .1 .to_string(), sid: query .iter() .find(|q| String::from(q.0) == String::from("sid")) .unwrap_or(&("", "")) .1 .to_string(), flow: query .iter() .find(|q| String::from(q.0) == String::from("flow")) .unwrap_or(&("", "")) .1 .to_string(), sni: query .iter() .find(|q| String::from(q.0) == String::from("sni")) .unwrap_or(&("", "")) .1 .to_string(), fp: query .iter() .find(|q| String::from(q.0) == String::from("fp")) .unwrap_or(&("", "")) .1 .to_string(), r#type: query .iter() .find(|q| String::from(q.0) == String::from("type")) .unwrap_or(&("", "")) .1 .to_string(), }; return a; } #[cfg(test)] mod tests { use super::*; #[test] fn vless_test() { let v = "vless://4d2c3e35-749d-52e3-bdb6-3f3f4950c183@tre.test.one:2053?security=reality&sni=bench.sh&fp=chrome&pbk=7xhH4b_VkliBxGulljcyPOH-bYUA2dl-XAdZAsfhk04&sid=6ba85179e30d4fc2&type=tcp&flow=xtls-rprx-vision#test-name"; get_vless_data(v); } #[test] fn parse_vless_query() { 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_data(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.r#flow, "xtls-rprx-vision"); assert_eq!(parsed_query.path, "/"); } #[test] fn parse_vless_query_with_defaults() { let query = ""; let parsed_query = parse_vless_query_data(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, ""); } #[test] fn parse_vless_host() { let raw_host = "uu0id@127.0.0.1:3012"; let parsed = parse_vless_uui_and_host(raw_host); assert_eq!(parsed.host, "127.0.0.1"); assert_eq!(parsed.port, 3012); assert_eq!(parsed.uuid, "uu0id"); } }