Files
shadowsocks-rust/tests/socks5.rs
2019-12-28 17:32:20 +08:00

134 lines
4.0 KiB
Rust

use std::net::{SocketAddr, ToSocketAddrs};
use tokio::{
prelude::*,
runtime::{Builder, Handle},
time::{self, Duration},
};
use shadowsocks::{
config::{Config, ConfigType, Mode, ServerConfig},
crypto::CipherType,
relay::{socks5::Address, tcprelay::client::Socks5Client},
run_local,
run_server,
};
pub struct Socks5TestServer {
local_addr: SocketAddr,
svr_config: Config,
cli_config: Config,
}
impl Socks5TestServer {
pub fn new<S, L>(svr_addr: S, local_addr: L, pwd: &str, method: CipherType, enable_udp: bool) -> Socks5TestServer
where
S: ToSocketAddrs,
L: ToSocketAddrs,
{
let svr_addr = svr_addr.to_socket_addrs().unwrap().next().unwrap();
let local_addr = local_addr.to_socket_addrs().unwrap().next().unwrap();
Socks5TestServer {
local_addr,
svr_config: {
let mut cfg = Config::new(ConfigType::Server);
cfg.server = vec![ServerConfig::basic(svr_addr, pwd.to_owned(), method)];
cfg.mode = if enable_udp { Mode::TcpAndUdp } else { Mode::TcpOnly };
cfg
},
cli_config: {
let mut cfg = Config::new(ConfigType::Socks5Local);
cfg.local = Some(local_addr);
cfg.server = vec![ServerConfig::basic(svr_addr, pwd.to_owned(), method)];
cfg.mode = if enable_udp { Mode::TcpAndUdp } else { Mode::TcpOnly };
cfg
},
}
}
pub fn client_addr(&self) -> &SocketAddr {
&self.local_addr
}
pub async fn run(&self, rt_handle: Handle) {
let svr_cfg = self.svr_config.clone();
tokio::spawn(run_server(svr_cfg, rt_handle.clone()));
let client_cfg = self.cli_config.clone();
tokio::spawn(run_local(client_cfg, rt_handle));
time::delay_for(Duration::from_secs(1)).await;
}
}
#[test]
fn socks5_relay_stream() {
let _ = env_logger::try_init();
const SERVER_ADDR: &str = "127.0.0.1:8100";
const LOCAL_ADDR: &str = "127.0.0.1:8200";
const PASSWORD: &str = "test-password";
const METHOD: CipherType = CipherType::Aes256Cfb;
let mut rt = Builder::new().basic_scheduler().enable_all().build().unwrap();
let rt_handle = rt.handle().clone();
rt.block_on(async move {
let svr = Socks5TestServer::new(SERVER_ADDR, LOCAL_ADDR, PASSWORD, METHOD, false);
svr.run(rt_handle).await;
let mut c = Socks5Client::connect(
Address::DomainNameAddress("www.example.com".to_owned(), 80),
svr.client_addr(),
)
.await
.unwrap();
let req = b"GET / HTTP/1.0\r\nHost: www.example.com\r\nAccept: */*\r\n\r\n";
c.write_all(req).await.unwrap();
c.flush().await.unwrap();
let mut buf = Vec::new();
c.read_to_end(&mut buf).await.unwrap();
println!("Got reply from server: {}", String::from_utf8(buf).unwrap());
});
}
#[test]
fn socks5_relay_aead() {
let _ = env_logger::try_init();
const SERVER_ADDR: &str = "127.0.0.1:8110";
const LOCAL_ADDR: &str = "127.0.0.1:8210";
const PASSWORD: &str = "test-password";
const METHOD: CipherType = CipherType::Aes256Gcm;
let mut rt = Builder::new().basic_scheduler().enable_all().build().unwrap();
let rt_handle = rt.handle().clone();
rt.block_on(async move {
let svr = Socks5TestServer::new(SERVER_ADDR, LOCAL_ADDR, PASSWORD, METHOD, false);
svr.run(rt_handle).await;
let mut c = Socks5Client::connect(
Address::DomainNameAddress("www.example.com".to_owned(), 80),
svr.client_addr(),
)
.await
.unwrap();
let req = b"GET / HTTP/1.0\r\nHost: www.example.com\r\nAccept: */*\r\n\r\n";
c.write_all(req).await.unwrap();
c.flush().await.unwrap();
let mut buf = Vec::new();
c.read_to_end(&mut buf).await.unwrap();
println!("Got reply from server: {}", String::from_utf8(buf).unwrap());
});
}