mirror of
https://github.com/telemt/telemt.git
synced 2026-06-19 02:00:08 +07:00
98c985091c
Signed-off-by: Alexey <247128645+axkurcom@users.noreply.github.com>
69 lines
2.0 KiB
Rust
69 lines
2.0 KiB
Rust
use crate::stats::UserStats;
|
|
use std::io;
|
|
use std::sync::atomic::Ordering;
|
|
|
|
#[derive(Debug)]
|
|
struct QuotaIoSentinel;
|
|
|
|
impl std::fmt::Display for QuotaIoSentinel {
|
|
fn fmt(&self, f: &mut std::fmt::Formatter<'_>) -> std::fmt::Result {
|
|
f.write_str("user data quota exceeded")
|
|
}
|
|
}
|
|
|
|
impl std::error::Error for QuotaIoSentinel {}
|
|
|
|
pub(super) fn quota_io_error() -> io::Error {
|
|
io::Error::new(io::ErrorKind::PermissionDenied, QuotaIoSentinel)
|
|
}
|
|
|
|
pub(in crate::proxy::relay) fn is_quota_io_error(err: &io::Error) -> bool {
|
|
err.kind() == io::ErrorKind::PermissionDenied
|
|
&& err
|
|
.get_ref()
|
|
.and_then(|source| source.downcast_ref::<QuotaIoSentinel>())
|
|
.is_some()
|
|
}
|
|
|
|
const QUOTA_NEAR_LIMIT_BYTES: u64 = 64 * 1024;
|
|
const QUOTA_LARGE_CHARGE_BYTES: u64 = 16 * 1024;
|
|
const QUOTA_ADAPTIVE_INTERVAL_MIN_BYTES: u64 = 4 * 1024;
|
|
const QUOTA_ADAPTIVE_INTERVAL_MAX_BYTES: u64 = 64 * 1024;
|
|
pub(super) const QUOTA_RESERVE_SPIN_RETRIES: usize = 64;
|
|
pub(super) const QUOTA_RESERVE_MAX_ROUNDS: usize = 8;
|
|
|
|
#[inline]
|
|
pub(in crate::proxy::relay) fn quota_adaptive_interval_bytes(remaining_before: u64) -> u64 {
|
|
remaining_before.saturating_div(2).clamp(
|
|
QUOTA_ADAPTIVE_INTERVAL_MIN_BYTES,
|
|
QUOTA_ADAPTIVE_INTERVAL_MAX_BYTES,
|
|
)
|
|
}
|
|
|
|
#[inline]
|
|
pub(in crate::proxy::relay) fn should_immediate_quota_check(
|
|
remaining_before: u64,
|
|
charge_bytes: u64,
|
|
) -> bool {
|
|
remaining_before <= QUOTA_NEAR_LIMIT_BYTES || charge_bytes >= QUOTA_LARGE_CHARGE_BYTES
|
|
}
|
|
|
|
pub(super) fn refund_reserved_quota_bytes(user_stats: &UserStats, reserved_bytes: u64) {
|
|
if reserved_bytes == 0 {
|
|
return;
|
|
}
|
|
let mut current = user_stats.quota_used.load(Ordering::Relaxed);
|
|
loop {
|
|
let next = current.saturating_sub(reserved_bytes);
|
|
match user_stats.quota_used.compare_exchange_weak(
|
|
current,
|
|
next,
|
|
Ordering::Relaxed,
|
|
Ordering::Relaxed,
|
|
) {
|
|
Ok(_) => return,
|
|
Err(observed) => current = observed,
|
|
}
|
|
}
|
|
}
|