Merge pull request #15 from Keivan-sf/13-extract-config-name

13 extract config name
This commit is contained in:
Keivan
2025-07-27 16:28:28 +03:30
committed by GitHub
6 changed files with 88 additions and 46 deletions

View File

@@ -4,16 +4,17 @@ V2ray URI parser for xray core
Currently supports: `vless` Currently supports: `vless`
``` ```
V2ray URI parser Parses V2ray URI and generates JSON config for xray
Usage: v2parser [OPTIONS] <URI> Usage: v2parser [OPTIONS] <uri>
Arguments: Arguments:
<URI> <uri> V2ray URI to parse
Options: Options:
--socksport <socksport> --socksport <PORT> Optional SOCKS proxy port for inbound
--httpport <httpport> --httpport <PORT> Optional HTTP proxy port for inbound
--get-name Only print the config name
-h, --help Print help -h, --help Print help
-V, --version Print version -V, --version Print version
``` ```

View File

@@ -186,6 +186,7 @@ pub struct Config {
#[derive(Serialize, Deserialize)] #[derive(Serialize, Deserialize)]
#[allow(non_snake_case)] #[allow(non_snake_case)]
pub struct RawData { pub struct RawData {
pub remarks: String,
pub security: Option<String>, pub security: Option<String>,
pub vnext_security: Option<String>, pub vnext_security: Option<String>,
pub sni: Option<String>, pub sni: Option<String>,

View File

@@ -1,20 +1,50 @@
mod parser; use clap::{value_parser, Arg, Command};
use clap::Parser;
pub mod config_models; pub mod config_models;
mod parser;
pub mod utils; pub mod utils;
#[derive(Parser)] fn main() {
#[command(author ,version = "0.1.1", about = "V2ray URI parser", long_about = None)] let matches = Command::new("v2ray-uri-parser")
struct Cli { .version("0.1.1")
uri: String, .about("Parses V2ray URI and generates JSON config for xray")
#[arg(long, value_name = "socksport")] .arg(
socksport: Option<u16>, Arg::new("uri")
#[arg(long, value_name = "httpport")] .help("V2ray URI to parse")
httpport: Option<u16>, .required(true)
.index(1),
)
.arg(
Arg::new("socksport")
.long("socksport")
.help("Optional SOCKS proxy port for inbound")
.value_name("PORT")
.value_parser(value_parser!(u16)),
)
.arg(
Arg::new("httpport")
.long("httpport")
.help("Optional HTTP proxy port for inbound")
.value_name("PORT")
.value_parser(value_parser!(u16)),
)
.arg(
Arg::new("get_name")
.long("get-name")
.help("Only print the config name")
.action(clap::ArgAction::SetTrue),
)
.get_matches();
let uri = matches.get_one::<String>("uri").unwrap();
let socksport = matches.get_one::<u16>("socksport").copied();
let httpport = matches.get_one::<u16>("httpport").copied();
let get_name = matches.get_flag("get_name");
if get_name {
print!("{}", parser::get_name(uri));
return;
} }
fn main() { let json_config = parser::create_json_config(uri, socksport, httpport);
let cli = Cli::parse();
let json_config = parser::create_json_config(&cli.uri, cli.socksport, cli.httpport);
println!("{}", json_config); println!("{}", json_config);
} }

View File

@@ -9,6 +9,11 @@ mod uri_identifier;
mod vless; mod vless;
mod vmess; mod vmess;
pub fn get_name(uri: &str) -> String {
let (_, data, _) = get_uri_data(uri);
return data.remarks;
}
pub fn create_json_config(uri: &str, socks_port: Option<u16>, http_port: Option<u16>) -> String { pub fn create_json_config(uri: &str, socks_port: Option<u16>, http_port: Option<u16>) -> String {
let config = create_config(uri, socks_port, http_port); let config = create_config(uri, socks_port, http_port);
let serialized = serde_json::to_string(&config).unwrap(); let serialized = serde_json::to_string(&config).unwrap();
@@ -34,25 +39,7 @@ pub fn create_config(
} }
pub fn create_outbound_object(uri: &str) -> config_models::Outbound { pub fn create_outbound_object(uri: &str) -> config_models::Outbound {
let protocol = uri_identifier::get_uri_protocol(uri); let (name, data, outbound_settings) = get_uri_data(uri);
let (name, data, outbound_settings): (String, RawData, OutboundSettings) = match protocol {
Some(uri_identifier::Protocols::Vless) => {
let d = vless::data::get_data(uri);
let s = vless::create_outbound_settings(&d);
(String::from("vless"), d, s)
}
Some(uri_identifier::Protocols::Vmess) => {
let d = vmess::data::get_data(uri);
let s = vmess::create_outbound_settings(&d);
(String::from("vmess"), d, s)
}
Some(_) => {
panic!("The protocol was recognized but is not supported yet");
}
None => {
panic!("The protocol is not supported");
}
};
let network_type = data.r#type.clone().unwrap_or(String::from("")); let network_type = data.r#type.clone().unwrap_or(String::from(""));
let allow_insecure = data.allowInsecure == Some(String::from("true")) let allow_insecure = data.allowInsecure == Some(String::from("true"))
@@ -161,3 +148,25 @@ pub fn create_outbound_object(uri: &str) -> config_models::Outbound {
return outbound; return outbound;
} }
fn get_uri_data(uri: &str) -> (String, RawData, OutboundSettings) {
let protocol = uri_identifier::get_uri_protocol(uri);
return match protocol {
Some(uri_identifier::Protocols::Vless) => {
let d = vless::data::get_data(uri);
let s = vless::create_outbound_settings(&d);
(String::from("vless"), d, s)
}
Some(uri_identifier::Protocols::Vmess) => {
let d = vmess::data::get_data(uri);
let s = vmess::create_outbound_settings(&d);
(String::from("vmess"), d, s)
}
Some(_) => {
panic!("The protocol was recognized but is not supported yet");
}
None => {
panic!("The protocol is not supported");
}
};
}

View File

@@ -6,14 +6,14 @@ use http::Uri;
pub fn get_data(uri: &str) -> RawData { pub fn get_data(uri: &str) -> RawData {
let data = uri.split_once("vless://").unwrap().1; let data = uri.split_once("vless://").unwrap().1;
let query_and_name = uri.split_once("?").unwrap().1; let query_and_name = uri.split_once("?").unwrap().1;
let raw_query = query_and_name let (raw_query, name) = query_and_name
.split_once("#") .split_once("#")
.unwrap_or((query_and_name, "")) .unwrap_or((query_and_name, ""));
.0;
let parsed_address = parse_vless_address(data.split_once("?").unwrap().0); let parsed_address = parse_vless_address(data.split_once("?").unwrap().0);
let query: Vec<(&str, &str)> = querystring::querify(raw_query); let query: Vec<(&str, &str)> = querystring::querify(raw_query);
return RawData { return RawData {
remarks: String::from(name),
uuid: Some(parsed_address.uuid), uuid: Some(parsed_address.uuid),
port: Some(parsed_address.port), port: Some(parsed_address.port),
address: Some(parsed_address.address), address: Some(parsed_address.address),

View File

@@ -19,6 +19,7 @@ fn get_raw_data_from_base64(decoded_base64: &Vec<u8>) -> RawData {
let json = serde_json::from_str::<Value>(json_str).unwrap(); let json = serde_json::from_str::<Value>(json_str).unwrap();
return RawData { return RawData {
remarks: get_str_field(&json, "ps").unwrap_or(String::from("")),
uuid: get_str_field(&json, "id"), uuid: get_str_field(&json, "id"),
port: get_str_field(&json, "port") port: get_str_field(&json, "port")
.and_then(|s| Some(s.parse::<u16>().expect("port is not a number"))), .and_then(|s| Some(s.parse::<u16>().expect("port is not a number"))),
@@ -65,14 +66,14 @@ fn get_raw_data_from_uri(uri: &str) -> RawData {
let data = uri.split_once("vmess://").unwrap().1; let data = uri.split_once("vmess://").unwrap().1;
let query_and_name = uri.split_once("?").unwrap().1; let query_and_name = uri.split_once("?").unwrap().1;
let raw_query = query_and_name let (raw_query, name) = query_and_name
.split_once("#") .split_once("#")
.unwrap_or((query_and_name, "")) .unwrap_or((query_and_name, ""));
.0;
let parsed_address = parse_vmess_address(data.split_once("?").unwrap().0); let parsed_address = parse_vmess_address(data.split_once("?").unwrap().0);
let query: Vec<(&str, &str)> = querystring::querify(raw_query); let query: Vec<(&str, &str)> = querystring::querify(raw_query);
return RawData { return RawData {
remarks: String::from(name),
uuid: Some(parsed_address.uuid), uuid: Some(parsed_address.uuid),
port: Some(parsed_address.port), port: Some(parsed_address.port),
address: Some(parsed_address.address), address: Some(parsed_address.address),