2022-03-25 20:31:45 +00:00
|
|
|
#![allow(clippy::derive_ord_xor_partial_ord)]
|
|
|
|
|
2022-05-14 12:35:08 +00:00
|
|
|
use crate::ctx::Context;
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::dbs::Options;
|
2022-02-15 01:00:30 +00:00
|
|
|
use crate::dbs::Transaction;
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::err::Error;
|
|
|
|
use crate::sql::array::{array, Array};
|
2023-02-21 14:15:02 +00:00
|
|
|
use crate::sql::block::{block, Block};
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::sql::common::commas;
|
2022-10-01 14:55:48 +00:00
|
|
|
use crate::sql::constant::{constant, Constant};
|
2022-02-22 19:08:22 +00:00
|
|
|
use crate::sql::datetime::{datetime, Datetime};
|
|
|
|
use crate::sql::duration::{duration, Duration};
|
2022-06-09 08:18:08 +00:00
|
|
|
use crate::sql::edges::{edges, Edges};
|
2022-01-16 20:31:50 +00:00
|
|
|
use crate::sql::error::IResult;
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::sql::expression::{expression, Expression};
|
2023-01-19 09:53:33 +00:00
|
|
|
use crate::sql::fmt::{Fmt, Pretty};
|
2023-03-31 15:42:29 +00:00
|
|
|
use crate::sql::function::{self, function, Function};
|
2022-10-31 23:12:41 +00:00
|
|
|
use crate::sql::future::{future, Future};
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::sql::geometry::{geometry, Geometry};
|
2022-03-18 07:24:36 +00:00
|
|
|
use crate::sql::id::Id;
|
2023-03-31 15:42:29 +00:00
|
|
|
use crate::sql::idiom::{self, Idiom};
|
2022-04-07 14:44:55 +00:00
|
|
|
use crate::sql::kind::Kind;
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::sql::model::{model, Model};
|
|
|
|
use crate::sql::number::{number, Number};
|
|
|
|
use crate::sql::object::{object, Object};
|
2022-01-22 21:16:13 +00:00
|
|
|
use crate::sql::operation::Operation;
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::sql::param::{param, Param};
|
|
|
|
use crate::sql::part::Part;
|
2022-08-28 22:19:59 +00:00
|
|
|
use crate::sql::range::{range, Range};
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::sql::regex::{regex, Regex};
|
2022-05-21 00:35:59 +00:00
|
|
|
use crate::sql::serde::is_internal_serialization;
|
2022-01-13 17:36:41 +00:00
|
|
|
use crate::sql::strand::{strand, Strand};
|
|
|
|
use crate::sql::subquery::{subquery, Subquery};
|
|
|
|
use crate::sql::table::{table, Table};
|
|
|
|
use crate::sql::thing::{thing, Thing};
|
2022-06-25 22:37:45 +00:00
|
|
|
use crate::sql::uuid::{uuid as unique, Uuid};
|
2022-01-14 08:12:56 +00:00
|
|
|
use async_recursion::async_recursion;
|
2022-03-17 21:01:29 +00:00
|
|
|
use bigdecimal::BigDecimal;
|
2022-01-13 17:36:41 +00:00
|
|
|
use chrono::{DateTime, Utc};
|
2022-02-26 00:34:05 +00:00
|
|
|
use derive::Store;
|
2022-01-13 17:36:41 +00:00
|
|
|
use fuzzy_matcher::skim::SkimMatcherV2;
|
|
|
|
use fuzzy_matcher::FuzzyMatcher;
|
|
|
|
use geo::Point;
|
|
|
|
use nom::branch::alt;
|
|
|
|
use nom::bytes::complete::tag_no_case;
|
|
|
|
use nom::combinator::map;
|
|
|
|
use nom::multi::separated_list1;
|
|
|
|
use once_cell::sync::Lazy;
|
|
|
|
use serde::{Deserialize, Serialize};
|
2022-03-25 18:43:22 +00:00
|
|
|
use std::cmp::Ordering;
|
2022-01-13 17:36:41 +00:00
|
|
|
use std::collections::BTreeMap;
|
|
|
|
use std::collections::HashMap;
|
2023-01-19 09:53:33 +00:00
|
|
|
use std::fmt::{self, Display, Formatter, Write};
|
2022-01-13 17:36:41 +00:00
|
|
|
use std::ops;
|
2022-03-25 18:43:22 +00:00
|
|
|
use std::ops::Deref;
|
2022-01-13 17:36:41 +00:00
|
|
|
use std::str::FromStr;
|
|
|
|
|
2022-02-22 14:16:50 +00:00
|
|
|
static MATCHER: Lazy<SkimMatcherV2> = Lazy::new(|| SkimMatcherV2::default().ignore_case());
|
2022-01-13 17:36:41 +00:00
|
|
|
|
2023-03-30 10:41:44 +00:00
|
|
|
pub(crate) const TOKEN: &str = "$surrealdb::private::sql::Value";
|
|
|
|
|
2022-10-27 12:23:24 +00:00
|
|
|
#[derive(Clone, Debug, Default, Eq, PartialEq, Serialize, Deserialize, Hash)]
|
2022-01-13 17:36:41 +00:00
|
|
|
pub struct Values(pub Vec<Value>);
|
|
|
|
|
2022-03-25 18:43:22 +00:00
|
|
|
impl Deref for Values {
|
|
|
|
type Target = Vec<Value>;
|
|
|
|
fn deref(&self) -> &Self::Target {
|
|
|
|
&self.0
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl IntoIterator for Values {
|
|
|
|
type Item = Value;
|
|
|
|
type IntoIter = std::vec::IntoIter<Self::Item>;
|
|
|
|
fn into_iter(self) -> Self::IntoIter {
|
|
|
|
self.0.into_iter()
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-04 21:51:18 +00:00
|
|
|
impl Display for Values {
|
|
|
|
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
|
|
|
|
Display::fmt(&Fmt::comma_separated(&self.0), f)
|
2022-03-25 18:43:22 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn values(i: &str) -> IResult<&str, Values> {
|
|
|
|
let (i, v) = separated_list1(commas, value)(i)?;
|
|
|
|
Ok((i, Values(v)))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn selects(i: &str) -> IResult<&str, Values> {
|
|
|
|
let (i, v) = separated_list1(commas, select)(i)?;
|
|
|
|
Ok((i, Values(v)))
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn whats(i: &str) -> IResult<&str, Values> {
|
|
|
|
let (i, v) = separated_list1(commas, what)(i)?;
|
|
|
|
Ok((i, Values(v)))
|
|
|
|
}
|
|
|
|
|
2023-03-18 09:18:22 +00:00
|
|
|
#[derive(Clone, Debug, Default, PartialEq, PartialOrd, Deserialize, Store, Hash)]
|
2022-01-13 17:36:41 +00:00
|
|
|
pub enum Value {
|
2023-03-18 09:18:22 +00:00
|
|
|
#[default]
|
2022-01-13 17:36:41 +00:00
|
|
|
None,
|
|
|
|
Null,
|
|
|
|
False,
|
|
|
|
True,
|
2022-03-23 11:56:39 +00:00
|
|
|
Number(Number),
|
|
|
|
Strand(Strand),
|
|
|
|
Duration(Duration),
|
2022-04-04 15:59:42 +00:00
|
|
|
Datetime(Datetime),
|
2022-06-25 22:37:45 +00:00
|
|
|
Uuid(Uuid),
|
2022-03-23 11:56:39 +00:00
|
|
|
Array(Array),
|
|
|
|
Object(Object),
|
|
|
|
Geometry(Geometry),
|
2023-03-31 18:44:44 +00:00
|
|
|
Bytes(Vec<u8>),
|
2022-03-23 11:56:39 +00:00
|
|
|
// ---
|
2022-01-13 17:36:41 +00:00
|
|
|
Param(Param),
|
|
|
|
Idiom(Idiom),
|
|
|
|
Table(Table),
|
|
|
|
Thing(Thing),
|
|
|
|
Model(Model),
|
|
|
|
Regex(Regex),
|
2023-02-21 14:15:02 +00:00
|
|
|
Block(Box<Block>),
|
2022-08-28 22:19:59 +00:00
|
|
|
Range(Box<Range>),
|
2022-06-09 08:18:08 +00:00
|
|
|
Edges(Box<Edges>),
|
2022-10-31 23:12:41 +00:00
|
|
|
Future(Box<Future>),
|
2022-10-01 14:55:48 +00:00
|
|
|
Constant(Constant),
|
2022-01-13 17:36:41 +00:00
|
|
|
Function(Box<Function>),
|
|
|
|
Subquery(Box<Subquery>),
|
|
|
|
Expression(Box<Expression>),
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Eq for Value {}
|
|
|
|
|
2022-03-25 18:43:22 +00:00
|
|
|
impl Ord for Value {
|
|
|
|
fn cmp(&self, other: &Self) -> Ordering {
|
|
|
|
self.partial_cmp(other).unwrap_or(Ordering::Equal)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-13 17:36:41 +00:00
|
|
|
impl From<bool> for Value {
|
2022-09-01 00:27:52 +00:00
|
|
|
#[inline]
|
2022-01-13 17:36:41 +00:00
|
|
|
fn from(v: bool) -> Self {
|
|
|
|
match v {
|
|
|
|
true => Value::True,
|
|
|
|
false => Value::False,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-25 22:37:45 +00:00
|
|
|
impl From<Uuid> for Value {
|
|
|
|
fn from(v: Uuid) -> Self {
|
|
|
|
Value::Uuid(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-13 17:36:41 +00:00
|
|
|
impl From<Param> for Value {
|
|
|
|
fn from(v: Param) -> Self {
|
|
|
|
Value::Param(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Idiom> for Value {
|
|
|
|
fn from(v: Idiom) -> Self {
|
|
|
|
Value::Idiom(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-09 08:18:08 +00:00
|
|
|
impl From<Model> for Value {
|
|
|
|
fn from(v: Model) -> Self {
|
|
|
|
Value::Model(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-13 17:36:41 +00:00
|
|
|
impl From<Table> for Value {
|
|
|
|
fn from(v: Table) -> Self {
|
|
|
|
Value::Table(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Thing> for Value {
|
|
|
|
fn from(v: Thing) -> Self {
|
|
|
|
Value::Thing(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Regex> for Value {
|
|
|
|
fn from(v: Regex) -> Self {
|
|
|
|
Value::Regex(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Array> for Value {
|
|
|
|
fn from(v: Array) -> Self {
|
|
|
|
Value::Array(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Object> for Value {
|
|
|
|
fn from(v: Object) -> Self {
|
|
|
|
Value::Object(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Number> for Value {
|
|
|
|
fn from(v: Number) -> Self {
|
|
|
|
Value::Number(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Strand> for Value {
|
|
|
|
fn from(v: Strand) -> Self {
|
|
|
|
Value::Strand(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Geometry> for Value {
|
|
|
|
fn from(v: Geometry) -> Self {
|
|
|
|
Value::Geometry(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Datetime> for Value {
|
|
|
|
fn from(v: Datetime) -> Self {
|
|
|
|
Value::Datetime(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Duration> for Value {
|
|
|
|
fn from(v: Duration) -> Self {
|
|
|
|
Value::Duration(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-01 14:55:48 +00:00
|
|
|
impl From<Constant> for Value {
|
|
|
|
fn from(v: Constant) -> Self {
|
|
|
|
Value::Constant(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-02-21 14:15:02 +00:00
|
|
|
impl From<Block> for Value {
|
|
|
|
fn from(v: Block) -> Self {
|
|
|
|
Value::Block(Box::new(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-08-28 22:19:59 +00:00
|
|
|
impl From<Range> for Value {
|
|
|
|
fn from(v: Range) -> Self {
|
|
|
|
Value::Range(Box::new(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-09 08:18:08 +00:00
|
|
|
impl From<Edges> for Value {
|
|
|
|
fn from(v: Edges) -> Self {
|
|
|
|
Value::Edges(Box::new(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-31 23:12:41 +00:00
|
|
|
impl From<Future> for Value {
|
|
|
|
fn from(v: Future) -> Self {
|
|
|
|
Value::Future(Box::new(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-13 17:36:41 +00:00
|
|
|
impl From<Function> for Value {
|
|
|
|
fn from(v: Function) -> Self {
|
|
|
|
Value::Function(Box::new(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-09 08:18:08 +00:00
|
|
|
impl From<Subquery> for Value {
|
|
|
|
fn from(v: Subquery) -> Self {
|
|
|
|
Value::Subquery(Box::new(v))
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-09 08:18:08 +00:00
|
|
|
impl From<Expression> for Value {
|
|
|
|
fn from(v: Expression) -> Self {
|
|
|
|
Value::Expression(Box::new(v))
|
2022-03-13 00:18:23 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-31 15:42:29 +00:00
|
|
|
impl From<Box<Edges>> for Value {
|
|
|
|
fn from(v: Box<Edges>) -> Self {
|
|
|
|
Value::Edges(v)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-13 17:36:41 +00:00
|
|
|
impl From<i8> for Value {
|
|
|
|
fn from(v: i8) -> Self {
|
|
|
|
Value::Number(Number::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<i16> for Value {
|
|
|
|
fn from(v: i16) -> Self {
|
|
|
|
Value::Number(Number::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<i32> for Value {
|
|
|
|
fn from(v: i32) -> Self {
|
|
|
|
Value::Number(Number::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<i64> for Value {
|
|
|
|
fn from(v: i64) -> Self {
|
|
|
|
Value::Number(Number::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<isize> for Value {
|
|
|
|
fn from(v: isize) -> Self {
|
|
|
|
Value::Number(Number::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<u8> for Value {
|
|
|
|
fn from(v: u8) -> Self {
|
|
|
|
Value::Number(Number::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<u16> for Value {
|
|
|
|
fn from(v: u16) -> Self {
|
|
|
|
Value::Number(Number::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<u32> for Value {
|
|
|
|
fn from(v: u32) -> Self {
|
|
|
|
Value::Number(Number::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<u64> for Value {
|
|
|
|
fn from(v: u64) -> Self {
|
|
|
|
Value::Number(Number::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<usize> for Value {
|
|
|
|
fn from(v: usize) -> Self {
|
|
|
|
Value::Number(Number::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<f32> for Value {
|
|
|
|
fn from(v: f32) -> Self {
|
|
|
|
Value::Number(Number::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<f64> for Value {
|
|
|
|
fn from(v: f64) -> Self {
|
|
|
|
Value::Number(Number::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-04 21:34:46 +00:00
|
|
|
impl From<BigDecimal> for Value {
|
|
|
|
fn from(v: BigDecimal) -> Self {
|
|
|
|
Value::Number(Number::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-13 17:36:41 +00:00
|
|
|
impl From<String> for Value {
|
|
|
|
fn from(v: String) -> Self {
|
|
|
|
Value::Strand(Strand::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-25 09:42:10 +00:00
|
|
|
impl From<&str> for Value {
|
2022-01-13 17:36:41 +00:00
|
|
|
fn from(v: &str) -> Self {
|
|
|
|
Value::Strand(Strand::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<DateTime<Utc>> for Value {
|
|
|
|
fn from(v: DateTime<Utc>) -> Self {
|
|
|
|
Value::Datetime(Datetime::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<(f64, f64)> for Value {
|
|
|
|
fn from(v: (f64, f64)) -> Self {
|
|
|
|
Value::Geometry(Geometry::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<[f64; 2]> for Value {
|
|
|
|
fn from(v: [f64; 2]) -> Self {
|
|
|
|
Value::Geometry(Geometry::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Point<f64>> for Value {
|
|
|
|
fn from(v: Point<f64>) -> Self {
|
|
|
|
Value::Geometry(Geometry::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-22 21:16:13 +00:00
|
|
|
impl From<Operation> for Value {
|
|
|
|
fn from(v: Operation) -> Self {
|
|
|
|
Value::Object(Object::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-17 09:02:01 +00:00
|
|
|
impl From<uuid::Uuid> for Value {
|
|
|
|
fn from(v: uuid::Uuid) -> Self {
|
|
|
|
Value::Uuid(Uuid(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-07-10 21:57:19 +00:00
|
|
|
impl From<Vec<&str>> for Value {
|
2022-01-13 17:36:41 +00:00
|
|
|
fn from(v: Vec<&str>) -> Self {
|
|
|
|
Value::Array(Array::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-07 09:52:34 +00:00
|
|
|
impl From<Vec<String>> for Value {
|
|
|
|
fn from(v: Vec<String>) -> Self {
|
|
|
|
Value::Array(Array::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-13 17:36:41 +00:00
|
|
|
impl From<Vec<i32>> for Value {
|
|
|
|
fn from(v: Vec<i32>) -> Self {
|
|
|
|
Value::Array(Array::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Vec<Value>> for Value {
|
|
|
|
fn from(v: Vec<Value>) -> Self {
|
|
|
|
Value::Array(Array::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-25 09:55:07 +00:00
|
|
|
impl From<Vec<Number>> for Value {
|
|
|
|
fn from(v: Vec<Number>) -> Self {
|
|
|
|
Value::Array(Array::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-22 21:16:13 +00:00
|
|
|
impl From<Vec<Operation>> for Value {
|
|
|
|
fn from(v: Vec<Operation>) -> Self {
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Array(Array::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<HashMap<String, Value>> for Value {
|
|
|
|
fn from(v: HashMap<String, Value>) -> Self {
|
|
|
|
Value::Object(Object::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<BTreeMap<String, Value>> for Value {
|
|
|
|
fn from(v: BTreeMap<String, Value>) -> Self {
|
|
|
|
Value::Object(Object::from(v))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Option<Value>> for Value {
|
|
|
|
fn from(v: Option<Value>) -> Self {
|
|
|
|
match v {
|
|
|
|
Some(v) => v,
|
|
|
|
None => Value::None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Option<String>> for Value {
|
|
|
|
fn from(v: Option<String>) -> Self {
|
|
|
|
match v {
|
|
|
|
Some(v) => Value::from(v),
|
|
|
|
None => Value::None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-18 07:24:36 +00:00
|
|
|
impl From<Id> for Value {
|
|
|
|
fn from(v: Id) -> Self {
|
|
|
|
match v {
|
2022-03-25 20:31:45 +00:00
|
|
|
Id::Number(v) => v.into(),
|
2022-11-23 08:58:56 +00:00
|
|
|
Id::String(v) => v.into(),
|
2022-08-26 23:23:24 +00:00
|
|
|
Id::Object(v) => v.into(),
|
|
|
|
Id::Array(v) => v.into(),
|
2022-03-18 07:24:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-26 14:01:09 +00:00
|
|
|
impl TryFrom<Value> for i64 {
|
|
|
|
type Error = Error;
|
|
|
|
fn try_from(value: Value) -> Result<Self, Self::Error> {
|
|
|
|
match value {
|
|
|
|
Value::Number(x) => x.try_into(),
|
|
|
|
_ => Err(Error::TryFromError(value.to_string(), "i64")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<Value> for f64 {
|
|
|
|
type Error = Error;
|
|
|
|
fn try_from(value: Value) -> Result<Self, Self::Error> {
|
|
|
|
match value {
|
|
|
|
Value::Number(x) => x.try_into(),
|
|
|
|
_ => Err(Error::TryFromError(value.to_string(), "f64")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<Value> for BigDecimal {
|
|
|
|
type Error = Error;
|
|
|
|
fn try_from(value: Value) -> Result<Self, Self::Error> {
|
|
|
|
match value {
|
|
|
|
Value::Number(x) => x.try_into(),
|
|
|
|
_ => Err(Error::TryFromError(value.to_string(), "BigDecimal")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<Value> for String {
|
|
|
|
type Error = Error;
|
|
|
|
fn try_from(value: Value) -> Result<Self, Self::Error> {
|
|
|
|
match value {
|
|
|
|
Value::Strand(x) => Ok(x.into()),
|
|
|
|
_ => Err(Error::TryFromError(value.to_string(), "String")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<Value> for bool {
|
|
|
|
type Error = Error;
|
|
|
|
fn try_from(value: Value) -> Result<Self, Self::Error> {
|
|
|
|
match value {
|
|
|
|
Value::True => Ok(true),
|
|
|
|
Value::False => Ok(false),
|
|
|
|
_ => Err(Error::TryFromError(value.to_string(), "bool")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<Value> for std::time::Duration {
|
|
|
|
type Error = Error;
|
|
|
|
fn try_from(value: Value) -> Result<Self, Self::Error> {
|
|
|
|
match value {
|
|
|
|
Value::Duration(x) => Ok(x.into()),
|
|
|
|
_ => Err(Error::TryFromError(value.to_string(), "time::Duration")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<Value> for DateTime<Utc> {
|
|
|
|
type Error = Error;
|
|
|
|
fn try_from(value: Value) -> Result<Self, Self::Error> {
|
|
|
|
match value {
|
|
|
|
Value::Datetime(x) => Ok(x.into()),
|
|
|
|
_ => Err(Error::TryFromError(value.to_string(), "chrono::DateTime<Utc>")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-28 11:35:01 +00:00
|
|
|
impl TryFrom<Value> for uuid::Uuid {
|
|
|
|
type Error = Error;
|
|
|
|
fn try_from(value: Value) -> Result<Self, Self::Error> {
|
|
|
|
match value {
|
|
|
|
Value::Uuid(x) => Ok(x.into()),
|
|
|
|
_ => Err(Error::TryFromError(value.to_string(), "uuid::Uuid")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<Value> for Vec<Value> {
|
|
|
|
type Error = Error;
|
|
|
|
fn try_from(value: Value) -> Result<Self, Self::Error> {
|
|
|
|
match value {
|
|
|
|
Value::Array(x) => Ok(x.into()),
|
|
|
|
_ => Err(Error::TryFromError(value.to_string(), "Vec<Value>")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl TryFrom<Value> for Object {
|
|
|
|
type Error = Error;
|
|
|
|
fn try_from(value: Value) -> Result<Self, Self::Error> {
|
|
|
|
match value {
|
|
|
|
Value::Object(x) => Ok(x),
|
|
|
|
_ => Err(Error::TryFromError(value.to_string(), "Object")),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-13 17:36:41 +00:00
|
|
|
impl Value {
|
2022-02-06 01:14:56 +00:00
|
|
|
// -----------------------------------
|
|
|
|
// Initial record value
|
|
|
|
// -----------------------------------
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Create an empty Object Value
|
2022-02-06 01:14:56 +00:00
|
|
|
pub fn base() -> Self {
|
|
|
|
Value::Object(Object::default())
|
|
|
|
}
|
|
|
|
|
2022-01-13 17:36:41 +00:00
|
|
|
// -----------------------------------
|
|
|
|
// Builtin types
|
|
|
|
// -----------------------------------
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Convert this Value to a Result
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn ok(self) -> Result<Value, Error> {
|
|
|
|
Ok(self)
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Convert this Value to an Option
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn output(self) -> Option<Value> {
|
|
|
|
match self {
|
|
|
|
Value::None => None,
|
|
|
|
_ => Some(self),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
// -----------------------------------
|
|
|
|
// Simple value detection
|
|
|
|
// -----------------------------------
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is NONE or NULL
|
2023-01-20 00:54:09 +00:00
|
|
|
pub fn is_none_or_null(&self) -> bool {
|
|
|
|
matches!(self, Value::None | Value::Null)
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is NONE
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn is_none(&self) -> bool {
|
2023-01-06 10:26:25 +00:00
|
|
|
matches!(self, Value::None)
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is NULL
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn is_null(&self) -> bool {
|
2023-01-06 10:26:25 +00:00
|
|
|
matches!(self, Value::Null)
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value not NONE or NULL
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn is_some(&self) -> bool {
|
2022-12-02 21:48:10 +00:00
|
|
|
!self.is_none() && !self.is_null()
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is TRUE or 'true'
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn is_true(&self) -> bool {
|
|
|
|
match self {
|
|
|
|
Value::True => true,
|
2022-09-01 00:27:52 +00:00
|
|
|
Value::Strand(v) => v.eq_ignore_ascii_case("true"),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is FALSE or 'false'
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn is_false(&self) -> bool {
|
|
|
|
match self {
|
|
|
|
Value::False => true,
|
2022-09-01 00:27:52 +00:00
|
|
|
Value::Strand(v) => v.eq_ignore_ascii_case("false"),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is truthy
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn is_truthy(&self) -> bool {
|
|
|
|
match self {
|
|
|
|
Value::True => true,
|
|
|
|
Value::False => false,
|
2022-10-05 18:22:32 +00:00
|
|
|
Value::Uuid(_) => true,
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Thing(_) => true,
|
|
|
|
Value::Geometry(_) => true,
|
2022-05-04 16:14:40 +00:00
|
|
|
Value::Array(v) => !v.is_empty(),
|
2022-05-04 13:09:57 +00:00
|
|
|
Value::Object(v) => !v.is_empty(),
|
2022-09-01 00:27:52 +00:00
|
|
|
Value::Strand(v) => !v.is_empty() && !v.eq_ignore_ascii_case("false"),
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Number(v) => v.is_truthy(),
|
2022-05-05 06:10:05 +00:00
|
|
|
Value::Duration(v) => v.as_nanos() > 0,
|
2022-05-05 05:01:00 +00:00
|
|
|
Value::Datetime(v) => v.timestamp() > 0,
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a UUID
|
2022-07-04 01:02:46 +00:00
|
|
|
pub fn is_uuid(&self) -> bool {
|
|
|
|
matches!(self, Value::Uuid(_))
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Thing
|
2022-07-04 01:02:46 +00:00
|
|
|
pub fn is_thing(&self) -> bool {
|
|
|
|
matches!(self, Value::Thing(_))
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Model
|
2022-09-22 15:58:59 +00:00
|
|
|
pub fn is_model(&self) -> bool {
|
|
|
|
matches!(self, Value::Model(_))
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Range
|
2022-09-22 15:58:59 +00:00
|
|
|
pub fn is_range(&self) -> bool {
|
|
|
|
matches!(self, Value::Range(_))
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Table
|
2023-02-14 11:01:31 +00:00
|
|
|
pub fn is_table(&self) -> bool {
|
|
|
|
matches!(self, Value::Table(_))
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Strand
|
2022-07-04 01:02:46 +00:00
|
|
|
pub fn is_strand(&self) -> bool {
|
|
|
|
matches!(self, Value::Strand(_))
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is an Array
|
2022-07-04 01:02:46 +00:00
|
|
|
pub fn is_array(&self) -> bool {
|
|
|
|
matches!(self, Value::Array(_))
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is an Object
|
2022-07-04 01:02:46 +00:00
|
|
|
pub fn is_object(&self) -> bool {
|
|
|
|
matches!(self, Value::Object(_))
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Number
|
2022-10-19 22:54:41 +00:00
|
|
|
pub fn is_number(&self) -> bool {
|
|
|
|
matches!(self, Value::Number(_))
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is an int Number
|
2022-10-17 01:57:53 +00:00
|
|
|
pub fn is_int(&self) -> bool {
|
|
|
|
matches!(self, Value::Number(Number::Int(_)))
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a float Number
|
2022-10-17 01:57:53 +00:00
|
|
|
pub fn is_float(&self) -> bool {
|
|
|
|
matches!(self, Value::Number(Number::Float(_)))
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a decimal Number
|
2022-10-17 01:57:53 +00:00
|
|
|
pub fn is_decimal(&self) -> bool {
|
|
|
|
matches!(self, Value::Number(Number::Decimal(_)))
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Number and is an integer
|
2022-10-17 01:57:53 +00:00
|
|
|
pub fn is_integer(&self) -> bool {
|
|
|
|
matches!(self, Value::Number(v) if v.is_integer())
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Number and is positive
|
2022-10-17 01:57:53 +00:00
|
|
|
pub fn is_positive(&self) -> bool {
|
|
|
|
matches!(self, Value::Number(v) if v.is_positive())
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Number and is negative
|
2022-12-09 16:02:36 +00:00
|
|
|
pub fn is_negative(&self) -> bool {
|
|
|
|
matches!(self, Value::Number(v) if v.is_negative())
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Number and is zero or positive
|
2022-12-09 16:02:36 +00:00
|
|
|
pub fn is_zero_or_positive(&self) -> bool {
|
|
|
|
matches!(self, Value::Number(v) if v.is_zero_or_positive())
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Number and is zero or negative
|
2022-12-09 16:02:36 +00:00
|
|
|
pub fn is_zero_or_negative(&self) -> bool {
|
|
|
|
matches!(self, Value::Number(v) if v.is_zero_or_negative())
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Datetime
|
2022-10-25 13:06:02 +00:00
|
|
|
pub fn is_datetime(&self) -> bool {
|
|
|
|
matches!(self, Value::Datetime(_))
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Thing of a specific type
|
2022-04-04 22:26:54 +00:00
|
|
|
pub fn is_type_record(&self, types: &[Table]) -> bool {
|
|
|
|
match self {
|
2023-03-26 22:54:57 +00:00
|
|
|
Value::Thing(v) => types.is_empty() || types.iter().any(|tb| tb.0 == v.tb),
|
2022-04-04 22:26:54 +00:00
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is a Geometry of a specific type
|
2022-04-04 22:26:54 +00:00
|
|
|
pub fn is_type_geometry(&self, types: &[String]) -> bool {
|
|
|
|
match self {
|
|
|
|
Value::Geometry(Geometry::Point(_)) => {
|
2022-09-01 00:27:52 +00:00
|
|
|
types.iter().any(|t| matches!(t.as_str(), "feature" | "point"))
|
2022-04-04 22:26:54 +00:00
|
|
|
}
|
|
|
|
Value::Geometry(Geometry::Line(_)) => {
|
2022-09-01 00:27:52 +00:00
|
|
|
types.iter().any(|t| matches!(t.as_str(), "feature" | "line"))
|
2022-04-04 22:26:54 +00:00
|
|
|
}
|
|
|
|
Value::Geometry(Geometry::Polygon(_)) => {
|
2022-09-01 00:27:52 +00:00
|
|
|
types.iter().any(|t| matches!(t.as_str(), "feature" | "polygon"))
|
2022-04-04 22:26:54 +00:00
|
|
|
}
|
|
|
|
Value::Geometry(Geometry::MultiPoint(_)) => {
|
2022-09-01 00:27:52 +00:00
|
|
|
types.iter().any(|t| matches!(t.as_str(), "feature" | "multipoint"))
|
2022-04-04 22:26:54 +00:00
|
|
|
}
|
|
|
|
Value::Geometry(Geometry::MultiLine(_)) => {
|
2022-09-01 00:27:52 +00:00
|
|
|
types.iter().any(|t| matches!(t.as_str(), "feature" | "multiline"))
|
2022-04-04 22:26:54 +00:00
|
|
|
}
|
|
|
|
Value::Geometry(Geometry::MultiPolygon(_)) => {
|
2022-09-01 00:27:52 +00:00
|
|
|
types.iter().any(|t| matches!(t.as_str(), "feature" | "multipolygon"))
|
2022-04-04 22:26:54 +00:00
|
|
|
}
|
|
|
|
Value::Geometry(Geometry::Collection(_)) => {
|
2022-09-01 00:27:52 +00:00
|
|
|
types.iter().any(|t| matches!(t.as_str(), "feature" | "collection"))
|
2022-04-04 22:26:54 +00:00
|
|
|
}
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-13 17:36:41 +00:00
|
|
|
// -----------------------------------
|
|
|
|
// Simple conversion of value
|
|
|
|
// -----------------------------------
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Convert this Value into an i64
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn as_int(self) -> i64 {
|
|
|
|
match self {
|
|
|
|
Value::True => 1,
|
2023-03-29 18:23:13 +00:00
|
|
|
Value::Strand(v) => Number::from(v.as_str()).as_int(),
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Number(v) => v.as_int(),
|
2022-05-05 06:10:05 +00:00
|
|
|
Value::Duration(v) => v.as_secs() as i64,
|
2022-05-05 05:01:00 +00:00
|
|
|
Value::Datetime(v) => v.timestamp(),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => 0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Convert this Value into a f64
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn as_float(self) -> f64 {
|
|
|
|
match self {
|
|
|
|
Value::True => 1.0,
|
2022-05-05 04:30:32 +00:00
|
|
|
Value::Strand(v) => v.parse::<f64>().unwrap_or(0.0),
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Number(v) => v.as_float(),
|
2022-05-05 06:10:05 +00:00
|
|
|
Value::Duration(v) => v.as_secs() as f64,
|
2022-05-05 05:01:00 +00:00
|
|
|
Value::Datetime(v) => v.timestamp() as f64,
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => 0.0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Convert this Value into a BigDecimal
|
2022-03-17 21:01:29 +00:00
|
|
|
pub fn as_decimal(self) -> BigDecimal {
|
2022-01-13 17:36:41 +00:00
|
|
|
match self {
|
2022-03-17 21:01:29 +00:00
|
|
|
Value::True => BigDecimal::from(1),
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Number(v) => v.as_decimal(),
|
2022-03-17 21:01:29 +00:00
|
|
|
Value::Strand(v) => BigDecimal::from_str(v.as_str()).unwrap_or_default(),
|
2022-05-05 06:10:05 +00:00
|
|
|
Value::Duration(v) => v.as_secs().into(),
|
2022-05-05 05:01:00 +00:00
|
|
|
Value::Datetime(v) => v.timestamp().into(),
|
2022-03-17 21:01:29 +00:00
|
|
|
_ => BigDecimal::default(),
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Convert this Value into a Number
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn as_number(self) -> Number {
|
|
|
|
match self {
|
|
|
|
Value::True => Number::from(1),
|
|
|
|
Value::Number(v) => v,
|
|
|
|
Value::Strand(v) => Number::from(v.as_str()),
|
2022-05-05 06:10:05 +00:00
|
|
|
Value::Duration(v) => v.as_secs().into(),
|
2022-05-05 05:01:00 +00:00
|
|
|
Value::Datetime(v) => v.timestamp().into(),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => Number::default(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Convert this Value into a Strand
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn as_strand(self) -> Strand {
|
|
|
|
match self {
|
|
|
|
Value::Strand(v) => v,
|
2022-10-16 16:19:36 +00:00
|
|
|
Value::Uuid(v) => v.to_raw().into(),
|
|
|
|
Value::Datetime(v) => v.to_raw().into(),
|
|
|
|
_ => self.to_string().into(),
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Convert this Value into a Datetime
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn as_datetime(self) -> Datetime {
|
|
|
|
match self {
|
|
|
|
Value::Strand(v) => Datetime::from(v.as_str()),
|
|
|
|
Value::Datetime(v) => v,
|
|
|
|
_ => Datetime::default(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Convert this Value into a Duration
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn as_duration(self) -> Duration {
|
|
|
|
match self {
|
|
|
|
Value::Strand(v) => Duration::from(v.as_str()),
|
|
|
|
Value::Duration(v) => v,
|
|
|
|
_ => Duration::default(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Convert this Value into a String
|
2022-05-05 04:30:32 +00:00
|
|
|
pub fn as_string(self) -> String {
|
|
|
|
match self {
|
2023-02-11 18:17:02 +00:00
|
|
|
Value::Strand(v) => v.0,
|
|
|
|
Value::Uuid(v) => v.to_raw(),
|
|
|
|
Value::Datetime(v) => v.to_raw(),
|
2022-05-05 04:30:32 +00:00
|
|
|
_ => self.to_string(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Convert this Value into a usize
|
2022-10-17 01:57:53 +00:00
|
|
|
pub fn as_usize(self) -> usize {
|
|
|
|
match self {
|
|
|
|
Value::Number(v) => v.as_usize(),
|
|
|
|
_ => 0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Converts this Value into an unquoted String
|
2023-01-14 18:33:12 +00:00
|
|
|
pub fn as_raw_string(self) -> String {
|
|
|
|
match self {
|
|
|
|
Value::Strand(v) => v.0,
|
|
|
|
Value::Uuid(v) => v.to_raw(),
|
|
|
|
Value::Datetime(v) => v.to_raw(),
|
2023-02-11 18:17:02 +00:00
|
|
|
_ => self.to_string(),
|
2023-01-14 18:33:12 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-13 17:36:41 +00:00
|
|
|
// -----------------------------------
|
|
|
|
// Expensive conversion of value
|
|
|
|
// -----------------------------------
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Convert this Value into a Number
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn to_number(&self) -> Number {
|
|
|
|
match self {
|
|
|
|
Value::True => Number::from(1),
|
|
|
|
Value::Number(v) => v.clone(),
|
|
|
|
Value::Strand(v) => Number::from(v.as_str()),
|
2022-05-05 06:10:05 +00:00
|
|
|
Value::Duration(v) => v.as_secs().into(),
|
2022-05-05 05:01:00 +00:00
|
|
|
Value::Datetime(v) => v.timestamp().into(),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => Number::default(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Converts this Value into a Strand
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn to_strand(&self) -> Strand {
|
|
|
|
match self {
|
|
|
|
Value::Strand(v) => v.clone(),
|
2022-10-16 16:19:36 +00:00
|
|
|
Value::Uuid(v) => v.to_raw().into(),
|
|
|
|
Value::Datetime(v) => v.to_raw().into(),
|
|
|
|
_ => self.to_string().into(),
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Converts this Value into a Datetime
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn to_datetime(&self) -> Datetime {
|
|
|
|
match self {
|
|
|
|
Value::Strand(v) => Datetime::from(v.as_str()),
|
|
|
|
Value::Datetime(v) => v.clone(),
|
|
|
|
_ => Datetime::default(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Converts this Value into a Duration
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn to_duration(&self) -> Duration {
|
|
|
|
match self {
|
|
|
|
Value::Strand(v) => Duration::from(v.as_str()),
|
|
|
|
Value::Duration(v) => v.clone(),
|
|
|
|
_ => Duration::default(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Converts this Value into an unquoted String
|
2023-01-14 18:33:12 +00:00
|
|
|
pub fn to_raw_string(&self) -> String {
|
|
|
|
match self {
|
|
|
|
Value::Strand(v) => v.0.to_owned(),
|
|
|
|
Value::Uuid(v) => v.to_raw(),
|
|
|
|
Value::Datetime(v) => v.to_raw(),
|
|
|
|
_ => self.to_string(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Converts this Value into a field name
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn to_idiom(&self) -> Idiom {
|
2022-05-31 19:44:27 +00:00
|
|
|
match self {
|
|
|
|
Value::Idiom(v) => v.simplify(),
|
2023-03-31 15:42:29 +00:00
|
|
|
Value::Param(v) => v.to_raw().into(),
|
2022-05-31 19:44:27 +00:00
|
|
|
Value::Strand(v) => v.0.to_string().into(),
|
|
|
|
Value::Datetime(v) => v.0.to_string().into(),
|
2023-03-25 19:44:03 +00:00
|
|
|
Value::Future(_) => "future".to_string().into(),
|
|
|
|
Value::Function(v) => v.to_idiom(),
|
2022-05-31 19:44:27 +00:00
|
|
|
_ => self.to_string().into(),
|
|
|
|
}
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Try to convert this Value into a set of JSONPatch operations
|
2022-07-07 09:58:35 +00:00
|
|
|
pub fn to_operations(&self) -> Result<Vec<Operation>, Error> {
|
|
|
|
match self {
|
|
|
|
Value::Array(v) => v
|
|
|
|
.iter()
|
|
|
|
.map(|v| match v {
|
|
|
|
Value::Object(v) => v.to_operation(),
|
|
|
|
_ => Err(Error::InvalidPatch {
|
|
|
|
message: String::from("Operation must be an object"),
|
|
|
|
}),
|
|
|
|
})
|
|
|
|
.collect::<Result<Vec<_>, Error>>(),
|
|
|
|
_ => Err(Error::InvalidPatch {
|
|
|
|
message: String::from("Operations must be an array"),
|
|
|
|
}),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-04 21:34:46 +00:00
|
|
|
// -----------------------------------
|
|
|
|
// Simple conversion of value
|
|
|
|
// -----------------------------------
|
|
|
|
|
|
|
|
pub fn make_bool(self) -> Value {
|
2022-04-07 14:44:55 +00:00
|
|
|
match self {
|
|
|
|
Value::True | Value::False => self,
|
|
|
|
_ => self.is_truthy().into(),
|
|
|
|
}
|
2022-04-04 21:34:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn make_int(self) -> Value {
|
2022-04-07 14:44:55 +00:00
|
|
|
match self {
|
|
|
|
Value::Number(Number::Int(_)) => self,
|
|
|
|
_ => self.as_int().into(),
|
|
|
|
}
|
2022-04-04 21:34:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn make_float(self) -> Value {
|
2022-04-07 14:44:55 +00:00
|
|
|
match self {
|
|
|
|
Value::Number(Number::Float(_)) => self,
|
|
|
|
_ => self.as_float().into(),
|
|
|
|
}
|
2022-04-04 21:34:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn make_decimal(self) -> Value {
|
2022-04-07 14:44:55 +00:00
|
|
|
match self {
|
|
|
|
Value::Number(Number::Decimal(_)) => self,
|
|
|
|
_ => self.as_decimal().into(),
|
|
|
|
}
|
2022-04-04 21:34:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn make_number(self) -> Value {
|
2022-04-07 14:44:55 +00:00
|
|
|
match self {
|
|
|
|
Value::Number(_) => self,
|
|
|
|
_ => self.as_number().into(),
|
|
|
|
}
|
2022-04-04 21:34:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn make_strand(self) -> Value {
|
2022-04-07 14:44:55 +00:00
|
|
|
match self {
|
|
|
|
Value::Strand(_) => self,
|
|
|
|
_ => self.as_strand().into(),
|
|
|
|
}
|
2022-04-04 21:34:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn make_datetime(self) -> Value {
|
2022-04-07 14:44:55 +00:00
|
|
|
match self {
|
|
|
|
Value::Datetime(_) => self,
|
|
|
|
_ => self.as_datetime().into(),
|
|
|
|
}
|
2022-04-04 21:34:46 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
pub fn make_duration(self) -> Value {
|
2022-04-07 14:44:55 +00:00
|
|
|
match self {
|
|
|
|
Value::Duration(_) => self,
|
|
|
|
_ => self.as_duration().into(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-25 13:08:09 +00:00
|
|
|
pub fn could_be_table(self) -> Value {
|
2022-07-04 01:02:46 +00:00
|
|
|
match self {
|
2022-10-25 13:08:09 +00:00
|
|
|
Value::Strand(v) => Table::from(v.0).into(),
|
|
|
|
_ => self,
|
2022-07-04 01:02:46 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-04-07 14:44:55 +00:00
|
|
|
pub fn convert_to(self, kind: &Kind) -> Value {
|
|
|
|
match kind {
|
|
|
|
Kind::Any => self,
|
|
|
|
Kind::Bool => self.make_bool(),
|
|
|
|
Kind::Int => self.make_int(),
|
|
|
|
Kind::Float => self.make_float(),
|
|
|
|
Kind::Decimal => self.make_decimal(),
|
|
|
|
Kind::Number => self.make_number(),
|
|
|
|
Kind::String => self.make_strand(),
|
|
|
|
Kind::Datetime => self.make_datetime(),
|
|
|
|
Kind::Duration => self.make_duration(),
|
|
|
|
Kind::Array => match self {
|
|
|
|
Value::Array(_) => self,
|
|
|
|
_ => Value::None,
|
|
|
|
},
|
|
|
|
Kind::Object => match self {
|
|
|
|
Value::Object(_) => self,
|
|
|
|
_ => Value::None,
|
|
|
|
},
|
|
|
|
Kind::Record(t) => match self.is_type_record(t) {
|
|
|
|
true => self,
|
|
|
|
_ => Value::None,
|
|
|
|
},
|
|
|
|
Kind::Geometry(t) => match self.is_type_geometry(t) {
|
|
|
|
true => self,
|
|
|
|
_ => Value::None,
|
|
|
|
},
|
|
|
|
}
|
2022-04-04 21:34:46 +00:00
|
|
|
}
|
|
|
|
|
2022-06-20 11:29:06 +00:00
|
|
|
// -----------------------------------
|
|
|
|
// Record ID extraction
|
|
|
|
// -----------------------------------
|
|
|
|
|
|
|
|
/// Fetch the record id if there is one
|
2022-09-19 11:28:41 +00:00
|
|
|
pub fn record(self) -> Option<Thing> {
|
2022-06-20 11:29:06 +00:00
|
|
|
match self {
|
2023-03-10 20:23:49 +00:00
|
|
|
// This is an object so look for the id field
|
2022-06-20 11:29:06 +00:00
|
|
|
Value::Object(mut v) => match v.remove("id") {
|
|
|
|
Some(Value::Thing(v)) => Some(v),
|
|
|
|
_ => None,
|
|
|
|
},
|
2023-03-10 20:23:49 +00:00
|
|
|
// This is an array so take the first item
|
2022-06-20 11:29:06 +00:00
|
|
|
Value::Array(mut v) => match v.len() {
|
2022-09-19 11:28:41 +00:00
|
|
|
1 => v.remove(0).record(),
|
2022-06-20 11:29:06 +00:00
|
|
|
_ => None,
|
|
|
|
},
|
2023-03-10 20:23:49 +00:00
|
|
|
// This is a record id already
|
2022-06-20 11:29:06 +00:00
|
|
|
Value::Thing(v) => Some(v),
|
2023-03-10 20:23:49 +00:00
|
|
|
// There is no valid record id
|
2022-06-20 11:29:06 +00:00
|
|
|
_ => None,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-31 19:44:27 +00:00
|
|
|
// -----------------------------------
|
|
|
|
// JSON Path conversion
|
|
|
|
// -----------------------------------
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Converts this Value into a JSONPatch path
|
2023-01-08 17:11:35 +00:00
|
|
|
pub(crate) fn jsonpath(&self) -> Idiom {
|
2022-05-31 19:44:27 +00:00
|
|
|
self.to_strand()
|
|
|
|
.as_str()
|
|
|
|
.trim_start_matches('/')
|
|
|
|
.split(&['.', '/'][..])
|
|
|
|
.map(Part::from)
|
|
|
|
.collect::<Vec<Part>>()
|
|
|
|
.into()
|
|
|
|
}
|
|
|
|
|
2023-01-08 17:11:35 +00:00
|
|
|
// -----------------------------------
|
|
|
|
// JSON Path conversion
|
|
|
|
// -----------------------------------
|
|
|
|
|
|
|
|
/// Checkes whether this value is a static value
|
|
|
|
pub(crate) fn is_static(&self) -> bool {
|
|
|
|
match self {
|
|
|
|
Value::None => true,
|
|
|
|
Value::Null => true,
|
|
|
|
Value::False => true,
|
|
|
|
Value::True => true,
|
|
|
|
Value::Uuid(_) => true,
|
|
|
|
Value::Number(_) => true,
|
|
|
|
Value::Strand(_) => true,
|
|
|
|
Value::Duration(_) => true,
|
|
|
|
Value::Datetime(_) => true,
|
|
|
|
Value::Geometry(_) => true,
|
|
|
|
Value::Array(v) => v.iter().all(Value::is_static),
|
|
|
|
Value::Object(v) => v.values().all(Value::is_static),
|
|
|
|
Value::Constant(_) => true,
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-13 17:36:41 +00:00
|
|
|
// -----------------------------------
|
|
|
|
// Value operations
|
|
|
|
// -----------------------------------
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value is equal to another Value
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn equal(&self, other: &Value) -> bool {
|
|
|
|
match self {
|
2022-03-04 16:01:32 +00:00
|
|
|
Value::None => other.is_none(),
|
|
|
|
Value::Null => other.is_null(),
|
|
|
|
Value::True => other.is_true(),
|
|
|
|
Value::False => other.is_false(),
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Thing(v) => match other {
|
|
|
|
Value::Thing(w) => v == w,
|
2022-05-07 13:57:14 +00:00
|
|
|
Value::Regex(w) => match w.regex() {
|
2022-03-04 16:01:32 +00:00
|
|
|
Some(ref r) => r.is_match(v.to_string().as_str()),
|
2022-01-13 17:36:41 +00:00
|
|
|
None => false,
|
|
|
|
},
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
Value::Regex(v) => match other {
|
|
|
|
Value::Regex(w) => v == w,
|
2022-05-07 13:57:14 +00:00
|
|
|
Value::Number(w) => match v.regex() {
|
2022-01-13 17:36:41 +00:00
|
|
|
Some(ref r) => r.is_match(w.to_string().as_str()),
|
|
|
|
None => false,
|
|
|
|
},
|
2022-05-07 13:57:14 +00:00
|
|
|
Value::Strand(w) => match v.regex() {
|
2022-01-13 17:36:41 +00:00
|
|
|
Some(ref r) => r.is_match(w.as_str()),
|
|
|
|
None => false,
|
|
|
|
},
|
|
|
|
_ => false,
|
|
|
|
},
|
2022-06-25 22:37:45 +00:00
|
|
|
Value::Uuid(v) => match other {
|
|
|
|
Value::Uuid(w) => v == w,
|
|
|
|
_ => false,
|
|
|
|
},
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Array(v) => match other {
|
|
|
|
Value::Array(w) => v == w,
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
Value::Object(v) => match other {
|
|
|
|
Value::Object(w) => v == w,
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
Value::Strand(v) => match other {
|
|
|
|
Value::Strand(w) => v == w,
|
2022-05-07 13:57:14 +00:00
|
|
|
Value::Regex(w) => match w.regex() {
|
2022-03-04 16:01:32 +00:00
|
|
|
Some(ref r) => r.is_match(v.as_str()),
|
2022-01-13 17:36:41 +00:00
|
|
|
None => false,
|
|
|
|
},
|
|
|
|
_ => v == &other.to_strand(),
|
|
|
|
},
|
|
|
|
Value::Number(v) => match other {
|
|
|
|
Value::Number(w) => v == w,
|
|
|
|
Value::Strand(_) => v == &other.to_number(),
|
2022-05-07 13:57:14 +00:00
|
|
|
Value::Regex(w) => match w.regex() {
|
2022-03-04 16:01:32 +00:00
|
|
|
Some(ref r) => r.is_match(v.to_string().as_str()),
|
2022-01-13 17:36:41 +00:00
|
|
|
None => false,
|
|
|
|
},
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
Value::Geometry(v) => match other {
|
|
|
|
Value::Geometry(w) => v == w,
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
Value::Duration(v) => match other {
|
|
|
|
Value::Duration(w) => v == w,
|
|
|
|
Value::Strand(_) => v == &other.to_duration(),
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
Value::Datetime(v) => match other {
|
|
|
|
Value::Datetime(w) => v == w,
|
|
|
|
Value::Strand(_) => v == &other.to_datetime(),
|
|
|
|
_ => false,
|
|
|
|
},
|
2022-10-20 14:54:35 +00:00
|
|
|
_ => self == other,
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if all Values in an Array are equal to another Value
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn all_equal(&self, other: &Value) -> bool {
|
|
|
|
match self {
|
2022-05-04 16:14:40 +00:00
|
|
|
Value::Array(v) => v.iter().all(|v| v.equal(other)),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => self.equal(other),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if any Values in an Array are equal to another Value
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn any_equal(&self, other: &Value) -> bool {
|
|
|
|
match self {
|
2022-05-04 16:14:40 +00:00
|
|
|
Value::Array(v) => v.iter().any(|v| v.equal(other)),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => self.equal(other),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Fuzzy check if this Value is equal to another Value
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn fuzzy(&self, other: &Value) -> bool {
|
|
|
|
match self {
|
|
|
|
Value::Strand(v) => match other {
|
|
|
|
Value::Strand(w) => MATCHER.fuzzy_match(v.as_str(), w.as_str()).is_some(),
|
2023-02-04 08:15:18 +00:00
|
|
|
_ => MATCHER.fuzzy_match(v.as_str(), other.to_raw_string().as_str()).is_some(),
|
2022-01-13 17:36:41 +00:00
|
|
|
},
|
|
|
|
_ => self.equal(other),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Fuzzy check if all Values in an Array are equal to another Value
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn all_fuzzy(&self, other: &Value) -> bool {
|
|
|
|
match self {
|
2022-05-04 16:14:40 +00:00
|
|
|
Value::Array(v) => v.iter().all(|v| v.fuzzy(other)),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => self.fuzzy(other),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Fuzzy check if any Values in an Array are equal to another Value
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn any_fuzzy(&self, other: &Value) -> bool {
|
|
|
|
match self {
|
2022-05-04 16:14:40 +00:00
|
|
|
Value::Array(v) => v.iter().any(|v| v.fuzzy(other)),
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => self.fuzzy(other),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value contains another Value
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn contains(&self, other: &Value) -> bool {
|
|
|
|
match self {
|
2022-05-04 16:14:40 +00:00
|
|
|
Value::Array(v) => v.iter().any(|v| v.equal(other)),
|
2022-10-17 22:05:35 +00:00
|
|
|
Value::Thing(v) => match other {
|
|
|
|
Value::Strand(w) => v.to_string().contains(w.as_str()),
|
2023-02-04 08:15:18 +00:00
|
|
|
_ => v.to_string().contains(other.to_raw_string().as_str()),
|
2022-10-17 22:05:35 +00:00
|
|
|
},
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Strand(v) => match other {
|
2022-05-05 04:30:32 +00:00
|
|
|
Value::Strand(w) => v.contains(w.as_str()),
|
2023-02-04 08:15:18 +00:00
|
|
|
_ => v.contains(other.to_raw_string().as_str()),
|
2022-01-13 17:36:41 +00:00
|
|
|
},
|
|
|
|
Value::Geometry(v) => match other {
|
|
|
|
Value::Geometry(w) => v.contains(w),
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if all Values in an Array contain another Value
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn contains_all(&self, other: &Value) -> bool {
|
|
|
|
match other {
|
2022-05-04 16:14:40 +00:00
|
|
|
Value::Array(v) => v.iter().all(|v| match self {
|
|
|
|
Value::Array(w) => w.iter().any(|w| v.equal(w)),
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Geometry(_) => self.contains(v),
|
|
|
|
_ => false,
|
|
|
|
}),
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if any Values in an Array contain another Value
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn contains_any(&self, other: &Value) -> bool {
|
|
|
|
match other {
|
2022-05-04 16:14:40 +00:00
|
|
|
Value::Array(v) => v.iter().any(|v| match self {
|
|
|
|
Value::Array(w) => w.iter().any(|w| v.equal(w)),
|
2022-01-13 17:36:41 +00:00
|
|
|
Value::Geometry(_) => self.contains(v),
|
|
|
|
_ => false,
|
|
|
|
}),
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Check if this Value intersects another Value
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn intersects(&self, other: &Value) -> bool {
|
|
|
|
match self {
|
|
|
|
Value::Geometry(v) => match other {
|
|
|
|
Value::Geometry(w) => v.intersects(w),
|
|
|
|
_ => false,
|
|
|
|
},
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
2022-04-27 15:21:51 +00:00
|
|
|
|
|
|
|
// -----------------------------------
|
|
|
|
// Sorting operations
|
|
|
|
// -----------------------------------
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Compare this Value to another Value lexicographically
|
2022-04-27 15:21:51 +00:00
|
|
|
pub fn lexical_cmp(&self, other: &Value) -> Option<Ordering> {
|
|
|
|
match (self, other) {
|
2023-03-26 21:13:43 +00:00
|
|
|
(Value::Strand(a), Value::Strand(b)) => Some(lexicmp::lexical_cmp(a, b)),
|
2022-04-27 15:21:51 +00:00
|
|
|
_ => self.partial_cmp(other),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Compare this Value to another Value using natrual numerical comparison
|
2022-04-27 15:21:51 +00:00
|
|
|
pub fn natural_cmp(&self, other: &Value) -> Option<Ordering> {
|
|
|
|
match (self, other) {
|
2023-03-26 21:13:43 +00:00
|
|
|
(Value::Strand(a), Value::Strand(b)) => Some(lexicmp::natural_cmp(a, b)),
|
2022-04-27 15:21:51 +00:00
|
|
|
_ => self.partial_cmp(other),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-10 20:23:49 +00:00
|
|
|
/// Compare this Value to another Value lexicographically and using natrual numerical comparison
|
2022-04-27 15:21:51 +00:00
|
|
|
pub fn natural_lexical_cmp(&self, other: &Value) -> Option<Ordering> {
|
|
|
|
match (self, other) {
|
2023-03-26 21:13:43 +00:00
|
|
|
(Value::Strand(a), Value::Strand(b)) => Some(lexicmp::natural_lexical_cmp(a, b)),
|
2022-04-27 15:21:51 +00:00
|
|
|
_ => self.partial_cmp(other),
|
|
|
|
}
|
|
|
|
}
|
2022-12-18 14:56:07 +00:00
|
|
|
|
|
|
|
// -----------------------------------
|
|
|
|
// Mathematical operations
|
|
|
|
// -----------------------------------
|
|
|
|
|
|
|
|
pub fn pow(self, other: Value) -> Value {
|
|
|
|
self.as_number().pow(other.as_number()).into()
|
|
|
|
}
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
impl fmt::Display for Value {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
2023-01-19 09:53:33 +00:00
|
|
|
let mut f = Pretty::from(f);
|
2022-01-13 17:36:41 +00:00
|
|
|
match self {
|
|
|
|
Value::None => write!(f, "NONE"),
|
|
|
|
Value::Null => write!(f, "NULL"),
|
|
|
|
Value::True => write!(f, "true"),
|
|
|
|
Value::False => write!(f, "false"),
|
2023-02-03 11:47:07 +00:00
|
|
|
Value::Number(v) => write!(f, "{v}"),
|
|
|
|
Value::Strand(v) => write!(f, "{v}"),
|
|
|
|
Value::Duration(v) => write!(f, "{v}"),
|
|
|
|
Value::Datetime(v) => write!(f, "{v}"),
|
|
|
|
Value::Uuid(v) => write!(f, "{v}"),
|
|
|
|
Value::Array(v) => write!(f, "{v}"),
|
|
|
|
Value::Object(v) => write!(f, "{v}"),
|
|
|
|
Value::Geometry(v) => write!(f, "{v}"),
|
|
|
|
Value::Param(v) => write!(f, "{v}"),
|
|
|
|
Value::Idiom(v) => write!(f, "{v}"),
|
|
|
|
Value::Table(v) => write!(f, "{v}"),
|
|
|
|
Value::Thing(v) => write!(f, "{v}"),
|
|
|
|
Value::Model(v) => write!(f, "{v}"),
|
|
|
|
Value::Regex(v) => write!(f, "{v}"),
|
2023-02-21 14:15:02 +00:00
|
|
|
Value::Block(v) => write!(f, "{v}"),
|
2023-02-03 11:47:07 +00:00
|
|
|
Value::Range(v) => write!(f, "{v}"),
|
|
|
|
Value::Edges(v) => write!(f, "{v}"),
|
|
|
|
Value::Future(v) => write!(f, "{v}"),
|
|
|
|
Value::Constant(v) => write!(f, "{v}"),
|
|
|
|
Value::Function(v) => write!(f, "{v}"),
|
|
|
|
Value::Subquery(v) => write!(f, "{v}"),
|
|
|
|
Value::Expression(v) => write!(f, "{v}"),
|
2023-03-31 18:44:44 +00:00
|
|
|
Value::Bytes(_) => write!(f, "<bytes>"),
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-14 08:12:56 +00:00
|
|
|
impl Value {
|
2022-05-14 21:30:49 +00:00
|
|
|
pub(crate) fn writeable(&self) -> bool {
|
|
|
|
match self {
|
2023-02-21 14:15:02 +00:00
|
|
|
Value::Block(v) => v.writeable(),
|
|
|
|
Value::Array(v) => v.iter().any(Value::writeable),
|
2022-05-14 21:30:49 +00:00
|
|
|
Value::Object(v) => v.iter().any(|(_, v)| v.writeable()),
|
2023-02-21 14:15:02 +00:00
|
|
|
Value::Function(v) => v.args().iter().any(Value::writeable),
|
2022-05-14 21:30:49 +00:00
|
|
|
Value::Subquery(v) => v.writeable(),
|
|
|
|
Value::Expression(v) => v.l.writeable() || v.r.writeable(),
|
|
|
|
_ => false,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-01-07 09:42:45 +00:00
|
|
|
#[cfg_attr(not(target_arch = "wasm32"), async_recursion)]
|
|
|
|
#[cfg_attr(target_arch = "wasm32", async_recursion(?Send))]
|
2022-05-01 22:25:53 +00:00
|
|
|
pub(crate) async fn compute(
|
2022-01-13 17:36:41 +00:00
|
|
|
&self,
|
2022-05-14 12:35:08 +00:00
|
|
|
ctx: &Context<'_>,
|
2022-02-06 21:06:52 +00:00
|
|
|
opt: &Options,
|
2022-02-15 03:33:16 +00:00
|
|
|
txn: &Transaction,
|
2022-01-14 08:12:56 +00:00
|
|
|
doc: Option<&'async_recursion Value>,
|
2022-01-13 17:36:41 +00:00
|
|
|
) -> Result<Value, Error> {
|
|
|
|
match self {
|
|
|
|
Value::None => Ok(Value::None),
|
|
|
|
Value::Null => Ok(Value::Null),
|
|
|
|
Value::True => Ok(Value::True),
|
|
|
|
Value::False => Ok(Value::False),
|
2022-09-04 09:52:01 +00:00
|
|
|
Value::Thing(v) => v.compute(ctx, opt, txn, doc).await,
|
2023-02-21 14:15:02 +00:00
|
|
|
Value::Block(v) => v.compute(ctx, opt, txn, doc).await,
|
2022-11-03 10:55:32 +00:00
|
|
|
Value::Range(v) => v.compute(ctx, opt, txn, doc).await,
|
2022-02-15 01:00:30 +00:00
|
|
|
Value::Param(v) => v.compute(ctx, opt, txn, doc).await,
|
|
|
|
Value::Idiom(v) => v.compute(ctx, opt, txn, doc).await,
|
|
|
|
Value::Array(v) => v.compute(ctx, opt, txn, doc).await,
|
|
|
|
Value::Object(v) => v.compute(ctx, opt, txn, doc).await,
|
2022-10-31 23:12:41 +00:00
|
|
|
Value::Future(v) => v.compute(ctx, opt, txn, doc).await,
|
2022-10-01 14:55:48 +00:00
|
|
|
Value::Constant(v) => v.compute(ctx, opt, txn, doc).await,
|
2022-02-15 01:00:30 +00:00
|
|
|
Value::Function(v) => v.compute(ctx, opt, txn, doc).await,
|
|
|
|
Value::Subquery(v) => v.compute(ctx, opt, txn, doc).await,
|
|
|
|
Value::Expression(v) => v.compute(ctx, opt, txn, doc).await,
|
2022-01-13 17:36:41 +00:00
|
|
|
_ => Ok(self.to_owned()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Serialize for Value {
|
|
|
|
fn serialize<S>(&self, s: S) -> Result<S::Ok, S::Error>
|
|
|
|
where
|
|
|
|
S: serde::Serializer,
|
|
|
|
{
|
2022-05-21 00:35:59 +00:00
|
|
|
if is_internal_serialization() {
|
2022-01-13 17:36:41 +00:00
|
|
|
match self {
|
2023-03-30 10:41:44 +00:00
|
|
|
Value::None => s.serialize_unit_variant(TOKEN, 0, "None"),
|
|
|
|
Value::Null => s.serialize_unit_variant(TOKEN, 1, "Null"),
|
|
|
|
Value::False => s.serialize_unit_variant(TOKEN, 2, "False"),
|
|
|
|
Value::True => s.serialize_unit_variant(TOKEN, 3, "True"),
|
|
|
|
Value::Number(v) => s.serialize_newtype_variant(TOKEN, 4, "Number", v),
|
|
|
|
Value::Strand(v) => s.serialize_newtype_variant(TOKEN, 5, "Strand", v),
|
|
|
|
Value::Duration(v) => s.serialize_newtype_variant(TOKEN, 6, "Duration", v),
|
|
|
|
Value::Datetime(v) => s.serialize_newtype_variant(TOKEN, 7, "Datetime", v),
|
|
|
|
Value::Uuid(v) => s.serialize_newtype_variant(TOKEN, 8, "Uuid", v),
|
|
|
|
Value::Array(v) => s.serialize_newtype_variant(TOKEN, 9, "Array", v),
|
|
|
|
Value::Object(v) => s.serialize_newtype_variant(TOKEN, 10, "Object", v),
|
|
|
|
Value::Geometry(v) => s.serialize_newtype_variant(TOKEN, 11, "Geometry", v),
|
2023-03-31 18:44:44 +00:00
|
|
|
Value::Bytes(v) => s.serialize_newtype_variant(TOKEN, 12, "Bytes", v),
|
|
|
|
Value::Param(v) => s.serialize_newtype_variant(TOKEN, 13, "Param", v),
|
|
|
|
Value::Idiom(v) => s.serialize_newtype_variant(TOKEN, 14, "Idiom", v),
|
|
|
|
Value::Table(v) => s.serialize_newtype_variant(TOKEN, 15, "Table", v),
|
|
|
|
Value::Thing(v) => s.serialize_newtype_variant(TOKEN, 16, "Thing", v),
|
|
|
|
Value::Model(v) => s.serialize_newtype_variant(TOKEN, 17, "Model", v),
|
|
|
|
Value::Regex(v) => s.serialize_newtype_variant(TOKEN, 18, "Regex", v),
|
|
|
|
Value::Block(v) => s.serialize_newtype_variant(TOKEN, 19, "Block", v),
|
|
|
|
Value::Range(v) => s.serialize_newtype_variant(TOKEN, 20, "Range", v),
|
|
|
|
Value::Edges(v) => s.serialize_newtype_variant(TOKEN, 21, "Edges", v),
|
|
|
|
Value::Future(v) => s.serialize_newtype_variant(TOKEN, 22, "Future", v),
|
|
|
|
Value::Constant(v) => s.serialize_newtype_variant(TOKEN, 23, "Constant", v),
|
|
|
|
Value::Function(v) => s.serialize_newtype_variant(TOKEN, 24, "Function", v),
|
|
|
|
Value::Subquery(v) => s.serialize_newtype_variant(TOKEN, 25, "Subquery", v),
|
|
|
|
Value::Expression(v) => s.serialize_newtype_variant(TOKEN, 26, "Expression", v),
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
2022-05-21 00:35:59 +00:00
|
|
|
} else {
|
|
|
|
match self {
|
|
|
|
Value::None => s.serialize_none(),
|
|
|
|
Value::Null => s.serialize_none(),
|
|
|
|
Value::True => s.serialize_bool(true),
|
|
|
|
Value::False => s.serialize_bool(false),
|
|
|
|
Value::Thing(v) => s.serialize_some(v),
|
2022-06-25 22:37:45 +00:00
|
|
|
Value::Uuid(v) => s.serialize_some(v),
|
2022-05-21 00:35:59 +00:00
|
|
|
Value::Array(v) => s.serialize_some(v),
|
|
|
|
Value::Object(v) => s.serialize_some(v),
|
|
|
|
Value::Number(v) => s.serialize_some(v),
|
|
|
|
Value::Strand(v) => s.serialize_some(v),
|
|
|
|
Value::Geometry(v) => s.serialize_some(v),
|
|
|
|
Value::Duration(v) => s.serialize_some(v),
|
|
|
|
Value::Datetime(v) => s.serialize_some(v),
|
2022-10-01 14:55:48 +00:00
|
|
|
Value::Constant(v) => s.serialize_some(v),
|
2022-05-21 00:35:59 +00:00
|
|
|
_ => s.serialize_none(),
|
|
|
|
}
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ops::Add for Value {
|
|
|
|
type Output = Self;
|
|
|
|
fn add(self, other: Self) -> Self {
|
|
|
|
match (self, other) {
|
|
|
|
(Value::Number(v), Value::Number(w)) => Value::Number(v + w),
|
|
|
|
(Value::Strand(v), Value::Strand(w)) => Value::Strand(v + w),
|
|
|
|
(Value::Datetime(v), Value::Duration(w)) => Value::Datetime(w + v),
|
|
|
|
(Value::Duration(v), Value::Datetime(w)) => Value::Datetime(v + w),
|
|
|
|
(Value::Duration(v), Value::Duration(w)) => Value::Duration(v + w),
|
|
|
|
(v, w) => Value::from(v.as_number() + w.as_number()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ops::Sub for Value {
|
|
|
|
type Output = Self;
|
|
|
|
fn sub(self, other: Self) -> Self {
|
|
|
|
match (self, other) {
|
|
|
|
(Value::Number(v), Value::Number(w)) => Value::Number(v - w),
|
2022-09-22 23:48:49 +00:00
|
|
|
(Value::Datetime(v), Value::Datetime(w)) => Value::Duration(v - w),
|
2022-01-13 17:36:41 +00:00
|
|
|
(Value::Datetime(v), Value::Duration(w)) => Value::Datetime(w - v),
|
|
|
|
(Value::Duration(v), Value::Datetime(w)) => Value::Datetime(v - w),
|
|
|
|
(Value::Duration(v), Value::Duration(w)) => Value::Duration(v - w),
|
|
|
|
(v, w) => Value::from(v.as_number() - w.as_number()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ops::Mul for Value {
|
|
|
|
type Output = Self;
|
|
|
|
fn mul(self, other: Self) -> Self {
|
|
|
|
match (self, other) {
|
|
|
|
(Value::Number(v), Value::Number(w)) => Value::Number(v * w),
|
|
|
|
(v, w) => Value::from(v.as_number() * w.as_number()),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl ops::Div for Value {
|
|
|
|
type Output = Self;
|
|
|
|
fn div(self, other: Self) -> Self {
|
2022-09-24 10:17:47 +00:00
|
|
|
match (self.as_number(), other.as_number()) {
|
|
|
|
(_, w) if w == Number::Int(0) => Value::None,
|
|
|
|
(v, w) => Value::Number(v / w),
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-03-31 15:42:29 +00:00
|
|
|
/// Parse any `Value` including binary expressions
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn value(i: &str) -> IResult<&str, Value> {
|
2023-03-31 15:42:29 +00:00
|
|
|
alt((map(expression, Value::from), single))(i)
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
|
2023-03-31 15:42:29 +00:00
|
|
|
/// Parse any `Value` excluding binary expressions
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn single(i: &str) -> IResult<&str, Value> {
|
|
|
|
alt((
|
2022-07-18 21:10:00 +00:00
|
|
|
alt((
|
|
|
|
map(tag_no_case("NONE"), |_| Value::None),
|
|
|
|
map(tag_no_case("NULL"), |_| Value::Null),
|
|
|
|
map(tag_no_case("true"), |_| Value::True),
|
|
|
|
map(tag_no_case("false"), |_| Value::False),
|
|
|
|
)),
|
|
|
|
alt((
|
2023-03-31 15:42:29 +00:00
|
|
|
map(idiom::multi, Value::from),
|
2022-07-18 21:10:00 +00:00
|
|
|
map(function, Value::from),
|
2023-03-31 15:42:29 +00:00
|
|
|
map(subquery, Value::from),
|
2022-09-30 21:22:00 +00:00
|
|
|
map(constant, Value::from),
|
2022-07-18 21:10:00 +00:00
|
|
|
map(datetime, Value::from),
|
|
|
|
map(duration, Value::from),
|
|
|
|
map(geometry, Value::from),
|
2022-10-31 23:12:41 +00:00
|
|
|
map(future, Value::from),
|
2022-07-18 21:10:00 +00:00
|
|
|
map(unique, Value::from),
|
|
|
|
map(number, Value::from),
|
|
|
|
map(object, Value::from),
|
|
|
|
map(array, Value::from),
|
2023-02-21 14:15:02 +00:00
|
|
|
map(block, Value::from),
|
2022-07-18 21:10:00 +00:00
|
|
|
map(param, Value::from),
|
|
|
|
map(regex, Value::from),
|
|
|
|
map(model, Value::from),
|
2023-03-31 15:42:29 +00:00
|
|
|
map(edges, Value::from),
|
2022-10-25 13:01:13 +00:00
|
|
|
map(range, Value::from),
|
2022-07-18 21:10:00 +00:00
|
|
|
map(thing, Value::from),
|
|
|
|
map(strand, Value::from),
|
2023-03-31 15:42:29 +00:00
|
|
|
map(idiom::path, Value::from),
|
2022-07-18 21:10:00 +00:00
|
|
|
)),
|
2022-01-13 17:36:41 +00:00
|
|
|
))(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn select(i: &str) -> IResult<&str, Value> {
|
|
|
|
alt((
|
2022-07-18 21:10:00 +00:00
|
|
|
alt((
|
2023-03-31 15:42:29 +00:00
|
|
|
map(expression, Value::from),
|
2022-07-18 21:10:00 +00:00
|
|
|
map(tag_no_case("NONE"), |_| Value::None),
|
|
|
|
map(tag_no_case("NULL"), |_| Value::Null),
|
|
|
|
map(tag_no_case("true"), |_| Value::True),
|
|
|
|
map(tag_no_case("false"), |_| Value::False),
|
|
|
|
)),
|
|
|
|
alt((
|
2023-03-31 15:42:29 +00:00
|
|
|
map(idiom::multi, Value::from),
|
2022-07-18 21:10:00 +00:00
|
|
|
map(function, Value::from),
|
2023-03-31 15:42:29 +00:00
|
|
|
map(subquery, Value::from),
|
2022-09-30 21:22:00 +00:00
|
|
|
map(constant, Value::from),
|
2022-07-18 21:10:00 +00:00
|
|
|
map(datetime, Value::from),
|
|
|
|
map(duration, Value::from),
|
|
|
|
map(geometry, Value::from),
|
2022-10-31 23:12:41 +00:00
|
|
|
map(future, Value::from),
|
2022-07-18 21:10:00 +00:00
|
|
|
map(unique, Value::from),
|
|
|
|
map(number, Value::from),
|
2023-03-31 15:42:29 +00:00
|
|
|
map(strand, Value::from),
|
2022-07-18 21:10:00 +00:00
|
|
|
map(object, Value::from),
|
|
|
|
map(array, Value::from),
|
2023-02-21 14:15:02 +00:00
|
|
|
map(block, Value::from),
|
2022-07-18 21:10:00 +00:00
|
|
|
map(param, Value::from),
|
|
|
|
map(regex, Value::from),
|
|
|
|
map(model, Value::from),
|
|
|
|
map(edges, Value::from),
|
2022-08-28 22:19:59 +00:00
|
|
|
map(range, Value::from),
|
2022-07-18 21:10:00 +00:00
|
|
|
map(thing, Value::from),
|
|
|
|
map(table, Value::from),
|
|
|
|
)),
|
2022-01-13 17:36:41 +00:00
|
|
|
))(i)
|
|
|
|
}
|
|
|
|
|
2023-03-31 15:42:29 +00:00
|
|
|
/// Used as the starting part of a complex Idiom
|
|
|
|
pub fn start(i: &str) -> IResult<&str, Value> {
|
2022-01-13 17:36:41 +00:00
|
|
|
alt((
|
2023-03-31 15:42:29 +00:00
|
|
|
map(function::normal, Value::from),
|
|
|
|
map(function::custom, Value::from),
|
2022-06-09 08:18:08 +00:00
|
|
|
map(subquery, Value::from),
|
2023-03-31 15:42:29 +00:00
|
|
|
map(constant, Value::from),
|
|
|
|
map(datetime, Value::from),
|
|
|
|
map(duration, Value::from),
|
|
|
|
map(unique, Value::from),
|
|
|
|
map(number, Value::from),
|
|
|
|
map(strand, Value::from),
|
|
|
|
map(object, Value::from),
|
|
|
|
map(array, Value::from),
|
|
|
|
map(param, Value::from),
|
|
|
|
map(edges, Value::from),
|
|
|
|
map(thing, Value::from),
|
|
|
|
))(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Used in CREATE, UPDATE, and DELETE clauses
|
|
|
|
pub fn what(i: &str) -> IResult<&str, Value> {
|
|
|
|
alt((
|
2022-06-09 08:18:08 +00:00
|
|
|
map(function, Value::from),
|
2023-03-31 15:42:29 +00:00
|
|
|
map(subquery, Value::from),
|
2022-09-30 21:22:00 +00:00
|
|
|
map(constant, Value::from),
|
2023-03-31 15:42:29 +00:00
|
|
|
map(datetime, Value::from),
|
|
|
|
map(duration, Value::from),
|
2022-10-31 23:12:41 +00:00
|
|
|
map(future, Value::from),
|
2023-02-21 14:15:02 +00:00
|
|
|
map(block, Value::from),
|
2022-06-09 08:18:08 +00:00
|
|
|
map(param, Value::from),
|
|
|
|
map(model, Value::from),
|
|
|
|
map(edges, Value::from),
|
2022-08-28 22:19:59 +00:00
|
|
|
map(range, Value::from),
|
2022-06-09 08:18:08 +00:00
|
|
|
map(thing, Value::from),
|
|
|
|
map(table, Value::from),
|
2022-01-13 17:36:41 +00:00
|
|
|
))(i)
|
|
|
|
}
|
|
|
|
|
2023-03-31 15:42:29 +00:00
|
|
|
/// Used to parse any simple JSON-like value
|
2022-01-13 17:36:41 +00:00
|
|
|
pub fn json(i: &str) -> IResult<&str, Value> {
|
|
|
|
alt((
|
|
|
|
map(tag_no_case("NULL"), |_| Value::Null),
|
|
|
|
map(tag_no_case("true"), |_| Value::True),
|
|
|
|
map(tag_no_case("false"), |_| Value::False),
|
2022-06-09 08:18:08 +00:00
|
|
|
map(datetime, Value::from),
|
|
|
|
map(duration, Value::from),
|
|
|
|
map(geometry, Value::from),
|
2022-06-25 22:37:45 +00:00
|
|
|
map(unique, Value::from),
|
2022-06-09 08:18:08 +00:00
|
|
|
map(number, Value::from),
|
|
|
|
map(object, Value::from),
|
|
|
|
map(array, Value::from),
|
2022-07-08 09:19:01 +00:00
|
|
|
map(thing, Value::from),
|
2022-06-09 08:18:08 +00:00
|
|
|
map(strand, Value::from),
|
2022-01-13 17:36:41 +00:00
|
|
|
))(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
|
|
|
|
use super::*;
|
2022-02-22 14:16:50 +00:00
|
|
|
use crate::sql::test::Parse;
|
2022-10-05 18:22:32 +00:00
|
|
|
use crate::sql::uuid::Uuid;
|
2022-01-13 17:36:41 +00:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn check_none() {
|
|
|
|
assert_eq!(true, Value::None.is_none());
|
2023-01-06 10:26:25 +00:00
|
|
|
assert_eq!(false, Value::Null.is_none());
|
2022-01-13 17:36:41 +00:00
|
|
|
assert_eq!(false, Value::from(1).is_none());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn check_null() {
|
|
|
|
assert_eq!(true, Value::Null.is_null());
|
2023-01-06 10:26:25 +00:00
|
|
|
assert_eq!(false, Value::None.is_null());
|
2022-01-13 17:36:41 +00:00
|
|
|
assert_eq!(false, Value::from(1).is_null());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn check_true() {
|
|
|
|
assert_eq!(false, Value::None.is_true());
|
|
|
|
assert_eq!(true, Value::True.is_true());
|
|
|
|
assert_eq!(false, Value::False.is_true());
|
|
|
|
assert_eq!(false, Value::from(1).is_true());
|
|
|
|
assert_eq!(true, Value::from("true").is_true());
|
|
|
|
assert_eq!(false, Value::from("false").is_true());
|
|
|
|
assert_eq!(false, Value::from("something").is_true());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn check_false() {
|
|
|
|
assert_eq!(false, Value::None.is_false());
|
|
|
|
assert_eq!(false, Value::True.is_false());
|
|
|
|
assert_eq!(true, Value::False.is_false());
|
|
|
|
assert_eq!(false, Value::from(1).is_false());
|
|
|
|
assert_eq!(false, Value::from("true").is_false());
|
|
|
|
assert_eq!(true, Value::from("false").is_false());
|
|
|
|
assert_eq!(false, Value::from("something").is_false());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn convert_bool() {
|
|
|
|
assert_eq!(false, Value::None.is_truthy());
|
|
|
|
assert_eq!(false, Value::Null.is_truthy());
|
|
|
|
assert_eq!(true, Value::True.is_truthy());
|
|
|
|
assert_eq!(false, Value::False.is_truthy());
|
|
|
|
assert_eq!(false, Value::from(0).is_truthy());
|
|
|
|
assert_eq!(true, Value::from(1).is_truthy());
|
|
|
|
assert_eq!(true, Value::from(-1).is_truthy());
|
|
|
|
assert_eq!(true, Value::from(1.1).is_truthy());
|
|
|
|
assert_eq!(true, Value::from(-1.1).is_truthy());
|
|
|
|
assert_eq!(true, Value::from("true").is_truthy());
|
|
|
|
assert_eq!(false, Value::from("false").is_truthy());
|
|
|
|
assert_eq!(true, Value::from("falsey").is_truthy());
|
|
|
|
assert_eq!(true, Value::from("something").is_truthy());
|
2022-10-05 18:22:32 +00:00
|
|
|
assert_eq!(true, Value::from(Uuid::new()).is_truthy());
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn convert_int() {
|
|
|
|
assert_eq!(0, Value::None.as_int());
|
|
|
|
assert_eq!(0, Value::Null.as_int());
|
|
|
|
assert_eq!(1, Value::True.as_int());
|
|
|
|
assert_eq!(0, Value::False.as_int());
|
|
|
|
assert_eq!(0, Value::from(0).as_int());
|
|
|
|
assert_eq!(1, Value::from(1).as_int());
|
|
|
|
assert_eq!(-1, Value::from(-1).as_int());
|
|
|
|
assert_eq!(1, Value::from(1.1).as_int());
|
|
|
|
assert_eq!(-1, Value::from(-1.1).as_int());
|
|
|
|
assert_eq!(3, Value::from("3").as_int());
|
|
|
|
assert_eq!(0, Value::from("true").as_int());
|
|
|
|
assert_eq!(0, Value::from("false").as_int());
|
|
|
|
assert_eq!(0, Value::from("something").as_int());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn convert_float() {
|
|
|
|
assert_eq!(0.0, Value::None.as_float());
|
|
|
|
assert_eq!(0.0, Value::Null.as_float());
|
|
|
|
assert_eq!(1.0, Value::True.as_float());
|
|
|
|
assert_eq!(0.0, Value::False.as_float());
|
|
|
|
assert_eq!(0.0, Value::from(0).as_float());
|
|
|
|
assert_eq!(1.0, Value::from(1).as_float());
|
|
|
|
assert_eq!(-1.0, Value::from(-1).as_float());
|
|
|
|
assert_eq!(1.1, Value::from(1.1).as_float());
|
|
|
|
assert_eq!(-1.1, Value::from(-1.1).as_float());
|
|
|
|
assert_eq!(3.0, Value::from("3").as_float());
|
|
|
|
assert_eq!(0.0, Value::from("true").as_float());
|
|
|
|
assert_eq!(0.0, Value::from("false").as_float());
|
|
|
|
assert_eq!(0.0, Value::from("something").as_float());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn convert_number() {
|
|
|
|
assert_eq!(Number::from(0), Value::None.as_number());
|
|
|
|
assert_eq!(Number::from(0), Value::Null.as_number());
|
|
|
|
assert_eq!(Number::from(1), Value::True.as_number());
|
|
|
|
assert_eq!(Number::from(0), Value::False.as_number());
|
|
|
|
assert_eq!(Number::from(0), Value::from(0).as_number());
|
|
|
|
assert_eq!(Number::from(1), Value::from(1).as_number());
|
|
|
|
assert_eq!(Number::from(-1), Value::from(-1).as_number());
|
|
|
|
assert_eq!(Number::from(1.1), Value::from(1.1).as_number());
|
|
|
|
assert_eq!(Number::from(-1.1), Value::from(-1.1).as_number());
|
|
|
|
assert_eq!(Number::from(3), Value::from("3").as_number());
|
|
|
|
assert_eq!(Number::from(0), Value::from("true").as_number());
|
|
|
|
assert_eq!(Number::from(0), Value::from("false").as_number());
|
|
|
|
assert_eq!(Number::from(0), Value::from("something").as_number());
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn convert_strand() {
|
|
|
|
assert_eq!(Strand::from("NONE"), Value::None.as_strand());
|
|
|
|
assert_eq!(Strand::from("NULL"), Value::Null.as_strand());
|
|
|
|
assert_eq!(Strand::from("true"), Value::True.as_strand());
|
|
|
|
assert_eq!(Strand::from("false"), Value::False.as_strand());
|
|
|
|
assert_eq!(Strand::from("0"), Value::from(0).as_strand());
|
|
|
|
assert_eq!(Strand::from("1"), Value::from(1).as_strand());
|
|
|
|
assert_eq!(Strand::from("-1"), Value::from(-1).as_strand());
|
|
|
|
assert_eq!(Strand::from("1.1"), Value::from(1.1).as_strand());
|
|
|
|
assert_eq!(Strand::from("-1.1"), Value::from(-1.1).as_strand());
|
|
|
|
assert_eq!(Strand::from("3"), Value::from("3").as_strand());
|
|
|
|
assert_eq!(Strand::from("true"), Value::from("true").as_strand());
|
|
|
|
assert_eq!(Strand::from("false"), Value::from("false").as_strand());
|
|
|
|
assert_eq!(Strand::from("something"), Value::from("something").as_strand());
|
|
|
|
}
|
2022-02-22 14:16:50 +00:00
|
|
|
|
2022-05-05 04:30:32 +00:00
|
|
|
#[test]
|
|
|
|
fn convert_string() {
|
|
|
|
assert_eq!(String::from("NONE"), Value::None.as_string());
|
|
|
|
assert_eq!(String::from("NULL"), Value::Null.as_string());
|
|
|
|
assert_eq!(String::from("true"), Value::True.as_string());
|
|
|
|
assert_eq!(String::from("false"), Value::False.as_string());
|
|
|
|
assert_eq!(String::from("0"), Value::from(0).as_string());
|
|
|
|
assert_eq!(String::from("1"), Value::from(1).as_string());
|
|
|
|
assert_eq!(String::from("-1"), Value::from(-1).as_string());
|
|
|
|
assert_eq!(String::from("1.1"), Value::from(1.1).as_string());
|
|
|
|
assert_eq!(String::from("-1.1"), Value::from(-1.1).as_string());
|
|
|
|
assert_eq!(String::from("3"), Value::from("3").as_string());
|
|
|
|
assert_eq!(String::from("true"), Value::from("true").as_string());
|
|
|
|
assert_eq!(String::from("false"), Value::from("false").as_string());
|
|
|
|
assert_eq!(String::from("something"), Value::from("something").as_string());
|
|
|
|
}
|
|
|
|
|
2022-05-06 22:33:19 +00:00
|
|
|
#[test]
|
|
|
|
fn check_size() {
|
2022-06-08 07:39:09 +00:00
|
|
|
assert_eq!(64, std::mem::size_of::<Value>());
|
2022-11-16 17:57:22 +00:00
|
|
|
assert_eq!(104, std::mem::size_of::<Result<Value, Error>>());
|
|
|
|
assert_eq!(40, std::mem::size_of::<crate::sql::number::Number>());
|
2022-05-06 22:33:19 +00:00
|
|
|
assert_eq!(24, std::mem::size_of::<crate::sql::strand::Strand>());
|
|
|
|
assert_eq!(16, std::mem::size_of::<crate::sql::duration::Duration>());
|
|
|
|
assert_eq!(12, std::mem::size_of::<crate::sql::datetime::Datetime>());
|
|
|
|
assert_eq!(24, std::mem::size_of::<crate::sql::array::Array>());
|
|
|
|
assert_eq!(24, std::mem::size_of::<crate::sql::object::Object>());
|
|
|
|
assert_eq!(56, std::mem::size_of::<crate::sql::geometry::Geometry>());
|
|
|
|
assert_eq!(24, std::mem::size_of::<crate::sql::param::Param>());
|
|
|
|
assert_eq!(24, std::mem::size_of::<crate::sql::idiom::Idiom>());
|
|
|
|
assert_eq!(24, std::mem::size_of::<crate::sql::table::Table>());
|
2022-06-08 07:39:09 +00:00
|
|
|
assert_eq!(56, std::mem::size_of::<crate::sql::thing::Thing>());
|
2023-01-29 12:32:17 +00:00
|
|
|
assert_eq!(48, std::mem::size_of::<crate::sql::model::Model>());
|
2022-05-09 06:18:30 +00:00
|
|
|
assert_eq!(24, std::mem::size_of::<crate::sql::regex::Regex>());
|
2022-08-28 22:19:59 +00:00
|
|
|
assert_eq!(8, std::mem::size_of::<Box<crate::sql::range::Range>>());
|
2022-06-09 08:18:08 +00:00
|
|
|
assert_eq!(8, std::mem::size_of::<Box<crate::sql::edges::Edges>>());
|
2022-05-06 22:33:19 +00:00
|
|
|
assert_eq!(8, std::mem::size_of::<Box<crate::sql::function::Function>>());
|
|
|
|
assert_eq!(8, std::mem::size_of::<Box<crate::sql::subquery::Subquery>>());
|
|
|
|
assert_eq!(8, std::mem::size_of::<Box<crate::sql::expression::Expression>>());
|
|
|
|
}
|
|
|
|
|
2022-02-22 14:16:50 +00:00
|
|
|
#[test]
|
|
|
|
fn serialize_deserialize() {
|
|
|
|
let val = Value::parse(
|
|
|
|
"{ test: { something: [1, 'two', null, test:tobie, { something: false }] } }",
|
|
|
|
);
|
|
|
|
let res = Value::parse(
|
|
|
|
"{ test: { something: [1, 'two', null, test:tobie, { something: false }] } }",
|
|
|
|
);
|
|
|
|
let enc: Vec<u8> = val.into();
|
|
|
|
let dec: Value = enc.into();
|
|
|
|
assert_eq!(res, dec);
|
|
|
|
}
|
2022-01-13 17:36:41 +00:00
|
|
|
}
|