git.delta.rocks / jrsonnet / refs/commits / 9c0fa0107dd8

difftreelog

feat detect infinite recursion in object evaluation

Yaroslav Bolyukin2022-04-20parent: #5a22275.patch.diff
in: master

3 files changed

modifiedcrates/jrsonnet-evaluator/src/error.rsdiffbeforeafterboth
--- 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")]
modifiedcrates/jrsonnet-evaluator/src/obj.rsdiffbeforeafterboth
--- 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<GcHashSet<ObjValue>>,
 	this_obj: Option<ObjValue>,
 	this_entries: Cc<GcHashMap<IStr, ObjMember>>,
-	value_cache: RefCell<GcHashMap<CacheKey, Option<Val>>>,
+	value_cache: RefCell<GcHashMap<CacheKey, CacheValue>>,
 }
 
 #[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<Val> {
modifiedcrates/jrsonnet-evaluator/src/val.rsdiffbeforeafterboth
47 match &*self.0.borrow() {47 match &*self.0.borrow() {
48 LazyValInternals::Computed(v) => return Ok(v.clone()),48 LazyValInternals::Computed(v) => return Ok(v.clone()),
49 LazyValInternals::Errored(e) => return Err(e.clone()),49 LazyValInternals::Errored(e) => return Err(e.clone()),
50 LazyValInternals::Pending => return Err(RecursiveLazyValueEvaluation.into()),50 LazyValInternals::Pending => return Err(InfiniteRecursionDetected.into()),
51 _ => (),51 _ => (),
52 };52 };
53 let value = if let LazyValInternals::Waiting(value) =53 let value = if let LazyValInternals::Waiting(value) =