adding ip filter and api key filter
This commit is contained in:
		@@ -13,7 +13,7 @@ http-body-util = "0.1.2"
 | 
			
		||||
hyper = { version = "1.5.2", features = ["http1", "server"] }
 | 
			
		||||
hyper-util = { version = "0.1", features = ["http1", "server", "tokio"] }
 | 
			
		||||
 | 
			
		||||
log = { version = "0.4.22", features=["kv"]}
 | 
			
		||||
log = { version = "0.4.25", features=["kv"]}
 | 
			
		||||
 | 
			
		||||
[lib]
 | 
			
		||||
name = "servme"
 | 
			
		||||
 
 | 
			
		||||
@@ -1,4 +1,5 @@
 | 
			
		||||
use crate::{config::ServerConfig, server::Server};
 | 
			
		||||
use std::sync::Arc;
 | 
			
		||||
 | 
			
		||||
pub struct ServerBuilder {
 | 
			
		||||
    pub config: ServerConfig,
 | 
			
		||||
@@ -11,9 +12,28 @@ impl ServerBuilder {
 | 
			
		||||
        self
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    pub fn private_ips(mut self) -> Self {
 | 
			
		||||
        self.config.ips_filter = true;
 | 
			
		||||
 | 
			
		||||
        self.config.private_ips = true;
 | 
			
		||||
        self
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    pub fn ips(mut self, ips: Vec<String>) -> Self {
 | 
			
		||||
        self.config.ips_filter = true;
 | 
			
		||||
 | 
			
		||||
        self.config.ips = ips;
 | 
			
		||||
        self
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    pub fn api_key(mut self, api_key: &str) -> Self {
 | 
			
		||||
        self.config.api_key = Some(api_key.to_string());
 | 
			
		||||
        self
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    pub fn build(self) -> Server {
 | 
			
		||||
        Server {
 | 
			
		||||
            config: self.config,
 | 
			
		||||
            config: Arc::new(self.config),
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,6 +1,12 @@
 | 
			
		||||
pub struct ServerConfig {
 | 
			
		||||
    pub ip: String,
 | 
			
		||||
    pub port: u16,
 | 
			
		||||
 | 
			
		||||
    pub ips_filter: bool,
 | 
			
		||||
    pub private_ips: bool,
 | 
			
		||||
    pub ips: Vec<String>,
 | 
			
		||||
 | 
			
		||||
    pub api_key: Option<String>,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
impl Default for ServerConfig {
 | 
			
		||||
@@ -8,6 +14,12 @@ impl Default for ServerConfig {
 | 
			
		||||
        ServerConfig {
 | 
			
		||||
            ip: "127.0.0.1".to_string(),
 | 
			
		||||
            port: 8080,
 | 
			
		||||
 | 
			
		||||
            ips_filter: false,
 | 
			
		||||
            private_ips: false,
 | 
			
		||||
            ips: Vec::new(),
 | 
			
		||||
 | 
			
		||||
            api_key: None,
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
@@ -1,14 +1,20 @@
 | 
			
		||||
use crate::{builder::ServerBuilder, config::ServerConfig};
 | 
			
		||||
use http::HeaderValue;
 | 
			
		||||
use http_body_util::Full;
 | 
			
		||||
use hyper::{body::Incoming, server::conn::http1, service::service_fn, Request, Response};
 | 
			
		||||
use hyper_util::rt::{TokioIo, TokioTimer};
 | 
			
		||||
use log::error;
 | 
			
		||||
use std::{convert::Infallible, future::Future, net::SocketAddr, sync::Arc};
 | 
			
		||||
use std::{
 | 
			
		||||
    convert::Infallible,
 | 
			
		||||
    future::Future,
 | 
			
		||||
    net::{IpAddr, SocketAddr},
 | 
			
		||||
    sync::Arc,
 | 
			
		||||
};
 | 
			
		||||
use tokio::{net::TcpListener, spawn};
 | 
			
		||||
use tokio_util::bytes::Bytes;
 | 
			
		||||
 | 
			
		||||
pub struct Server {
 | 
			
		||||
    pub config: ServerConfig,
 | 
			
		||||
    pub config: Arc<ServerConfig>,
 | 
			
		||||
}
 | 
			
		||||
 | 
			
		||||
impl Server {
 | 
			
		||||
@@ -30,14 +36,40 @@ impl Server {
 | 
			
		||||
        let handler = Arc::new(handler);
 | 
			
		||||
 | 
			
		||||
        loop {
 | 
			
		||||
            let (tcp, _) = listener.accept().await.unwrap();
 | 
			
		||||
            let (tcp, client_addr) = listener.accept().await.unwrap();
 | 
			
		||||
            let io = TokioIo::new(tcp);
 | 
			
		||||
 | 
			
		||||
            let config = Arc::clone(&self.config);
 | 
			
		||||
            if self.config.ips_filter && !self.is_ip_authorized(&client_addr.ip()) {
 | 
			
		||||
                continue;
 | 
			
		||||
            }
 | 
			
		||||
 | 
			
		||||
            let handler = Arc::clone(&handler);
 | 
			
		||||
            spawn(async move {
 | 
			
		||||
                if let Err(error) = http1::Builder::new()
 | 
			
		||||
                    .timer(TokioTimer::new())
 | 
			
		||||
                    .serve_connection(io, service_fn(move |req| handler(req)))
 | 
			
		||||
                    .serve_connection(
 | 
			
		||||
                        io,
 | 
			
		||||
                        service_fn(move |req| {
 | 
			
		||||
                            let config = Arc::clone(&config);
 | 
			
		||||
                            let handler = Arc::clone(&handler);
 | 
			
		||||
 | 
			
		||||
                            async move {
 | 
			
		||||
                                if let Some(ref token) = config.api_key {
 | 
			
		||||
                                    if req.headers().get("X-API-Key")
 | 
			
		||||
                                        != Some(&HeaderValue::from_str(token).unwrap())
 | 
			
		||||
                                    {
 | 
			
		||||
                                        return Ok(Response::builder()
 | 
			
		||||
                                            .status(401)
 | 
			
		||||
                                            .body(Full::new(Bytes::from("Unauthorized")))
 | 
			
		||||
                                            .unwrap());
 | 
			
		||||
                                    }
 | 
			
		||||
                                }
 | 
			
		||||
 | 
			
		||||
                                handler(req).await
 | 
			
		||||
                            }
 | 
			
		||||
                        }),
 | 
			
		||||
                    )
 | 
			
		||||
                    .await
 | 
			
		||||
                {
 | 
			
		||||
                    error!(error = error.to_string().as_str();
 | 
			
		||||
@@ -47,4 +79,25 @@ impl Server {
 | 
			
		||||
            });
 | 
			
		||||
        }
 | 
			
		||||
    }
 | 
			
		||||
 | 
			
		||||
    fn is_ip_authorized(&self, ip: &IpAddr) -> bool {
 | 
			
		||||
        if self.config.private_ips {
 | 
			
		||||
            let is_private = match ip {
 | 
			
		||||
                IpAddr::V4(ip4) => ip4.is_private(),
 | 
			
		||||
                IpAddr::V6(_) => false,
 | 
			
		||||
            };
 | 
			
		||||
 | 
			
		||||
            if is_private {
 | 
			
		||||
                return true;
 | 
			
		||||
            }
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        let ips = &self.config.ips;
 | 
			
		||||
        if ips.is_empty() {
 | 
			
		||||
            return true;
 | 
			
		||||
        }
 | 
			
		||||
 | 
			
		||||
        ips.iter()
 | 
			
		||||
            .any(|authorized_ip| &ip.to_string() == authorized_ip)
 | 
			
		||||
    }
 | 
			
		||||
}
 | 
			
		||||
 
 | 
			
		||||
		Reference in New Issue
	
	Block a user