From 9c0fa0107dd855faf59e4af7792d511bd4585c0a Mon Sep 17 00:00:00 2001 From: Yaroslav Bolyukin Date: Wed, 20 Apr 2022 15:57:22 +0000 Subject: [PATCH] feat: detect infinite recursion in object evaluation --- --- a/crates/jrsonnet-evaluator/src/error.rs +++ b/crates/jrsonnet-evaluator/src/error.rs @@ -106,7 +106,7 @@ #[error("stack overflow, try to reduce recursion, or set --max-stack to bigger value")] StackOverflow, #[error("infinite recursion detected")] - RecursiveLazyValueEvaluation, + InfiniteRecursionDetected, #[error("tried to index by fractional value")] FractionalIndex, #[error("attempted to divide by zero")] --- a/crates/jrsonnet-evaluator/src/obj.rs +++ b/crates/jrsonnet-evaluator/src/obj.rs @@ -1,3 +1,4 @@ +use crate::error::LocError; use crate::function::CallLocation; use crate::gc::{GcHashMap, GcHashSet, TraceBox}; use crate::operator::evaluate_add_op; @@ -28,6 +29,15 @@ // Field => This type CacheKey = (IStr, WeakObjValue); + +#[derive(Trace)] +enum CacheValue { + Cached(Val), + NotFound, + Pending, + Errored(LocError), +} + #[derive(Trace)] #[force_tracking] pub struct ObjValueInternals { @@ -36,7 +46,7 @@ assertions_ran: RefCell>, this_obj: Option, this_entries: Cc>, - value_cache: RefCell>>, + value_cache: RefCell>, } #[derive(Clone, Trace)] @@ -234,8 +244,17 @@ let cache_key = (key.clone(), WeakObjValue(real_this.0.downgrade())); if let Some(v) = self.0.value_cache.borrow().get(&cache_key) { - return Ok(v.clone()); + return Ok(match v { + CacheValue::Cached(v) => Some(v.clone()), + CacheValue::NotFound => None, + CacheValue::Pending => throw!(InfiniteRecursionDetected), + CacheValue::Errored(e) => return Err(e.clone()), + }); } + self.0 + .value_cache + .borrow_mut() + .insert(cache_key.clone(), CacheValue::Pending); let value = match (self.0.this_entries.get(&key), &self.0.super_obj) { (Some(k), None) => Ok(Some(self.evaluate_this(k, real_this)?)), (Some(k), Some(s)) => { @@ -251,11 +270,24 @@ } (None, Some(s)) => s.get_raw(key, Some(real_this)), (None, None) => Ok(None), - }?; - self.0 - .value_cache - .borrow_mut() - .insert(cache_key, value.clone()); + }; + let value = match value { + Ok(v) => v, + Err(e) => { + self.0 + .value_cache + .borrow_mut() + .insert(cache_key, CacheValue::Errored(e.clone())); + return Err(e); + } + }; + self.0.value_cache.borrow_mut().insert( + cache_key, + match &value { + Some(v) => CacheValue::Cached(v.clone()), + None => CacheValue::NotFound, + }, + ); Ok(value) } fn evaluate_this(&self, v: &ObjMember, real_this: &Self) -> Result { --- a/crates/jrsonnet-evaluator/src/val.rs +++ b/crates/jrsonnet-evaluator/src/val.rs @@ -47,7 +47,7 @@ match &*self.0.borrow() { LazyValInternals::Computed(v) => return Ok(v.clone()), LazyValInternals::Errored(e) => return Err(e.clone()), - LazyValInternals::Pending => return Err(RecursiveLazyValueEvaluation.into()), + LazyValInternals::Pending => return Err(InfiniteRecursionDetected.into()), _ => (), }; let value = if let LazyValInternals::Waiting(value) = -- gitstuff