2022-12-30 08:23:19 +00:00
|
|
|
//! The different options and types for use in API functions
|
|
|
|
|
|
|
|
pub mod auth;
|
|
|
|
|
2023-07-26 11:54:48 +00:00
|
|
|
mod config;
|
2022-12-30 08:23:19 +00:00
|
|
|
mod endpoint;
|
|
|
|
mod query;
|
|
|
|
mod resource;
|
|
|
|
mod strict;
|
|
|
|
mod tls;
|
|
|
|
|
|
|
|
use crate::api::err::Error;
|
2023-05-20 07:28:52 +00:00
|
|
|
use crate::sql::constant::ConstantValue;
|
2023-08-20 03:27:20 +00:00
|
|
|
use crate::sql::id::Gen;
|
2023-03-30 10:41:44 +00:00
|
|
|
use crate::sql::to_value;
|
2022-12-30 08:23:19 +00:00
|
|
|
use crate::sql::Thing;
|
|
|
|
use crate::sql::Value;
|
|
|
|
use dmp::Diff;
|
|
|
|
use serde::de::DeserializeOwned;
|
|
|
|
use serde::Serialize;
|
2023-04-29 08:06:02 +00:00
|
|
|
use serde_json::json;
|
2023-03-30 10:41:44 +00:00
|
|
|
use serde_json::Map;
|
2022-12-30 08:23:19 +00:00
|
|
|
use serde_json::Value as JsonValue;
|
|
|
|
|
2023-07-26 11:54:48 +00:00
|
|
|
pub use config::*;
|
2022-12-30 08:23:19 +00:00
|
|
|
pub use endpoint::*;
|
|
|
|
pub use query::*;
|
|
|
|
pub use resource::*;
|
|
|
|
pub use strict::*;
|
|
|
|
pub use tls::*;
|
|
|
|
|
|
|
|
/// Record ID
|
|
|
|
pub type RecordId = Thing;
|
|
|
|
|
|
|
|
type UnitOp<'a> = InnerOp<'a, ()>;
|
|
|
|
|
|
|
|
#[derive(Debug, Serialize)]
|
|
|
|
#[serde(tag = "op", rename_all = "lowercase")]
|
|
|
|
enum InnerOp<'a, T> {
|
|
|
|
Add {
|
|
|
|
path: &'a str,
|
|
|
|
value: T,
|
|
|
|
},
|
|
|
|
Remove {
|
|
|
|
path: &'a str,
|
|
|
|
},
|
|
|
|
Replace {
|
|
|
|
path: &'a str,
|
|
|
|
value: T,
|
|
|
|
},
|
|
|
|
Change {
|
|
|
|
path: &'a str,
|
|
|
|
value: String,
|
|
|
|
},
|
|
|
|
}
|
|
|
|
|
|
|
|
/// A [JSON Patch] operation
|
|
|
|
///
|
|
|
|
/// From the official website:
|
|
|
|
///
|
|
|
|
/// > JSON Patch is a format for describing changes to a JSON document.
|
|
|
|
/// > It can be used to avoid sending a whole document when only a part has changed.
|
|
|
|
///
|
|
|
|
/// [JSON Patch]: https://jsonpatch.com/
|
|
|
|
#[derive(Debug)]
|
2023-04-19 08:26:22 +00:00
|
|
|
#[must_use]
|
2023-03-30 10:41:44 +00:00
|
|
|
pub struct PatchOp(pub(crate) Result<Value, crate::err::Error>);
|
2022-12-30 08:23:19 +00:00
|
|
|
|
|
|
|
impl PatchOp {
|
|
|
|
/// Adds a value to an object or inserts it into an array.
|
|
|
|
///
|
|
|
|
/// In the case of an array, the value is inserted before the given index.
|
|
|
|
/// The `-` character can be used instead of an index to insert at the end of an array.
|
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// # use serde_json::json;
|
|
|
|
/// # use surrealdb::opt::PatchOp;
|
|
|
|
/// PatchOp::add("/biscuits/1", json!({ "name": "Ginger Nut" }))
|
|
|
|
/// # ;
|
|
|
|
/// ```
|
|
|
|
pub fn add<T>(path: &str, value: T) -> Self
|
|
|
|
where
|
|
|
|
T: Serialize,
|
|
|
|
{
|
2023-03-30 10:41:44 +00:00
|
|
|
Self(to_value(InnerOp::Add {
|
2022-12-30 08:23:19 +00:00
|
|
|
path,
|
2023-03-30 10:41:44 +00:00
|
|
|
value,
|
|
|
|
}))
|
2022-12-30 08:23:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Removes a value from an object or array.
|
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// # use surrealdb::opt::PatchOp;
|
|
|
|
/// PatchOp::remove("/biscuits")
|
|
|
|
/// # ;
|
|
|
|
/// ```
|
|
|
|
///
|
|
|
|
/// Remove the first element of the array at `biscuits`
|
|
|
|
/// (or just removes the “0” key if `biscuits` is an object)
|
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// # use surrealdb::opt::PatchOp;
|
|
|
|
/// PatchOp::remove("/biscuits/0")
|
|
|
|
/// # ;
|
|
|
|
/// ```
|
|
|
|
pub fn remove(path: &str) -> Self {
|
2023-03-30 10:41:44 +00:00
|
|
|
Self(to_value(UnitOp::Remove {
|
|
|
|
path,
|
|
|
|
}))
|
2022-12-30 08:23:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Replaces a value.
|
|
|
|
///
|
|
|
|
/// Equivalent to a “remove” followed by an “add”.
|
|
|
|
///
|
|
|
|
/// # Examples
|
|
|
|
///
|
|
|
|
/// ```
|
|
|
|
/// # use surrealdb::opt::PatchOp;
|
|
|
|
/// PatchOp::replace("/biscuits/0/name", "Chocolate Digestive")
|
|
|
|
/// # ;
|
|
|
|
/// ```
|
|
|
|
pub fn replace<T>(path: &str, value: T) -> Self
|
|
|
|
where
|
|
|
|
T: Serialize,
|
|
|
|
{
|
2023-03-30 10:41:44 +00:00
|
|
|
Self(to_value(InnerOp::Replace {
|
2022-12-30 08:23:19 +00:00
|
|
|
path,
|
2023-03-30 10:41:44 +00:00
|
|
|
value,
|
|
|
|
}))
|
2022-12-30 08:23:19 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
/// Changes a value
|
|
|
|
pub fn change(path: &str, diff: Diff) -> Self {
|
2023-03-30 10:41:44 +00:00
|
|
|
Self(to_value(UnitOp::Change {
|
2022-12-30 08:23:19 +00:00
|
|
|
path,
|
|
|
|
value: diff.text,
|
2023-03-30 10:41:44 +00:00
|
|
|
}))
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-29 08:06:02 +00:00
|
|
|
impl From<Value> for serde_json::Value {
|
|
|
|
fn from(value: Value) -> Self {
|
|
|
|
into_json(value, true)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
fn into_json(value: Value, simplify: bool) -> JsonValue {
|
2023-03-30 10:41:44 +00:00
|
|
|
use crate::sql;
|
|
|
|
use crate::sql::Number;
|
|
|
|
|
|
|
|
#[derive(Serialize)]
|
|
|
|
struct Array(Vec<JsonValue>);
|
|
|
|
|
2023-04-29 08:06:02 +00:00
|
|
|
impl From<(sql::Array, bool)> for Array {
|
|
|
|
fn from((arr, simplify): (sql::Array, bool)) -> Self {
|
2023-03-30 10:41:44 +00:00
|
|
|
let mut vec = Vec::with_capacity(arr.0.len());
|
|
|
|
for value in arr.0 {
|
2023-04-29 08:06:02 +00:00
|
|
|
vec.push(into_json(value, simplify));
|
2023-03-30 10:41:44 +00:00
|
|
|
}
|
2023-04-29 08:06:02 +00:00
|
|
|
Self(vec)
|
2023-03-30 10:41:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Serialize)]
|
|
|
|
struct Object(Map<String, JsonValue>);
|
|
|
|
|
2023-04-29 08:06:02 +00:00
|
|
|
impl From<(sql::Object, bool)> for Object {
|
|
|
|
fn from((obj, simplify): (sql::Object, bool)) -> Self {
|
2023-03-30 10:41:44 +00:00
|
|
|
let mut map = Map::with_capacity(obj.0.len());
|
|
|
|
for (key, value) in obj.0 {
|
2023-04-29 08:06:02 +00:00
|
|
|
map.insert(key.to_owned(), into_json(value, simplify));
|
2023-03-30 10:41:44 +00:00
|
|
|
}
|
2023-04-29 08:06:02 +00:00
|
|
|
Self(map)
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Serialize)]
|
|
|
|
enum CoordinatesType {
|
|
|
|
Point,
|
|
|
|
LineString,
|
|
|
|
Polygon,
|
|
|
|
MultiPoint,
|
|
|
|
MultiLineString,
|
|
|
|
MultiPolygon,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Serialize)]
|
|
|
|
struct Coordinates {
|
|
|
|
#[serde(rename = "type")]
|
|
|
|
typ: CoordinatesType,
|
|
|
|
coordinates: JsonValue,
|
|
|
|
}
|
|
|
|
|
|
|
|
struct GeometryCollection;
|
|
|
|
|
2023-07-14 23:01:02 +00:00
|
|
|
impl Serialize for GeometryCollection {
|
|
|
|
fn serialize<S>(&self, s: S) -> Result<S::Ok, S::Error>
|
|
|
|
where
|
|
|
|
S: serde::Serializer,
|
|
|
|
{
|
|
|
|
s.serialize_str("GeometryCollection")
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-29 08:06:02 +00:00
|
|
|
#[derive(Serialize)]
|
|
|
|
struct Geometries {
|
|
|
|
#[serde(rename = "type")]
|
|
|
|
typ: GeometryCollection,
|
|
|
|
geometries: Vec<JsonValue>,
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Serialize)]
|
|
|
|
struct Geometry(JsonValue);
|
|
|
|
|
|
|
|
impl From<sql::Geometry> for Geometry {
|
|
|
|
fn from(geo: sql::Geometry) -> Self {
|
|
|
|
Self(match geo {
|
|
|
|
sql::Geometry::Point(v) => json!(Coordinates {
|
|
|
|
typ: CoordinatesType::Point,
|
|
|
|
coordinates: vec![json!(v.x()), json!(v.y())].into(),
|
|
|
|
}),
|
|
|
|
sql::Geometry::Line(v) => json!(Coordinates {
|
|
|
|
typ: CoordinatesType::LineString,
|
|
|
|
coordinates: v
|
|
|
|
.points()
|
|
|
|
.map(|p| vec![json!(p.x()), json!(p.y())].into())
|
|
|
|
.collect::<Vec<JsonValue>>()
|
|
|
|
.into(),
|
|
|
|
}),
|
|
|
|
sql::Geometry::Polygon(v) => json!(Coordinates {
|
|
|
|
typ: CoordinatesType::Polygon,
|
|
|
|
coordinates: vec![v
|
|
|
|
.exterior()
|
|
|
|
.points()
|
|
|
|
.map(|p| vec![json!(p.x()), json!(p.y())].into())
|
|
|
|
.collect::<Vec<JsonValue>>()]
|
|
|
|
.into_iter()
|
|
|
|
.chain(
|
|
|
|
v.interiors()
|
|
|
|
.iter()
|
|
|
|
.map(|i| {
|
|
|
|
i.points()
|
|
|
|
.map(|p| vec![json!(p.x()), json!(p.y())].into())
|
|
|
|
.collect::<Vec<JsonValue>>()
|
|
|
|
})
|
|
|
|
.collect::<Vec<Vec<JsonValue>>>(),
|
|
|
|
)
|
|
|
|
.collect::<Vec<Vec<JsonValue>>>()
|
|
|
|
.into(),
|
|
|
|
}),
|
|
|
|
sql::Geometry::MultiPoint(v) => json!(Coordinates {
|
|
|
|
typ: CoordinatesType::MultiPoint,
|
|
|
|
coordinates: v
|
|
|
|
.0
|
|
|
|
.iter()
|
|
|
|
.map(|v| vec![json!(v.x()), json!(v.y())].into())
|
|
|
|
.collect::<Vec<JsonValue>>()
|
|
|
|
.into()
|
|
|
|
}),
|
|
|
|
sql::Geometry::MultiLine(v) => json!(Coordinates {
|
|
|
|
typ: CoordinatesType::MultiLineString,
|
|
|
|
coordinates: v
|
|
|
|
.0
|
|
|
|
.iter()
|
|
|
|
.map(|v| {
|
|
|
|
v.points()
|
|
|
|
.map(|v| vec![json!(v.x()), json!(v.y())].into())
|
|
|
|
.collect::<Vec<JsonValue>>()
|
|
|
|
})
|
|
|
|
.collect::<Vec<Vec<JsonValue>>>()
|
|
|
|
.into()
|
|
|
|
}),
|
|
|
|
sql::Geometry::MultiPolygon(v) => json!(Coordinates {
|
|
|
|
typ: CoordinatesType::MultiPolygon,
|
|
|
|
coordinates: v
|
|
|
|
.0
|
|
|
|
.iter()
|
|
|
|
.map(|v| {
|
|
|
|
vec![v
|
|
|
|
.exterior()
|
|
|
|
.points()
|
|
|
|
.map(|p| vec![json!(p.x()), json!(p.y())].into())
|
|
|
|
.collect::<Vec<JsonValue>>()]
|
|
|
|
.into_iter()
|
|
|
|
.chain(
|
|
|
|
v.interiors()
|
|
|
|
.iter()
|
|
|
|
.map(|i| {
|
|
|
|
i.points()
|
|
|
|
.map(|p| vec![json!(p.x()), json!(p.y())].into())
|
|
|
|
.collect::<Vec<JsonValue>>()
|
|
|
|
})
|
|
|
|
.collect::<Vec<Vec<JsonValue>>>(),
|
|
|
|
)
|
|
|
|
.collect::<Vec<Vec<JsonValue>>>()
|
|
|
|
})
|
|
|
|
.collect::<Vec<Vec<Vec<JsonValue>>>>()
|
|
|
|
.into(),
|
|
|
|
}),
|
|
|
|
sql::Geometry::Collection(v) => json!(Geometries {
|
|
|
|
typ: GeometryCollection,
|
|
|
|
geometries: v.into_iter().map(Geometry::from).map(|x| x.0).collect(),
|
|
|
|
}),
|
|
|
|
})
|
2023-03-30 10:41:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Serialize)]
|
|
|
|
enum Id {
|
|
|
|
Number(i64),
|
|
|
|
String(String),
|
|
|
|
Array(Array),
|
|
|
|
Object(Object),
|
|
|
|
}
|
|
|
|
|
2023-04-29 08:06:02 +00:00
|
|
|
impl From<(sql::Id, bool)> for Id {
|
|
|
|
fn from((id, simplify): (sql::Id, bool)) -> Self {
|
|
|
|
match id {
|
|
|
|
sql::Id::Number(n) => Id::Number(n),
|
|
|
|
sql::Id::String(s) => Id::String(s),
|
|
|
|
sql::Id::Array(arr) => Id::Array((arr, simplify).into()),
|
|
|
|
sql::Id::Object(obj) => Id::Object((obj, simplify).into()),
|
2023-08-20 03:27:20 +00:00
|
|
|
sql::Id::Generate(v) => match v {
|
|
|
|
Gen::Rand => Id::from((sql::Id::rand(), simplify)),
|
|
|
|
Gen::Ulid => Id::from((sql::Id::ulid(), simplify)),
|
|
|
|
Gen::Uuid => Id::from((sql::Id::uuid(), simplify)),
|
|
|
|
},
|
2023-04-29 08:06:02 +00:00
|
|
|
}
|
2023-03-30 10:41:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[derive(Serialize)]
|
|
|
|
struct Thing {
|
|
|
|
tb: String,
|
|
|
|
id: Id,
|
|
|
|
}
|
|
|
|
|
2023-04-29 08:06:02 +00:00
|
|
|
impl From<(sql::Thing, bool)> for Thing {
|
|
|
|
fn from((thing, simplify): (sql::Thing, bool)) -> Self {
|
|
|
|
Self {
|
2023-03-30 10:41:44 +00:00
|
|
|
tb: thing.tb,
|
2023-04-29 08:06:02 +00:00
|
|
|
id: (thing.id, simplify).into(),
|
|
|
|
}
|
2023-03-30 10:41:44 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
match value {
|
2023-07-14 23:01:02 +00:00
|
|
|
// These value types are simple values which
|
|
|
|
// can be used in query responses sent to
|
|
|
|
// the client.
|
2023-04-29 08:06:02 +00:00
|
|
|
Value::None | Value::Null => JsonValue::Null,
|
2023-04-29 15:58:22 +00:00
|
|
|
Value::Bool(boolean) => boolean.into(),
|
2023-07-14 23:01:02 +00:00
|
|
|
Value::Number(number) => match number {
|
|
|
|
Number::Int(int) => int.into(),
|
|
|
|
Number::Float(float) => float.into(),
|
|
|
|
Number::Decimal(decimal) => json!(decimal),
|
2023-04-29 08:06:02 +00:00
|
|
|
},
|
2023-07-14 23:01:02 +00:00
|
|
|
Value::Strand(strand) => strand.0.into(),
|
|
|
|
Value::Duration(duration) => match simplify {
|
|
|
|
true => duration.to_raw().into(),
|
|
|
|
false => json!(duration.0),
|
2023-04-29 08:06:02 +00:00
|
|
|
},
|
2023-07-14 23:01:02 +00:00
|
|
|
Value::Datetime(datetime) => json!(datetime.0),
|
|
|
|
Value::Uuid(uuid) => json!(uuid.0),
|
|
|
|
Value::Array(array) => JsonValue::Array(Array::from((array, simplify)).0),
|
|
|
|
Value::Object(object) => JsonValue::Object(Object::from((object, simplify)).0),
|
2023-04-29 08:06:02 +00:00
|
|
|
Value::Geometry(geo) => match simplify {
|
|
|
|
true => Geometry::from(geo).0,
|
2023-07-14 23:01:02 +00:00
|
|
|
false => match geo {
|
|
|
|
sql::Geometry::Point(geo) => json!(geo),
|
|
|
|
sql::Geometry::Line(geo) => json!(geo),
|
|
|
|
sql::Geometry::Polygon(geo) => json!(geo),
|
|
|
|
sql::Geometry::MultiPoint(geo) => json!(geo),
|
|
|
|
sql::Geometry::MultiLine(geo) => json!(geo),
|
|
|
|
sql::Geometry::MultiPolygon(geo) => json!(geo),
|
|
|
|
sql::Geometry::Collection(geo) => json!(geo),
|
|
|
|
},
|
2023-04-29 08:06:02 +00:00
|
|
|
},
|
2023-07-14 23:01:02 +00:00
|
|
|
Value::Bytes(bytes) => json!(bytes.0),
|
2023-04-29 08:06:02 +00:00
|
|
|
Value::Thing(thing) => match simplify {
|
|
|
|
true => thing.to_string().into(),
|
|
|
|
false => json!(thing),
|
|
|
|
},
|
2023-07-14 23:01:02 +00:00
|
|
|
// These Value types are un-computed values
|
|
|
|
// and are not used in query responses sent
|
|
|
|
// to the client.
|
|
|
|
Value::Param(param) => json!(param),
|
|
|
|
Value::Idiom(idiom) => json!(idiom),
|
|
|
|
Value::Table(table) => json!(table),
|
2023-04-29 08:06:02 +00:00
|
|
|
Value::Model(model) => json!(model),
|
|
|
|
Value::Regex(regex) => json!(regex),
|
|
|
|
Value::Block(block) => json!(block),
|
|
|
|
Value::Range(range) => json!(range),
|
|
|
|
Value::Edges(edges) => json!(edges),
|
|
|
|
Value::Future(future) => json!(future),
|
|
|
|
Value::Constant(constant) => match simplify {
|
2023-05-20 07:28:52 +00:00
|
|
|
true => match constant.value() {
|
2023-07-14 23:01:02 +00:00
|
|
|
ConstantValue::Datetime(datetime) => json!(datetime.0),
|
|
|
|
ConstantValue::Float(float) => float.into(),
|
2023-05-20 07:28:52 +00:00
|
|
|
},
|
2023-04-29 08:06:02 +00:00
|
|
|
false => json!(constant),
|
|
|
|
},
|
2023-06-06 06:12:59 +00:00
|
|
|
Value::Cast(cast) => json!(cast),
|
2023-04-29 08:06:02 +00:00
|
|
|
Value::Function(function) => json!(function),
|
2023-08-18 22:51:56 +00:00
|
|
|
Value::Query(query) => json!(query),
|
2023-04-29 08:06:02 +00:00
|
|
|
Value::Subquery(subquery) => json!(subquery),
|
|
|
|
Value::Expression(expression) => json!(expression),
|
2022-12-30 08:23:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
/// Deserializes a value `T` from `SurrealDB` [`Value`]
|
2023-03-30 10:41:44 +00:00
|
|
|
pub(crate) fn from_value<T>(value: Value) -> Result<T, Error>
|
2022-12-30 08:23:19 +00:00
|
|
|
where
|
|
|
|
T: DeserializeOwned,
|
|
|
|
{
|
2023-04-29 15:58:22 +00:00
|
|
|
let json = into_json(value.clone(), false);
|
2023-03-30 10:41:44 +00:00
|
|
|
serde_json::from_value(json).map_err(|error| Error::FromValue {
|
|
|
|
value,
|
|
|
|
error: error.to_string(),
|
|
|
|
})
|
2022-12-30 08:23:19 +00:00
|
|
|
}
|
2023-07-14 23:01:02 +00:00
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
mod into_json {
|
|
|
|
use crate::opt::from_value;
|
|
|
|
use crate::opt::into_json;
|
|
|
|
use crate::sql;
|
|
|
|
use crate::sql::Value;
|
|
|
|
use chrono::DateTime;
|
|
|
|
use chrono::Utc;
|
|
|
|
use geo::line_string;
|
|
|
|
use geo::point;
|
|
|
|
use geo::polygon;
|
|
|
|
use geo::LineString;
|
|
|
|
use geo::MultiLineString;
|
|
|
|
use geo::MultiPoint;
|
|
|
|
use geo::MultiPolygon;
|
|
|
|
use geo::Point;
|
|
|
|
use geo::Polygon;
|
|
|
|
use rust_decimal::Decimal;
|
|
|
|
use serde_json::json;
|
|
|
|
use std::collections::BTreeMap;
|
|
|
|
use std::time::Duration;
|
|
|
|
use uuid::Uuid;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn none_or_null() {
|
|
|
|
for value in [Value::None, Value::Null] {
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!(null));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(null));
|
|
|
|
|
|
|
|
let response: Option<String> = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, None);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn bool() {
|
|
|
|
for boolean in [true, false] {
|
|
|
|
let value = Value::Bool(boolean);
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!(boolean));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(boolean));
|
|
|
|
|
|
|
|
let response: bool = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, boolean);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn number_int() {
|
|
|
|
for num in [i64::MIN, 0, i64::MAX] {
|
|
|
|
let value = Value::Number(sql::Number::Int(num));
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!(num));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(num));
|
|
|
|
|
|
|
|
let response: i64 = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, num);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn number_float() {
|
|
|
|
for num in [f64::NEG_INFINITY, f64::MIN, 0.0, f64::MAX, f64::INFINITY, f64::NAN] {
|
|
|
|
let value = Value::Number(sql::Number::Float(num));
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!(num));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(num));
|
|
|
|
|
|
|
|
if num.is_finite() {
|
|
|
|
let response: f64 = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, num);
|
|
|
|
} else {
|
|
|
|
let response: Option<f64> = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, None);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn number_decimal() {
|
|
|
|
for num in [i64::MIN, 0, i64::MAX] {
|
|
|
|
let num = Decimal::new(num, 0);
|
|
|
|
let value = Value::Number(sql::Number::Decimal(num));
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!(num.to_string()));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(num));
|
|
|
|
|
|
|
|
let response: Decimal = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, num);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn strand() {
|
|
|
|
for str in ["", "foo"] {
|
|
|
|
let value = Value::Strand(str.into());
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!(str));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(str));
|
|
|
|
|
|
|
|
let response: String = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, str);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn duration() {
|
|
|
|
for duration in [Duration::ZERO, Duration::MAX] {
|
|
|
|
let value = Value::Duration(duration.into());
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!(sql::Duration(duration).to_raw()));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(duration));
|
|
|
|
|
|
|
|
let response: Duration = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, duration);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn datetime() {
|
|
|
|
for datetime in [DateTime::<Utc>::MIN_UTC, DateTime::<Utc>::MAX_UTC] {
|
|
|
|
let value = Value::Datetime(datetime.into());
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!(datetime));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(datetime));
|
|
|
|
|
|
|
|
let response: DateTime<Utc> = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, datetime);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn uuid() {
|
|
|
|
for uuid in [Uuid::nil(), Uuid::max()] {
|
|
|
|
let value = Value::Uuid(uuid.into());
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!(uuid));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(uuid));
|
|
|
|
|
|
|
|
let response: Uuid = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, uuid);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn array() {
|
|
|
|
for vec in [vec![], vec![true, false]] {
|
|
|
|
let value =
|
|
|
|
Value::Array(sql::Array(vec.iter().copied().map(Value::from).collect()));
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!(vec));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(vec));
|
|
|
|
|
|
|
|
let response: Vec<bool> = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, vec);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn object() {
|
|
|
|
for map in [BTreeMap::new(), map!("done".to_owned() => true)] {
|
|
|
|
let value = Value::Object(sql::Object(
|
|
|
|
map.iter().map(|(key, value)| (key.clone(), Value::from(*value))).collect(),
|
|
|
|
));
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!(map));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(map));
|
|
|
|
|
|
|
|
let response: BTreeMap<String, bool> = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, map);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn geometry_point() {
|
|
|
|
let point = point! { x: 10., y: 20. };
|
|
|
|
let value = Value::Geometry(sql::Geometry::Point(point));
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!({ "type": "Point", "coordinates": [10., 20.]}));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(point));
|
|
|
|
|
|
|
|
let response: Point = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, point);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn geometry_line() {
|
|
|
|
let line_string = line_string![
|
|
|
|
( x: 0., y: 0. ),
|
|
|
|
( x: 10., y: 0. ),
|
|
|
|
];
|
|
|
|
let value = Value::Geometry(sql::Geometry::Line(line_string.clone()));
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(
|
|
|
|
simple_json,
|
|
|
|
json!({ "type": "LineString", "coordinates": [[0., 0.], [10., 0.]]})
|
|
|
|
);
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(line_string));
|
|
|
|
|
|
|
|
let response: LineString = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, line_string);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn geometry_polygon() {
|
|
|
|
let polygon = polygon![
|
|
|
|
(x: -111., y: 45.),
|
|
|
|
(x: -111., y: 41.),
|
|
|
|
(x: -104., y: 41.),
|
|
|
|
(x: -104., y: 45.),
|
|
|
|
];
|
|
|
|
let value = Value::Geometry(sql::Geometry::Polygon(polygon.clone()));
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(
|
|
|
|
simple_json,
|
|
|
|
json!({ "type": "Polygon", "coordinates": [[
|
|
|
|
[-111., 45.],
|
|
|
|
[-111., 41.],
|
|
|
|
[-104., 41.],
|
|
|
|
[-104., 45.],
|
|
|
|
[-111., 45.],
|
|
|
|
]]})
|
|
|
|
);
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(polygon));
|
|
|
|
|
|
|
|
let response: Polygon = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, polygon);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn geometry_multi_point() {
|
|
|
|
let multi_point: MultiPoint =
|
|
|
|
vec![point! { x: 0., y: 0. }, point! { x: 1., y: 2. }].into();
|
|
|
|
let value = Value::Geometry(sql::Geometry::MultiPoint(multi_point.clone()));
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(
|
|
|
|
simple_json,
|
|
|
|
json!({ "type": "MultiPoint", "coordinates": [[0., 0.], [1., 2.]]})
|
|
|
|
);
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(multi_point));
|
|
|
|
|
|
|
|
let response: MultiPoint = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, multi_point);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn geometry_multi_line() {
|
|
|
|
let multi_line = MultiLineString::new(vec![line_string![
|
|
|
|
( x: 0., y: 0. ),
|
|
|
|
( x: 1., y: 2. ),
|
|
|
|
]]);
|
|
|
|
let value = Value::Geometry(sql::Geometry::MultiLine(multi_line.clone()));
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(
|
|
|
|
simple_json,
|
|
|
|
json!({ "type": "MultiLineString", "coordinates": [[[0., 0.], [1., 2.]]]})
|
|
|
|
);
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(multi_line));
|
|
|
|
|
|
|
|
let response: MultiLineString = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, multi_line);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn geometry_multi_polygon() {
|
|
|
|
let multi_polygon: MultiPolygon = vec![polygon![
|
|
|
|
(x: -111., y: 45.),
|
|
|
|
(x: -111., y: 41.),
|
|
|
|
(x: -104., y: 41.),
|
|
|
|
(x: -104., y: 45.),
|
|
|
|
]]
|
|
|
|
.into();
|
|
|
|
let value = Value::Geometry(sql::Geometry::MultiPolygon(multi_polygon.clone()));
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(
|
|
|
|
simple_json,
|
|
|
|
json!({ "type": "MultiPolygon", "coordinates": [[[
|
|
|
|
[-111., 45.],
|
|
|
|
[-111., 41.],
|
|
|
|
[-104., 41.],
|
|
|
|
[-104., 45.],
|
|
|
|
[-111., 45.],
|
|
|
|
]]]})
|
|
|
|
);
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(multi_polygon));
|
|
|
|
|
|
|
|
let response: MultiPolygon = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, multi_polygon);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn geometry_collection() {
|
|
|
|
for geometries in [vec![], vec![sql::Geometry::Point(point! { x: 10., y: 20. })]] {
|
|
|
|
let value = Value::Geometry(geometries.clone().into());
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(
|
|
|
|
simple_json,
|
|
|
|
json!({
|
|
|
|
"type": "GeometryCollection",
|
|
|
|
"geometries": geometries.clone().into_iter().map(|geo| into_json(Value::from(geo), true)).collect::<Vec<_>>(),
|
|
|
|
})
|
|
|
|
);
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(geometries));
|
|
|
|
|
|
|
|
let response: Vec<sql::Geometry> = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, geometries);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn bytes() {
|
|
|
|
for bytes in [vec![], b"foo".to_vec()] {
|
|
|
|
let value = Value::Bytes(sql::Bytes(bytes.clone()));
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!(bytes));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(bytes));
|
|
|
|
|
|
|
|
let response: Vec<u8> = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, bytes);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn thing() {
|
|
|
|
let record_id = "foo:bar";
|
|
|
|
let thing = sql::thing(record_id).unwrap();
|
|
|
|
let value = Value::Thing(thing.clone());
|
|
|
|
|
|
|
|
let simple_json = into_json(value.clone(), true);
|
|
|
|
assert_eq!(simple_json, json!(record_id));
|
|
|
|
|
|
|
|
let json = into_json(value.clone(), false);
|
|
|
|
assert_eq!(json, json!(thing));
|
|
|
|
|
|
|
|
let response: sql::Thing = from_value(value).unwrap();
|
|
|
|
assert_eq!(response, thing);
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|