diff --git a/Cargo.lock b/Cargo.lock index 7b16e84..c5e1ac7 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -70,16 +70,6 @@ dependencies = [ "tower-service", ] -[[package]] -name = "axum-client-ip" -version = "0.3.1" -source = "registry+https://github.com/rust-lang/crates.io-index" -checksum = "ddfb5a3ddd6367075d50629546fb46710584016ae7704cd03b6d41cb5be82e5a" -dependencies = [ - "axum", - "forwarded-header-value", -] - [[package]] name = "axum-core" version = "0.3.2" @@ -1724,7 +1714,6 @@ name = "xssbook" version = "0.0.1" dependencies = [ "axum", - "axum-client-ip", "bytes", "image", "lazy_static", diff --git a/Cargo.toml b/Cargo.toml index 272112f..55be8cf 100644 --- a/Cargo.toml +++ b/Cargo.toml @@ -6,7 +6,6 @@ edition = "2021" [dependencies] tokio = { version = "1.23.0", features = ["full"] } axum = { version = "0.6.4", features = ["headers", "query"] } -axum-client-ip = "0.3.1" tower-http = { version = "0.3.5", features = ["fs"] } tower_governor = "0.0.4" tower-cookies = "0.8.0" diff --git a/public/css/home.css b/public/css/home.css index b68314d..60dc3a5 100644 --- a/public/css/home.css +++ b/public/css/home.css @@ -80,7 +80,7 @@ body { } .icons { - background-image: url(''); + background-image: url('/images/icons.png'); display: inline-block; width: 18px; height: 18px; diff --git a/public/css/main.css b/public/css/main.css index 1926ed9..0314b3f 100644 --- a/public/css/main.css +++ b/public/css/main.css @@ -188,7 +188,7 @@ select { border: 1px solid var(--light); color: var(--extreme); font-size: 15px; - background-image: url(""); + background-image: url("/image/arrow.png"); background-position: right 10px center; background-repeat: no-repeat; background-size: 15px; @@ -291,7 +291,7 @@ footer { cursor: pointer; background-size: 20px; background-position: right; - background-image: url(''); + background-image: url('/image/close.png'); } .hidden { diff --git a/public/css/profile.css b/public/css/profile.css index 1077919..4aff91b 100644 --- a/public/css/profile.css +++ b/public/css/profile.css @@ -67,7 +67,7 @@ body { background-color: var(--secondary); z-index: 10000 !important; text-align: center; - background-image: url(''); + background-image: url('/image/change.png'); cursor: pointer; } diff --git a/public/image/arrow.png b/public/image/arrow.png new file mode 100644 index 0000000..8220278 Binary files /dev/null and b/public/image/arrow.png differ diff --git a/public/image/change.png b/public/image/change.png new file mode 100644 index 0000000..1fcbfea Binary files /dev/null and b/public/image/change.png differ diff --git a/public/image/close.png b/public/image/close.png new file mode 100644 index 0000000..516bf05 Binary files /dev/null and b/public/image/close.png differ diff --git a/public/image/icons.png b/public/image/icons.png new file mode 100644 index 0000000..39a2d74 Binary files /dev/null and b/public/image/icons.png differ diff --git a/public/js/main.js b/public/js/main.js index ffbc1f3..5737173 100644 --- a/public/js/main.js +++ b/public/js/main.js @@ -33,11 +33,11 @@ function remove(id) { } function pfp(id) { - return `` + return `` } function banner(id) { - return `` + return `` } const months = ['Jan', 'Feb', 'Mar', 'Apr', 'May', 'Jun', diff --git a/src/main.rs b/src/main.rs index 74f0a0b..e8f17b0 100644 --- a/src/main.rs +++ b/src/main.rs @@ -5,14 +5,13 @@ use axum::{ response::Response, RequestExt, Router, extract::DefaultBodyLimit, }; -use axum_client_ip::ClientIp; use std::{net::SocketAddr, process::exit, fs}; use tower_cookies::CookieManagerLayer; use tracing::{error, info, metadata::LevelFilter}; use tracing_subscriber::{ filter::filter_fn, prelude::__tracing_subscriber_SubscriberExt, util::SubscriberInitExt, Layer, }; -use types::http::ResponseCode; +use types::{http::ResponseCode, extract::RequestIp}; use crate::api::{pages, image}; @@ -38,7 +37,7 @@ async fn log(mut req: Request, next: Next) -> Response where B: Send + Sync + 'static + HttpBody, { - let Ok(ClientIp(ip)) = req.extract_parts::().await else { + let Ok(RequestIp(ip)) = req.extract_parts::().await else { return next.run(req).await }; @@ -79,7 +78,7 @@ async fn main() { .layer(middleware::from_fn(serve)) .nest("/", pages::router()) .nest("/api", api::router()) - .nest("/image", image::router()) + .nest("/cdn", image::router()) .layer(CookieManagerLayer::new()) .layer(DefaultBodyLimit::max(512_000)); diff --git a/src/types/extract.rs b/src/types/extract.rs index 54f250a..4d7ac51 100644 --- a/src/types/extract.rs +++ b/src/types/extract.rs @@ -1,14 +1,13 @@ -use std::io::{Read, Cursor}; +use std::{io::{Read, Cursor}, net::{IpAddr, SocketAddr}}; use axum::{ async_trait, body::HttpBody, - extract::{FromRequest, FromRequestParts}, + extract::{FromRequest, FromRequestParts, ConnectInfo}, http::{request::Parts, Request}, response::Response, BoxError, RequestExt, }; -use axum_client_ip::ClientIp; use bytes::Bytes; use image::{io::Reader, ImageFormat, DynamicImage}; use serde::de::DeserializeOwned; @@ -23,6 +22,57 @@ use crate::{ }, }; +pub struct RequestIp(pub IpAddr); + +#[async_trait] +impl FromRequestParts for RequestIp +where + S: Send + Sync, +{ + type Rejection = Response; + + async fn from_request_parts(parts: &mut Parts, _state: &S) -> Result { + + let headers = &parts.headers; + + let forwardedfor = headers.get("x-forwarded-for") + .and_then(|h| h.to_str().ok()) + .and_then(|h| + h.split(',') + .rev() + .find_map(|s| s.trim().parse::().ok()) + ); + + if let Some(forwardedfor) = forwardedfor { + return Ok(RequestIp(forwardedfor)) + } + + let realip = headers.get("x-real-ip") + .and_then(|hv| hv.to_str().ok()) + .and_then(|s| s.parse::().ok()); + + if let Some(realip) = realip { + return Ok(RequestIp(realip)) + } + + let realip = headers.get("x-real-ip") + .and_then(|hv| hv.to_str().ok()) + .and_then(|s| s.parse::().ok()); + + if let Some(realip) = realip { + return Ok(RequestIp(realip)) + } + + let info = parts.extensions.get::>(); + + if let Some(info) = info { + return Ok(RequestIp(info.0.ip())) + } + + Err(ResponseCode::Forbidden.text("You have no ip")) + } +} + pub struct AuthorizedUser(pub User); #[async_trait] @@ -189,7 +239,7 @@ where S: Send + Sync, { - let Ok(ClientIp(ip)) = req.extract_parts::().await else { + let Ok(RequestIp(ip)) = req.extract_parts::().await else { tracing::error!("Failed to read client ip"); return Err(ResponseCode::InternalServerError.text("Failed to read client ip")); }; @@ -224,7 +274,7 @@ where B::Error: Into, S: Send + Sync, { - let Ok(ClientIp(ip)) = req.extract_parts::().await else { + let Ok(RequestIp(ip)) = req.extract_parts::().await else { tracing::error!("Failed to read client ip"); return Err(ResponseCode::InternalServerError.text("Failed to read client ip")); };