mirror of
https://github.com/shadowsocks/shadowsocks-rust.git
synced 2026-02-09 01:59:16 +08:00
134 lines
4.0 KiB
Rust
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());
|
|
});
|
|
}
|