/rusocks

A socks4(a)/5 server

Primary LanguageRustApache License 2.0Apache-2.0

rusocks

A socks4(a)/5 server

features:

  • socks4/4a protocol

    • connect
    • bind
  • socks5 protocol

    • no authentication
    • username/password
  • socks5 command

    • connect
    • bind
    • udp associate

example

use std::net::SocketAddr;

use async_trait::async_trait;
use rusocks::{
    address::Address,
    error::Error,
    socks4::{reply::Socks4Reply, request::Socks4Request, Socks4Handler},
    socks5::{method::Socks5Method, reply::Socks5Reply, request::Socks5Request, Socks5Handler},
    Socks,
};
use tokio::{
    io::{self, AsyncReadExt, AsyncWriteExt},
    net::{TcpListener, TcpStream},
    select, time,
};

#[tokio::main]
async fn main() {
    // curl -x socks4://127.0.0.1:1080 http://127.0.0.1:8080
    // curl -x socks5://127.0.0.1:1080 http://127.0.0.1:8080
    let listener = TcpListener::bind("127.0.0.1:1080").await.unwrap();

    while let Ok((mut stream, _)) = listener.accept().await {
        tokio::spawn(async move {
            let peer_addr = stream.peer_addr().unwrap();
            let local_addr = stream.local_addr().unwrap();

            let handler = Handler {};
            let mut socks = Socks::from_stream(&mut stream, peer_addr, local_addr, handler)
                .await
                .unwrap();

            match socks.accept(&mut stream).await {
                Ok(_) => {
                    println!("success");
                }
                Err(e) => {
                    println!("error: {}", e);
                }
            }
        });
    }
}

struct Handler {}

#[async_trait]
impl Socks4Handler for Handler {
    type ConnectStream = TcpStream;

    async fn connect(&self, address: &Address) -> Result<(Self::ConnectStream, SocketAddr), Error> {
        let stream = TcpStream::connect((address.domain(), address.port())).await?;
        let bind_addr = stream.local_addr()?;

        Ok((stream, bind_addr))
    }

    async fn bind<S>(
        &self,
        request: &mut Socks4Request<S>,
        bind_addr: &Address,
    ) -> Result<(), Error>
    where
        S: AsyncReadExt + AsyncWriteExt + Unpin + Send,
    {
        let listener = TcpListener::bind((bind_addr.domain(), bind_addr.port())).await?;
        let bind_addr = listener.local_addr()?.clone();
        request.reply(Socks4Reply::Granted, bind_addr).await?;

        let (mut connect, peer_addr) = listener.accept().await?;
        // 异步处理连接
        println!("Accepted connection from {:?}", peer_addr);
        io::copy_bidirectional(&mut request.stream, &mut connect)
            .await
            .unwrap();

        Ok(())
    }
}

#[async_trait]
impl Socks5Handler for Handler {
    type ConnectStream = TcpStream;

    async fn negotiate_method(&self, _methods: &[Socks5Method]) -> Result<Socks5Method, Error> {
        Ok(Socks5Method::None)
    }

    async fn auth_by_user_pass(&self, username: &str, password: &str) -> Result<bool, Error> {
        println!("username: {}, password: {}", username, password);
        Ok(false)
    }

    async fn connect(&self, address: &Address) -> Result<(Self::ConnectStream, SocketAddr), Error> {
        let stream = TcpStream::connect((address.domain(), address.port())).await?;
        let bind_addr = stream.local_addr()?;

        Ok((stream, bind_addr))
    }

    async fn bind<S>(
        &self,
        request: &mut Socks5Request<S>,
        bind_addr: &Address,
    ) -> Result<(), Error>
    where
        S: AsyncReadExt + AsyncWriteExt + Unpin + Send,
    {
        let listener = TcpListener::bind((bind_addr.domain(), bind_addr.port())).await?;
        let bind_addr = listener.local_addr()?.clone();

        request.reply(Socks5Reply::Succeeded, bind_addr).await?;

        select! {
            // 超时关闭
            _ = time::sleep(time::Duration::from_secs(20)) => {
                println!("Shutdown signal received. Stopping the listener...");
                return Ok(());
            }
            Ok((mut connect, peer_addr)) = listener.accept() => {
                println!("Accepted connection from {:?}", peer_addr);
                request.reply(Socks5Reply::Succeeded, peer_addr).await?;
                io::copy_bidirectional(&mut request.stream, &mut connect).await?;
            }
        }

        Ok(())
    }
}