|
@@ -8,7 +8,10 @@ use crate::{
|
|
|
};
|
|
|
use chrono::{DateTime, Utc};
|
|
|
use num::CheckedAdd;
|
|
|
-use std::ops::{Add, Deref};
|
|
|
+use std::{
|
|
|
+ ops::{Add, Deref},
|
|
|
+ rc::{Rc, Weak},
|
|
|
+};
|
|
|
|
|
|
#[derive(Clone, Debug, PartialEq, PartialOrd)]
|
|
|
pub enum Value {
|
|
@@ -73,10 +76,27 @@ impl Value {
|
|
|
Value::Bool(b) => Ok(*b),
|
|
|
Value::Nil => Ok(false),
|
|
|
Value::Number(x) => Ok(*x > 0),
|
|
|
+ /// TODO: Throw an error . Add types for convertion errors
|
|
|
_ => panic!("Cannot convert to bool"),
|
|
|
}
|
|
|
}
|
|
|
|
|
|
+ /// Checks if the value is TRUE, any other value will return FALSE.
|
|
|
+ pub fn is_true(&self) -> bool {
|
|
|
+ match self {
|
|
|
+ Value::Bool(b) => *b,
|
|
|
+ _ => false,
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
+ /// Checks if the value is FALSE, any other value will return FALSE.
|
|
|
+ pub fn is_false(&self) -> bool {
|
|
|
+ match self {
|
|
|
+ Value::Bool(b) => *b == false,
|
|
|
+ _ => false,
|
|
|
+ }
|
|
|
+ }
|
|
|
+
|
|
|
pub fn as_number(&self) -> Result<i128, Error> {
|
|
|
match self {
|
|
|
Value::Number(x) => Ok(*x),
|
|
@@ -108,63 +128,60 @@ impl CheckedAdd for Value {
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-#[derive(Debug, PartialOrd)]
|
|
|
+#[derive(Debug, Clone)]
|
|
|
/// Value or reference to a value.
|
|
|
///
|
|
|
-/// A reference to a value is being used to avoid cloning from the source code to the registers,
|
|
|
-/// instead to just use a readonly reference
|
|
|
-pub enum ValueOrRef<'a> {
|
|
|
- Ref(&'a Value),
|
|
|
+/// This is a sorf of attempt to avoid cloning from the source code to the
|
|
|
+/// registers, specially when the value is being defined in the source code,
|
|
|
+/// instead
|
|
|
+/// a Weak reference is being used to avoid cloning the value.
|
|
|
+pub enum ValueOrRef {
|
|
|
+ WeakRef(Weak<Value>),
|
|
|
Value(Value),
|
|
|
}
|
|
|
|
|
|
-impl PartialEq for ValueOrRef<'_> {
|
|
|
- fn eq(&self, other: &Self) -> bool {
|
|
|
- match (self, other) {
|
|
|
- (ValueOrRef::Ref(a), ValueOrRef::Ref(b)) => a == b,
|
|
|
- (ValueOrRef::Value(a), ValueOrRef::Value(b)) => a == b,
|
|
|
- (ValueOrRef::Ref(a), ValueOrRef::Value(b)) => *a == b,
|
|
|
- (ValueOrRef::Value(a), ValueOrRef::Ref(b)) => a == *b,
|
|
|
- }
|
|
|
- }
|
|
|
-}
|
|
|
+impl Deref for ValueOrRef {
|
|
|
+ type Target = Value;
|
|
|
|
|
|
-impl<'a> Clone for ValueOrRef<'a> {
|
|
|
- fn clone(&self) -> Self {
|
|
|
+ fn deref(&self) -> &Self::Target {
|
|
|
match self {
|
|
|
- ValueOrRef::Ref(value) => ValueOrRef::Value((*value).clone()),
|
|
|
- ValueOrRef::Value(value) => ValueOrRef::Value(value.clone()),
|
|
|
+ ValueOrRef::WeakRef(value) => {
|
|
|
+ if let Some(rc_value) = value.upgrade() {
|
|
|
+ unsafe { &*(Rc::into_raw(rc_value) as *const Value) }
|
|
|
+ } else {
|
|
|
+ &Value::Nil
|
|
|
+ }
|
|
|
+ }
|
|
|
+ ValueOrRef::Value(v) => v,
|
|
|
}
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-impl<'a> Into<Value> for ValueOrRef<'a> {
|
|
|
- fn into(self) -> Value {
|
|
|
- match self {
|
|
|
- ValueOrRef::Ref(value) => value.clone(),
|
|
|
- ValueOrRef::Value(value) => value,
|
|
|
- }
|
|
|
+impl PartialEq for ValueOrRef {
|
|
|
+ fn eq(&self, other: &Self) -> bool {
|
|
|
+ self.deref() == other.deref()
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-impl Deref for ValueOrRef<'_> {
|
|
|
- type Target = Value;
|
|
|
+impl PartialOrd for ValueOrRef {
|
|
|
+ fn partial_cmp(&self, other: &Self) -> Option<std::cmp::Ordering> {
|
|
|
+ self.partial_cmp(other)
|
|
|
+ }
|
|
|
+}
|
|
|
|
|
|
- fn deref(&self) -> &Self::Target {
|
|
|
- match self {
|
|
|
- ValueOrRef::Ref(value) => *value,
|
|
|
- ValueOrRef::Value(value) => &value,
|
|
|
- }
|
|
|
+impl From<Rc<Value>> for ValueOrRef {
|
|
|
+ fn from(value: Rc<Value>) -> Self {
|
|
|
+ ValueOrRef::WeakRef(Rc::downgrade(&value))
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-impl<'a> From<&'a Value> for ValueOrRef<'a> {
|
|
|
- fn from(value: &'a Value) -> Self {
|
|
|
- ValueOrRef::Ref(value)
|
|
|
+impl From<&Rc<Value>> for ValueOrRef {
|
|
|
+ fn from(value: &Rc<Value>) -> Self {
|
|
|
+ ValueOrRef::WeakRef(Rc::downgrade(&value))
|
|
|
}
|
|
|
}
|
|
|
|
|
|
-impl<'a> From<Value> for ValueOrRef<'a> {
|
|
|
+impl From<Value> for ValueOrRef {
|
|
|
fn from(value: Value) -> Self {
|
|
|
ValueOrRef::Value(value)
|
|
|
}
|