2022-03-18 07:24:36 +00:00
|
|
|
use crate::cnf::ID_CHARS;
|
2022-11-03 10:55:32 +00:00
|
|
|
use crate::ctx::Context;
|
|
|
|
use crate::dbs::Options;
|
|
|
|
use crate::dbs::Transaction;
|
|
|
|
use crate::err::Error;
|
2022-08-26 23:23:24 +00:00
|
|
|
use crate::sql::array::{array, Array};
|
2022-03-18 07:24:36 +00:00
|
|
|
use crate::sql::error::IResult;
|
2022-10-16 23:04:07 +00:00
|
|
|
use crate::sql::escape::escape_rid;
|
2022-03-18 07:24:36 +00:00
|
|
|
use crate::sql::ident::ident_raw;
|
2022-06-08 07:39:09 +00:00
|
|
|
use crate::sql::number::integer;
|
2023-04-25 10:13:04 +00:00
|
|
|
use crate::sql::number::Number;
|
2022-08-26 23:23:24 +00:00
|
|
|
use crate::sql::object::{object, Object};
|
2022-09-24 01:57:27 +00:00
|
|
|
use crate::sql::strand::Strand;
|
2022-11-23 08:58:56 +00:00
|
|
|
use crate::sql::thing::Thing;
|
2022-09-24 01:57:27 +00:00
|
|
|
use crate::sql::uuid::Uuid;
|
2022-10-03 23:27:38 +00:00
|
|
|
use crate::sql::value::Value;
|
2022-03-18 07:24:36 +00:00
|
|
|
use nanoid::nanoid;
|
|
|
|
use nom::branch::alt;
|
|
|
|
use nom::combinator::map;
|
|
|
|
use serde::{Deserialize, Serialize};
|
2022-10-04 21:51:18 +00:00
|
|
|
use std::fmt::{self, Display, Formatter};
|
2023-01-17 09:53:30 +00:00
|
|
|
use ulid::Ulid;
|
2022-03-18 07:24:36 +00:00
|
|
|
|
2023-03-30 10:41:44 +00:00
|
|
|
#[derive(Clone, Debug, Eq, PartialEq, Ord, PartialOrd, Serialize, Deserialize, Hash)]
|
2022-03-18 07:24:36 +00:00
|
|
|
pub enum Id {
|
2022-06-08 07:39:09 +00:00
|
|
|
Number(i64),
|
2023-05-09 17:48:14 +00:00
|
|
|
/// Invariant: Doesn't contain NUL bytes.
|
2022-03-18 07:24:36 +00:00
|
|
|
String(String),
|
2022-08-26 23:23:24 +00:00
|
|
|
Array(Array),
|
2022-08-31 13:37:22 +00:00
|
|
|
Object(Object),
|
2022-03-18 07:24:36 +00:00
|
|
|
}
|
|
|
|
|
2022-06-08 07:39:09 +00:00
|
|
|
impl From<i64> for Id {
|
|
|
|
fn from(v: i64) -> Self {
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::Number(v)
|
2022-03-18 07:24:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-06-08 07:39:09 +00:00
|
|
|
impl From<i32> for Id {
|
|
|
|
fn from(v: i32) -> Self {
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::Number(v as i64)
|
2022-06-08 07:39:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<u64> for Id {
|
|
|
|
fn from(v: u64) -> Self {
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::Number(v as i64)
|
2022-06-08 07:39:09 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-24 01:57:27 +00:00
|
|
|
impl From<String> for Id {
|
|
|
|
fn from(v: String) -> Self {
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::String(v)
|
2022-09-24 01:57:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-14 22:40:19 +00:00
|
|
|
impl From<Array> for Id {
|
|
|
|
fn from(v: Array) -> Self {
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::Array(v)
|
2022-09-14 22:40:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Object> for Id {
|
|
|
|
fn from(v: Object) -> Self {
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::Object(v)
|
2022-09-14 22:40:19 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-09-24 01:57:27 +00:00
|
|
|
impl From<Uuid> for Id {
|
|
|
|
fn from(v: Uuid) -> Self {
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::String(v.to_raw())
|
2022-09-24 01:57:27 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Strand> for Id {
|
|
|
|
fn from(v: Strand) -> Self {
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::String(v.as_string())
|
2022-03-18 07:24:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<&str> for Id {
|
|
|
|
fn from(v: &str) -> Self {
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::String(v.to_owned())
|
2022-03-18 07:24:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-12-30 08:23:19 +00:00
|
|
|
impl From<&String> for Id {
|
|
|
|
fn from(v: &String) -> Self {
|
|
|
|
Self::String(v.to_owned())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Vec<&str>> for Id {
|
|
|
|
fn from(v: Vec<&str>) -> Self {
|
|
|
|
Id::Array(v.into())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
impl From<Vec<String>> for Id {
|
|
|
|
fn from(v: Vec<String>) -> Self {
|
|
|
|
Id::Array(v.into())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-03 23:27:38 +00:00
|
|
|
impl From<Vec<Value>> for Id {
|
|
|
|
fn from(v: Vec<Value>) -> Self {
|
|
|
|
Id::Array(v.into())
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2023-04-25 10:13:04 +00:00
|
|
|
impl From<Number> for Id {
|
|
|
|
fn from(v: Number) -> Self {
|
|
|
|
match v {
|
|
|
|
Number::Int(v) => v.into(),
|
|
|
|
Number::Float(v) => v.to_string().into(),
|
|
|
|
Number::Decimal(v) => v.to_string().into(),
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-23 08:58:56 +00:00
|
|
|
impl From<Thing> for Id {
|
|
|
|
fn from(v: Thing) -> Self {
|
|
|
|
v.id
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-18 07:24:36 +00:00
|
|
|
impl Id {
|
2022-10-19 09:55:19 +00:00
|
|
|
/// Generate a new random ID
|
2022-10-04 21:51:18 +00:00
|
|
|
pub fn rand() -> Self {
|
|
|
|
Self::String(nanoid!(20, &ID_CHARS))
|
2022-03-18 07:24:36 +00:00
|
|
|
}
|
2023-01-17 09:53:30 +00:00
|
|
|
/// Generate a new random ULID
|
|
|
|
pub fn ulid() -> Self {
|
|
|
|
Self::String(Ulid::new().to_string())
|
|
|
|
}
|
|
|
|
/// Generate a new random UUID
|
|
|
|
#[cfg(uuid_unstable)]
|
|
|
|
pub fn uuid() -> Self {
|
|
|
|
Self::String(Uuid::new_v7().to_raw())
|
|
|
|
}
|
|
|
|
/// Generate a new random UUID
|
|
|
|
#[cfg(not(uuid_unstable))]
|
|
|
|
pub fn uuid() -> Self {
|
|
|
|
Self::String(Uuid::new_v4().to_raw())
|
|
|
|
}
|
2022-10-19 09:55:19 +00:00
|
|
|
/// Convert the Id to a raw String
|
2022-07-06 09:17:16 +00:00
|
|
|
pub fn to_raw(&self) -> String {
|
|
|
|
match self {
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::Number(v) => v.to_string(),
|
|
|
|
Self::String(v) => v.to_string(),
|
|
|
|
Self::Object(v) => v.to_string(),
|
|
|
|
Self::Array(v) => v.to_string(),
|
2022-07-06 09:17:16 +00:00
|
|
|
}
|
|
|
|
}
|
2022-03-18 07:24:36 +00:00
|
|
|
}
|
|
|
|
|
2022-10-04 21:51:18 +00:00
|
|
|
impl Display for Id {
|
|
|
|
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
|
2022-03-18 07:24:36 +00:00
|
|
|
match self {
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::Number(v) => Display::fmt(v, f),
|
2022-10-16 23:04:07 +00:00
|
|
|
Self::String(v) => Display::fmt(&escape_rid(v), f),
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::Object(v) => Display::fmt(v, f),
|
|
|
|
Self::Array(v) => Display::fmt(v, f),
|
2022-03-18 07:24:36 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-11-03 10:55:32 +00:00
|
|
|
impl Id {
|
|
|
|
pub(crate) async fn compute(
|
|
|
|
&self,
|
|
|
|
ctx: &Context<'_>,
|
|
|
|
opt: &Options,
|
|
|
|
txn: &Transaction,
|
|
|
|
doc: Option<&Value>,
|
|
|
|
) -> Result<Id, Error> {
|
|
|
|
match self {
|
|
|
|
Id::Number(v) => Ok(Id::Number(*v)),
|
|
|
|
Id::String(v) => Ok(Id::String(v.clone())),
|
|
|
|
Id::Object(v) => match v.compute(ctx, opt, txn, doc).await? {
|
|
|
|
Value::Object(v) => Ok(Id::Object(v)),
|
|
|
|
_ => unreachable!(),
|
|
|
|
},
|
|
|
|
Id::Array(v) => match v.compute(ctx, opt, txn, doc).await? {
|
|
|
|
Value::Array(v) => Ok(Id::Array(v)),
|
|
|
|
_ => unreachable!(),
|
|
|
|
},
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-03-18 07:24:36 +00:00
|
|
|
pub fn id(i: &str) -> IResult<&str, Id> {
|
2022-08-26 23:23:24 +00:00
|
|
|
alt((
|
|
|
|
map(integer, Id::Number),
|
|
|
|
map(ident_raw, Id::String),
|
|
|
|
map(object, Id::Object),
|
|
|
|
map(array, Id::Array),
|
|
|
|
))(i)
|
2022-03-18 07:24:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
2022-09-14 22:40:19 +00:00
|
|
|
#[test]
|
|
|
|
fn id_int() {
|
|
|
|
let sql = "001";
|
|
|
|
let res = id(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!(Id::from(1), out);
|
2022-10-16 23:04:07 +00:00
|
|
|
assert_eq!("1", format!("{}", out));
|
2022-09-14 22:40:19 +00:00
|
|
|
}
|
|
|
|
|
2022-03-18 07:24:36 +00:00
|
|
|
#[test]
|
|
|
|
fn id_number() {
|
|
|
|
let sql = "100";
|
|
|
|
let res = id(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!(Id::from(100), out);
|
2022-10-16 23:04:07 +00:00
|
|
|
assert_eq!("100", format!("{}", out));
|
2022-03-18 07:24:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn id_string() {
|
|
|
|
let sql = "test";
|
|
|
|
let res = id(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!(Id::from("test"), out);
|
2022-10-16 23:04:07 +00:00
|
|
|
assert_eq!("test", format!("{}", out));
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn id_numeric() {
|
|
|
|
let sql = "⟨100⟩";
|
|
|
|
let res = id(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!(Id::from("100"), out);
|
|
|
|
assert_eq!("⟨100⟩", format!("{}", out));
|
2022-03-18 07:24:36 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn id_either() {
|
|
|
|
let sql = "100test";
|
|
|
|
let res = id(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!(Id::from("100test"), out);
|
2022-10-16 23:04:07 +00:00
|
|
|
assert_eq!("100test", format!("{}", out));
|
2022-03-18 07:24:36 +00:00
|
|
|
}
|
|
|
|
}
|