mirror of
https://github.com/LemmyNet/lemmy.git
synced 2024-11-26 03:11:08 +00:00
* Dont log errors when rate limit is hit (fixes #2157) * Clone service rather than http request * some cleanup/refactoring Co-authored-by: Aode (Lion) <asonix@asonix.dog>
This commit is contained in:
parent
c681bb79b1
commit
a5ff629b24
3 changed files with 73 additions and 106 deletions
|
@ -1,10 +1,14 @@
|
||||||
use crate::{settings::structs::RateLimitConfig, utils::get_ip, IpAddr, LemmyError};
|
use crate::{settings::structs::RateLimitConfig, utils::get_ip, IpAddr};
|
||||||
use actix_web::dev::{Service, ServiceRequest, ServiceResponse, Transform};
|
use actix_web::{
|
||||||
|
dev::{Service, ServiceRequest, ServiceResponse, Transform},
|
||||||
|
HttpResponse,
|
||||||
|
};
|
||||||
use futures::future::{ok, Ready};
|
use futures::future::{ok, Ready};
|
||||||
use rate_limiter::{RateLimitType, RateLimiter};
|
use rate_limiter::{RateLimitType, RateLimiter};
|
||||||
use std::{
|
use std::{
|
||||||
future::Future,
|
future::Future,
|
||||||
pin::Pin,
|
pin::Pin,
|
||||||
|
rc::Rc,
|
||||||
sync::Arc,
|
sync::Arc,
|
||||||
task::{Context, Poll},
|
task::{Context, Poll},
|
||||||
};
|
};
|
||||||
|
@ -29,7 +33,7 @@ pub struct RateLimited {
|
||||||
|
|
||||||
pub struct RateLimitedMiddleware<S> {
|
pub struct RateLimitedMiddleware<S> {
|
||||||
rate_limited: RateLimited,
|
rate_limited: RateLimited,
|
||||||
service: S,
|
service: Rc<S>,
|
||||||
}
|
}
|
||||||
|
|
||||||
impl RateLimit {
|
impl RateLimit {
|
||||||
|
@ -63,78 +67,28 @@ impl RateLimit {
|
||||||
}
|
}
|
||||||
|
|
||||||
impl RateLimited {
|
impl RateLimited {
|
||||||
pub async fn wrap<T, E>(
|
/// Returns true if the request passed the rate limit, false if it failed and should be rejected.
|
||||||
self,
|
pub async fn check(self, ip_addr: IpAddr) -> bool {
|
||||||
ip_addr: IpAddr,
|
|
||||||
fut: impl Future<Output = Result<T, E>>,
|
|
||||||
) -> Result<T, E>
|
|
||||||
where
|
|
||||||
E: From<LemmyError>,
|
|
||||||
{
|
|
||||||
// Does not need to be blocking because the RwLock in settings never held across await points,
|
// Does not need to be blocking because the RwLock in settings never held across await points,
|
||||||
// and the operation here locks only long enough to clone
|
// and the operation here locks only long enough to clone
|
||||||
let rate_limit = self.rate_limit_config;
|
let rate_limit = self.rate_limit_config;
|
||||||
|
|
||||||
// before
|
let mut limiter = self.rate_limiter.lock().await;
|
||||||
{
|
|
||||||
let mut limiter = self.rate_limiter.lock().await;
|
|
||||||
|
|
||||||
match self.type_ {
|
let (kind, interval) = match self.type_ {
|
||||||
RateLimitType::Message => {
|
RateLimitType::Message => (rate_limit.message, rate_limit.message_per_second),
|
||||||
limiter.check_rate_limit_full(
|
RateLimitType::Post => (rate_limit.post, rate_limit.post_per_second),
|
||||||
self.type_,
|
RateLimitType::Register => (rate_limit.register, rate_limit.register_per_second),
|
||||||
&ip_addr,
|
RateLimitType::Image => (rate_limit.image, rate_limit.image_per_second),
|
||||||
rate_limit.message,
|
RateLimitType::Comment => (rate_limit.comment, rate_limit.comment_per_second),
|
||||||
rate_limit.message_per_second,
|
};
|
||||||
)?;
|
limiter.check_rate_limit_full(self.type_, &ip_addr, kind, interval)
|
||||||
|
|
||||||
drop(limiter);
|
|
||||||
return fut.await;
|
|
||||||
}
|
|
||||||
RateLimitType::Post => {
|
|
||||||
limiter.check_rate_limit_full(
|
|
||||||
self.type_,
|
|
||||||
&ip_addr,
|
|
||||||
rate_limit.post,
|
|
||||||
rate_limit.post_per_second,
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
RateLimitType::Register => {
|
|
||||||
limiter.check_rate_limit_full(
|
|
||||||
self.type_,
|
|
||||||
&ip_addr,
|
|
||||||
rate_limit.register,
|
|
||||||
rate_limit.register_per_second,
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
RateLimitType::Image => {
|
|
||||||
limiter.check_rate_limit_full(
|
|
||||||
self.type_,
|
|
||||||
&ip_addr,
|
|
||||||
rate_limit.image,
|
|
||||||
rate_limit.image_per_second,
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
RateLimitType::Comment => {
|
|
||||||
limiter.check_rate_limit_full(
|
|
||||||
self.type_,
|
|
||||||
&ip_addr,
|
|
||||||
rate_limit.comment,
|
|
||||||
rate_limit.comment_per_second,
|
|
||||||
)?;
|
|
||||||
}
|
|
||||||
};
|
|
||||||
}
|
|
||||||
|
|
||||||
let res = fut.await;
|
|
||||||
|
|
||||||
res
|
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
impl<S> Transform<S, ServiceRequest> for RateLimited
|
impl<S> Transform<S, ServiceRequest> for RateLimited
|
||||||
where
|
where
|
||||||
S: Service<ServiceRequest, Response = ServiceResponse, Error = actix_web::Error>,
|
S: Service<ServiceRequest, Response = ServiceResponse, Error = actix_web::Error> + 'static,
|
||||||
S::Future: 'static,
|
S::Future: 'static,
|
||||||
{
|
{
|
||||||
type Response = S::Response;
|
type Response = S::Response;
|
||||||
|
@ -146,7 +100,7 @@ where
|
||||||
fn new_transform(&self, service: S) -> Self::Future {
|
fn new_transform(&self, service: S) -> Self::Future {
|
||||||
ok(RateLimitedMiddleware {
|
ok(RateLimitedMiddleware {
|
||||||
rate_limited: self.clone(),
|
rate_limited: self.clone(),
|
||||||
service,
|
service: Rc::new(service),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
@ -155,7 +109,7 @@ type FutResult<T, E> = dyn Future<Output = Result<T, E>>;
|
||||||
|
|
||||||
impl<S> Service<ServiceRequest> for RateLimitedMiddleware<S>
|
impl<S> Service<ServiceRequest> for RateLimitedMiddleware<S>
|
||||||
where
|
where
|
||||||
S: Service<ServiceRequest, Response = ServiceResponse, Error = actix_web::Error>,
|
S: Service<ServiceRequest, Response = ServiceResponse, Error = actix_web::Error> + 'static,
|
||||||
S::Future: 'static,
|
S::Future: 'static,
|
||||||
{
|
{
|
||||||
type Response = S::Response;
|
type Response = S::Response;
|
||||||
|
@ -169,11 +123,20 @@ where
|
||||||
fn call(&self, req: ServiceRequest) -> Self::Future {
|
fn call(&self, req: ServiceRequest) -> Self::Future {
|
||||||
let ip_addr = get_ip(&req.connection_info());
|
let ip_addr = get_ip(&req.connection_info());
|
||||||
|
|
||||||
let fut = self
|
let rate_limited = self.rate_limited.clone();
|
||||||
.rate_limited
|
let service = self.service.clone();
|
||||||
.clone()
|
|
||||||
.wrap(ip_addr, self.service.call(req));
|
|
||||||
|
|
||||||
Box::pin(async move { fut.await.map_err(actix_web::Error::from) })
|
Box::pin(async move {
|
||||||
|
if rate_limited.check(ip_addr).await {
|
||||||
|
service.call(req).await
|
||||||
|
} else {
|
||||||
|
let (http_req, _) = req.into_parts();
|
||||||
|
// if rate limit was hit, respond with http 400
|
||||||
|
Ok(ServiceResponse::new(
|
||||||
|
http_req,
|
||||||
|
HttpResponse::BadRequest().finish(),
|
||||||
|
))
|
||||||
|
}
|
||||||
|
})
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -1,11 +1,11 @@
|
||||||
use crate::{IpAddr, LemmyError};
|
use crate::IpAddr;
|
||||||
use std::{collections::HashMap, time::SystemTime};
|
use std::{collections::HashMap, time::Instant};
|
||||||
use strum::IntoEnumIterator;
|
use strum::IntoEnumIterator;
|
||||||
use tracing::debug;
|
use tracing::debug;
|
||||||
|
|
||||||
#[derive(Debug, Clone)]
|
#[derive(Debug, Clone)]
|
||||||
struct RateLimitBucket {
|
struct RateLimitBucket {
|
||||||
last_checked: SystemTime,
|
last_checked: Instant,
|
||||||
allowance: f64,
|
allowance: f64,
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -36,7 +36,7 @@ impl RateLimiter {
|
||||||
bucket.insert(
|
bucket.insert(
|
||||||
ip.clone(),
|
ip.clone(),
|
||||||
RateLimitBucket {
|
RateLimitBucket {
|
||||||
last_checked: SystemTime::now(),
|
last_checked: Instant::now(),
|
||||||
allowance: -2f64,
|
allowance: -2f64,
|
||||||
},
|
},
|
||||||
);
|
);
|
||||||
|
@ -46,6 +46,8 @@ impl RateLimiter {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Rate limiting Algorithm described here: https://stackoverflow.com/a/668327/1655478
|
/// Rate limiting Algorithm described here: https://stackoverflow.com/a/668327/1655478
|
||||||
|
///
|
||||||
|
/// Returns true if the request passed the rate limit, false if it failed and should be rejected.
|
||||||
#[allow(clippy::float_cmp)]
|
#[allow(clippy::float_cmp)]
|
||||||
pub(super) fn check_rate_limit_full(
|
pub(super) fn check_rate_limit_full(
|
||||||
&mut self,
|
&mut self,
|
||||||
|
@ -53,12 +55,12 @@ impl RateLimiter {
|
||||||
ip: &IpAddr,
|
ip: &IpAddr,
|
||||||
rate: i32,
|
rate: i32,
|
||||||
per: i32,
|
per: i32,
|
||||||
) -> Result<(), LemmyError> {
|
) -> bool {
|
||||||
self.insert_ip(ip);
|
self.insert_ip(ip);
|
||||||
if let Some(bucket) = self.buckets.get_mut(&type_) {
|
if let Some(bucket) = self.buckets.get_mut(&type_) {
|
||||||
if let Some(rate_limit) = bucket.get_mut(ip) {
|
if let Some(rate_limit) = bucket.get_mut(ip) {
|
||||||
let current = SystemTime::now();
|
let current = Instant::now();
|
||||||
let time_passed = current.duration_since(rate_limit.last_checked)?.as_secs() as f64;
|
let time_passed = current.duration_since(rate_limit.last_checked).as_secs() as f64;
|
||||||
|
|
||||||
// The initial value
|
// The initial value
|
||||||
if rate_limit.allowance == -2f64 {
|
if rate_limit.allowance == -2f64 {
|
||||||
|
@ -79,25 +81,16 @@ impl RateLimiter {
|
||||||
time_passed,
|
time_passed,
|
||||||
rate_limit.allowance
|
rate_limit.allowance
|
||||||
);
|
);
|
||||||
Err(LemmyError::from_error_message(
|
false
|
||||||
anyhow::anyhow!(
|
|
||||||
"Too many requests. type: {}, IP: {}, {} per {} seconds",
|
|
||||||
type_.as_ref(),
|
|
||||||
ip,
|
|
||||||
rate,
|
|
||||||
per
|
|
||||||
),
|
|
||||||
"too_many_requests",
|
|
||||||
))
|
|
||||||
} else {
|
} else {
|
||||||
rate_limit.allowance -= 1.0;
|
rate_limit.allowance -= 1.0;
|
||||||
Ok(())
|
true
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Ok(())
|
true
|
||||||
}
|
}
|
||||||
} else {
|
} else {
|
||||||
Ok(())
|
true
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -478,22 +478,33 @@ impl ChatServer {
|
||||||
.as_str()
|
.as_str()
|
||||||
.ok_or_else(|| LemmyError::from_message("missing op"))?;
|
.ok_or_else(|| LemmyError::from_message("missing op"))?;
|
||||||
|
|
||||||
if let Ok(user_operation_crud) = UserOperationCrud::from_str(op) {
|
// check if api call passes the rate limit, and generate future for later execution
|
||||||
let fut = (message_handler_crud)(context, msg.id, user_operation_crud.clone(), data);
|
let (passed, fut) = if let Ok(user_operation_crud) = UserOperationCrud::from_str(op) {
|
||||||
match user_operation_crud {
|
let passed = match user_operation_crud {
|
||||||
UserOperationCrud::Register => rate_limiter.register().wrap(ip, fut).await,
|
UserOperationCrud::Register => rate_limiter.register().check(ip).await,
|
||||||
UserOperationCrud::CreatePost => rate_limiter.post().wrap(ip, fut).await,
|
UserOperationCrud::CreatePost => rate_limiter.post().check(ip).await,
|
||||||
UserOperationCrud::CreateCommunity => rate_limiter.register().wrap(ip, fut).await,
|
UserOperationCrud::CreateCommunity => rate_limiter.register().check(ip).await,
|
||||||
UserOperationCrud::CreateComment => rate_limiter.comment().wrap(ip, fut).await,
|
UserOperationCrud::CreateComment => rate_limiter.comment().check(ip).await,
|
||||||
_ => rate_limiter.message().wrap(ip, fut).await,
|
_ => rate_limiter.message().check(ip).await,
|
||||||
}
|
};
|
||||||
|
let fut = (message_handler_crud)(context, msg.id, user_operation_crud, data);
|
||||||
|
(passed, fut)
|
||||||
} else {
|
} else {
|
||||||
let user_operation = UserOperation::from_str(op)?;
|
let user_operation = UserOperation::from_str(op)?;
|
||||||
let fut = (message_handler)(context, msg.id, user_operation.clone(), data);
|
let passed = match user_operation {
|
||||||
match user_operation {
|
UserOperation::GetCaptcha => rate_limiter.post().check(ip).await,
|
||||||
UserOperation::GetCaptcha => rate_limiter.post().wrap(ip, fut).await,
|
_ => rate_limiter.message().check(ip).await,
|
||||||
_ => rate_limiter.message().wrap(ip, fut).await,
|
};
|
||||||
}
|
let fut = (message_handler)(context, msg.id, user_operation, data);
|
||||||
|
(passed, fut)
|
||||||
|
};
|
||||||
|
|
||||||
|
// if rate limit passed, execute api call future
|
||||||
|
if passed {
|
||||||
|
fut.await
|
||||||
|
} else {
|
||||||
|
// if rate limit was hit, respond with empty message
|
||||||
|
Ok("".to_string())
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
Loading…
Reference in a new issue