#![allow(clippy::cast_possible_truncation)]
use std::sync::atomic::{AtomicU64, AtomicUsize, Ordering};
use std::time::{Duration, Instant};
use super::limits::{GuardRailViolation, QueryLimits};
#[derive(Debug)]
pub struct QueryContext {
pub limits: QueryLimits,
start_time: Instant,
current_depth: AtomicU64,
current_cardinality: AtomicUsize,
memory_used: AtomicUsize,
}
impl QueryContext {
#[must_use]
pub fn new(limits: QueryLimits) -> Self {
Self {
limits,
start_time: Instant::now(),
current_depth: AtomicU64::new(0),
current_cardinality: AtomicUsize::new(0),
memory_used: AtomicUsize::new(0),
}
}
pub fn check_timeout(&self) -> Result<(), GuardRailViolation> {
if self.limits.timeout_ms == 0 {
return Ok(());
}
let elapsed_ms = self.start_time.elapsed().as_millis() as u64;
if elapsed_ms >= self.limits.timeout_ms {
return Err(GuardRailViolation::Timeout {
max_ms: self.limits.timeout_ms,
elapsed_ms,
});
}
Ok(())
}
pub fn check_depth(&self, depth: u32) -> Result<(), GuardRailViolation> {
self.current_depth
.store(u64::from(depth), Ordering::Relaxed);
if depth > self.limits.max_depth {
return Err(GuardRailViolation::DepthExceeded {
max: self.limits.max_depth,
actual: depth,
});
}
Ok(())
}
pub fn check_cardinality(&self, count: usize) -> Result<(), GuardRailViolation> {
let current = self.current_cardinality.fetch_add(count, Ordering::Relaxed) + count;
if current > self.limits.max_cardinality {
return Err(GuardRailViolation::CardinalityExceeded {
max: self.limits.max_cardinality,
actual: current,
});
}
Ok(())
}
pub fn check_memory(&self, bytes: usize) -> Result<(), GuardRailViolation> {
let current = self.memory_used.fetch_add(bytes, Ordering::Relaxed) + bytes;
if current > self.limits.memory_limit_bytes {
return Err(GuardRailViolation::MemoryExceeded {
max_bytes: self.limits.memory_limit_bytes,
used_bytes: current,
});
}
Ok(())
}
#[must_use]
pub fn elapsed(&self) -> Duration {
self.start_time.elapsed()
}
#[must_use]
pub fn memory_used(&self) -> usize {
self.memory_used.load(Ordering::Relaxed)
}
}