pub mod compress; pub mod platform; pub mod protos; pub use bytes; use config::Config; pub use futures; pub use protobuf; pub use protos::message as message_proto; pub use protos::rendezvous as rendezvous_proto; use std::{ fs::File, io::{self, BufRead}, net::{IpAddr, Ipv4Addr, SocketAddr, SocketAddrV4}, path::Path, time::{self, SystemTime, UNIX_EPOCH}, }; pub use tokio; pub use tokio_util; pub mod socket_client; pub mod tcp; pub mod udp; pub use env_logger; pub use log; pub mod bytes_codec; #[cfg(feature = "quic")] pub mod quic; pub use anyhow::{self, bail}; pub use futures_util; pub mod config; pub mod fs; pub use lazy_static; #[cfg(not(any(target_os = "android", target_os = "ios")))] pub use mac_address; pub use rand; pub use regex; pub use sodiumoxide; pub use tokio_socks; pub use tokio_socks::IntoTargetAddr; pub use tokio_socks::TargetAddr; pub mod password_security; pub use chrono; pub use directories_next; pub mod keyboard; #[cfg(feature = "quic")] pub type Stream = quic::Connection; #[cfg(not(feature = "quic"))] pub type Stream = tcp::FramedStream; #[inline] pub async fn sleep(sec: f32) { tokio::time::sleep(time::Duration::from_secs_f32(sec)).await; } #[macro_export] macro_rules! allow_err { ($e:expr) => { if let Err(err) = $e { log::debug!( "{:?}, {}:{}:{}:{}", err, module_path!(), file!(), line!(), column!() ); } else { } }; ($e:expr, $($arg:tt)*) => { if let Err(err) = $e { log::debug!( "{:?}, {}, {}:{}:{}:{}", err, format_args!($($arg)*), module_path!(), file!(), line!(), column!() ); } else { } }; } #[inline] pub fn timeout(ms: u64, future: T) -> tokio::time::Timeout { tokio::time::timeout(std::time::Duration::from_millis(ms), future) } pub type ResultType = anyhow::Result; /// Certain router and firewalls scan the packet and if they /// find an IP address belonging to their pool that they use to do the NAT mapping/translation, so here we mangle the ip address pub struct AddrMangle(); impl AddrMangle { pub fn encode(addr: SocketAddr) -> Vec { match addr { SocketAddr::V4(addr_v4) => { let tm = (SystemTime::now() .duration_since(UNIX_EPOCH) .unwrap() .as_micros() as u32) as u128; let ip = u32::from_le_bytes(addr_v4.ip().octets()) as u128; let port = addr.port() as u128; let v = ((ip + tm) << 49) | (tm << 17) | (port + (tm & 0xFFFF)); let bytes = v.to_le_bytes(); let mut n_padding = 0; for i in bytes.iter().rev() { if i == &0u8 { n_padding += 1; } else { break; } } bytes[..(16 - n_padding)].to_vec() } SocketAddr::V6(addr_v6) => { let mut x = addr_v6.ip().octets().to_vec(); let port: [u8; 2] = addr_v6.port().to_le_bytes(); x.push(port[0]); x.push(port[1]); x } } } pub fn decode(bytes: &[u8]) -> SocketAddr { if bytes.len() > 16 { if bytes.len() != 18 { return Config::get_any_listen_addr(false); } #[allow(invalid_value)] let mut tmp: [u8; 2] = unsafe { std::mem::MaybeUninit::uninit().assume_init() }; tmp.copy_from_slice(&bytes[16..]); let port = u16::from_le_bytes(tmp); #[allow(invalid_value)] let mut tmp: [u8; 16] = unsafe { std::mem::MaybeUninit::uninit().assume_init() }; tmp.copy_from_slice(&bytes[..16]); let ip = std::net::Ipv6Addr::from(tmp); return SocketAddr::new(IpAddr::V6(ip), port); } let mut padded = [0u8; 16]; padded[..bytes.len()].copy_from_slice(&bytes); let number = u128::from_le_bytes(padded); let tm = (number >> 17) & (u32::max_value() as u128); let ip = (((number >> 49) - tm) as u32).to_le_bytes(); let port = (number & 0xFFFFFF) - (tm & 0xFFFF); SocketAddr::V4(SocketAddrV4::new( Ipv4Addr::new(ip[0], ip[1], ip[2], ip[3]), port as u16, )) } } pub fn get_version_from_url(url: &str) -> String { let n = url.chars().count(); let a = url .chars() .rev() .enumerate() .filter(|(_, x)| x == &'-') .next() .map(|(i, _)| i); if let Some(a) = a { let b = url .chars() .rev() .enumerate() .filter(|(_, x)| x == &'.') .next() .map(|(i, _)| i); if let Some(b) = b { if a > b { if url .chars() .skip(n - b) .collect::() .parse::() .is_ok() { return url.chars().skip(n - a).collect(); } else { return url.chars().skip(n - a).take(a - b - 1).collect(); } } else { return url.chars().skip(n - a).collect(); } } } "".to_owned() } pub fn gen_version() { use std::io::prelude::*; let mut file = File::create("./src/version.rs").unwrap(); for line in read_lines("Cargo.toml").unwrap() { if let Ok(line) = line { let ab: Vec<&str> = line.split("=").map(|x| x.trim()).collect(); if ab.len() == 2 && ab[0] == "version" { file.write_all(format!("pub const VERSION: &str = {};\n", ab[1]).as_bytes()) .ok(); break; } } } // generate build date let build_date = format!("{}", chrono::Local::now().format("%Y-%m-%d %H:%M")); file.write_all(format!("pub const BUILD_DATE: &str = \"{}\";", build_date).as_bytes()) .ok(); file.sync_all().ok(); } fn read_lines

(filename: P) -> io::Result>> where P: AsRef, { let file = File::open(filename)?; Ok(io::BufReader::new(file).lines()) } pub fn is_valid_custom_id(id: &str) -> bool { regex::Regex::new(r"^[a-zA-Z]\w{5,15}$") .unwrap() .is_match(id) } pub fn get_version_number(v: &str) -> i64 { let mut n = 0; for x in v.split(".") { n = n * 1000 + x.parse::().unwrap_or(0); } n } pub fn get_modified_time(path: &std::path::Path) -> SystemTime { std::fs::metadata(&path) .map(|m| m.modified().unwrap_or(UNIX_EPOCH)) .unwrap_or(UNIX_EPOCH) } pub fn get_created_time(path: &std::path::Path) -> SystemTime { std::fs::metadata(&path) .map(|m| m.created().unwrap_or(UNIX_EPOCH)) .unwrap_or(UNIX_EPOCH) } pub fn get_exe_time() -> SystemTime { std::env::current_exe().map_or(UNIX_EPOCH, |path| { let m = get_modified_time(&path); let c = get_created_time(&path); if m > c { m } else { c } }) } pub fn get_uuid() -> Vec { #[cfg(not(any(target_os = "android", target_os = "ios")))] if let Ok(id) = machine_uid::get() { return id.into(); } Config::get_key_pair().1 } #[inline] pub fn get_time() -> i64 { std::time::SystemTime::now() .duration_since(std::time::UNIX_EPOCH) .map(|d| d.as_millis()) .unwrap_or(0) as _ } #[cfg(test)] mod tests { use super::*; #[test] fn test_mangle() { let addr = SocketAddr::V4(SocketAddrV4::new(Ipv4Addr::new(192, 168, 16, 32), 21116)); assert_eq!(addr, AddrMangle::decode(&AddrMangle::encode(addr))); let addr = "[2001:db8::1]:8080".parse::().unwrap(); assert_eq!(addr, AddrMangle::decode(&AddrMangle::encode(addr))); let addr = "[2001:db8:ff::1111]:80".parse::().unwrap(); assert_eq!(addr, AddrMangle::decode(&AddrMangle::encode(addr))); } #[test] fn test_allow_err() { allow_err!(Err("test err") as Result<(), &str>); allow_err!( Err("test err with msg") as Result<(), &str>, "prompt {}", "failed" ); } } #[inline] pub fn is_ipv4_str(id: &str) -> bool { regex::Regex::new(r"^\d+\.\d+\.\d+\.\d+(:\d+)?$") .unwrap() .is_match(id) } #[inline] pub fn is_ipv6_str(id: &str) -> bool { regex::Regex::new(r"^((([a-fA-F0-9]{1,4}:{1,2})+[a-fA-F0-9]{1,4})|(\[([a-fA-F0-9]{1,4}:{1,2})+[a-fA-F0-9]{1,4}\]:\d+))$") .unwrap() .is_match(id) } #[inline] pub fn is_ip_str(id: &str) -> bool { is_ipv4_str(id) || is_ipv6_str(id) } #[inline] pub fn is_hostname_port_str(id: &str) -> bool { regex::Regex::new(r"^[\-.0-9a-zA-Z]+:\d{1,5}$") .unwrap() .is_match(id) } #[cfg(test)] mod test_lib { use super::*; #[test] fn test_ipv6() { assert_eq!(is_ipv6_str("1:2:3"), true); assert_eq!(is_ipv6_str("[ab:2:3]:12"), true); assert_eq!(is_ipv6_str("[ABEF:2a:3]:12"), true); assert_eq!(is_ipv6_str("[ABEG:2a:3]:12"), false); assert_eq!(is_ipv6_str("1[ab:2:3]:12"), false); assert_eq!(is_ipv6_str("1.1.1.1"), false); assert_eq!(is_ip_str("1.1.1.1"), true); assert_eq!(is_ipv6_str("1:2:"), false); assert_eq!(is_ipv6_str("1:2::0"), true); assert_eq!(is_ipv6_str("[1:2::0]:1"), true); assert_eq!(is_ipv6_str("[1:2::0]:"), false); assert_eq!(is_ipv6_str("1:2::0]:1"), false); } #[test] fn test_hostname_port() { assert_eq!(is_hostname_port_str("a:12"), true); assert_eq!(is_hostname_port_str("a.b.c:12"), true); assert_eq!(is_hostname_port_str("test.com:12"), true); assert_eq!(is_hostname_port_str("1.2.3:12"), true); assert_eq!(is_hostname_port_str("a.b.c:123456"), false); // todo: should we also check for these edge case? // out-of-range port assert_eq!(is_hostname_port_str("test.com:0"), true); assert_eq!(is_hostname_port_str("test.com:98989"), true); // invalid hostname assert_eq!(is_hostname_port_str("---:12"), true); assert_eq!(is_hostname_port_str(".:12"), true); } }