Add integration tests for the built-in SQL functions (#1893)
This commit is contained in:
parent
12bf8dea6b
commit
453d368e69
7 changed files with 3517 additions and 18 deletions
|
@ -178,6 +178,7 @@ pub fn synchronous(ctx: &Context<'_>, name: &str, args: Vec<Value>) -> Result<Va
|
||||||
"parse::url::query" => parse::url::query,
|
"parse::url::query" => parse::url::query,
|
||||||
"parse::url::scheme" => parse::url::scheme,
|
"parse::url::scheme" => parse::url::scheme,
|
||||||
//
|
//
|
||||||
|
"rand" => rand::rand,
|
||||||
"rand::bool" => rand::bool,
|
"rand::bool" => rand::bool,
|
||||||
"rand::enum" => rand::r#enum,
|
"rand::enum" => rand::r#enum,
|
||||||
"rand::float" => rand::float,
|
"rand::float" => rand::float,
|
||||||
|
@ -189,7 +190,6 @@ pub fn synchronous(ctx: &Context<'_>, name: &str, args: Vec<Value>) -> Result<Va
|
||||||
"rand::uuid::v4" => rand::uuid::v4,
|
"rand::uuid::v4" => rand::uuid::v4,
|
||||||
"rand::uuid::v7" => rand::uuid::v7,
|
"rand::uuid::v7" => rand::uuid::v7,
|
||||||
"rand::uuid" => rand::uuid,
|
"rand::uuid" => rand::uuid,
|
||||||
"rand" => rand::rand,
|
|
||||||
//
|
//
|
||||||
"session::db" => session::db(ctx),
|
"session::db" => session::db(ctx),
|
||||||
"session::id" => session::id(ctx),
|
"session::id" => session::id(ctx),
|
||||||
|
|
|
@ -91,8 +91,8 @@ pub fn minute((val,): (Option<Datetime>,)) -> Result<Value, Error> {
|
||||||
|
|
||||||
pub fn month((val,): (Option<Datetime>,)) -> Result<Value, Error> {
|
pub fn month((val,): (Option<Datetime>,)) -> Result<Value, Error> {
|
||||||
Ok(match val {
|
Ok(match val {
|
||||||
Some(v) => v.minute().into(),
|
Some(v) => v.month().into(),
|
||||||
None => Datetime::default().minute().into(),
|
None => Datetime::default().month().into(),
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
|
@ -6,8 +6,13 @@ pub trait Mean {
|
||||||
|
|
||||||
impl Mean for Vec<Number> {
|
impl Mean for Vec<Number> {
|
||||||
fn mean(&self) -> Number {
|
fn mean(&self) -> Number {
|
||||||
|
match self.len() {
|
||||||
|
0 => Number::NAN,
|
||||||
|
_ => {
|
||||||
let len = Number::from(self.len());
|
let len = Number::from(self.len());
|
||||||
let sum = self.iter().sum::<Number>();
|
let sum = self.iter().sum::<Number>();
|
||||||
sum / len
|
sum / len
|
||||||
}
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
}
|
}
|
||||||
|
|
|
@ -20,6 +20,7 @@ static SECONDS_PER_DAY: u64 = 24 * SECONDS_PER_HOUR;
|
||||||
static SECONDS_PER_HOUR: u64 = 60 * SECONDS_PER_MINUTE;
|
static SECONDS_PER_HOUR: u64 = 60 * SECONDS_PER_MINUTE;
|
||||||
static SECONDS_PER_MINUTE: u64 = 60;
|
static SECONDS_PER_MINUTE: u64 = 60;
|
||||||
static NANOSECONDS_PER_MILLISECOND: u32 = 1000000;
|
static NANOSECONDS_PER_MILLISECOND: u32 = 1000000;
|
||||||
|
static NANOSECONDS_PER_MICROSECOND: u32 = 1000;
|
||||||
|
|
||||||
pub(crate) const TOKEN: &str = "$surrealdb::private::sql::Duration";
|
pub(crate) const TOKEN: &str = "$surrealdb::private::sql::Duration";
|
||||||
|
|
||||||
|
@ -179,6 +180,9 @@ impl fmt::Display for Duration {
|
||||||
// Calculate the total millseconds
|
// Calculate the total millseconds
|
||||||
let msec = nano / NANOSECONDS_PER_MILLISECOND;
|
let msec = nano / NANOSECONDS_PER_MILLISECOND;
|
||||||
let nano = nano % NANOSECONDS_PER_MILLISECOND;
|
let nano = nano % NANOSECONDS_PER_MILLISECOND;
|
||||||
|
// Calculate the total microseconds
|
||||||
|
let usec = nano / NANOSECONDS_PER_MICROSECOND;
|
||||||
|
let nano = nano % NANOSECONDS_PER_MICROSECOND;
|
||||||
// Write the different parts
|
// Write the different parts
|
||||||
if year > 0 {
|
if year > 0 {
|
||||||
write!(f, "{year}y")?;
|
write!(f, "{year}y")?;
|
||||||
|
@ -201,6 +205,9 @@ impl fmt::Display for Duration {
|
||||||
if msec > 0 {
|
if msec > 0 {
|
||||||
write!(f, "{msec}ms")?;
|
write!(f, "{msec}ms")?;
|
||||||
}
|
}
|
||||||
|
if usec > 0 {
|
||||||
|
write!(f, "{usec}µs")?;
|
||||||
|
}
|
||||||
if nano > 0 {
|
if nano > 0 {
|
||||||
write!(f, "{nano}ns")?;
|
write!(f, "{nano}ns")?;
|
||||||
}
|
}
|
||||||
|
|
|
@ -179,6 +179,10 @@ impl Number {
|
||||||
// Simple number detection
|
// Simple number detection
|
||||||
// -----------------------------------
|
// -----------------------------------
|
||||||
|
|
||||||
|
pub fn is_nan(&self) -> bool {
|
||||||
|
matches!(self, Number::Float(v) if v.is_nan())
|
||||||
|
}
|
||||||
|
|
||||||
pub fn is_int(&self) -> bool {
|
pub fn is_int(&self) -> bool {
|
||||||
matches!(self, Number::Int(_))
|
matches!(self, Number::Int(_))
|
||||||
}
|
}
|
||||||
|
|
|
@ -872,6 +872,11 @@ impl Value {
|
||||||
matches!(self, Value::Number(Number::Decimal(_)))
|
matches!(self, Value::Number(Number::Decimal(_)))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
/// Check if this Value is a Number but is a NAN
|
||||||
|
pub fn is_nan(&self) -> bool {
|
||||||
|
matches!(self, Value::Number(v) if v.is_nan())
|
||||||
|
}
|
||||||
|
|
||||||
/// Check if this Value is a Number and is an integer
|
/// Check if this Value is a Number and is an integer
|
||||||
pub fn is_integer(&self) -> bool {
|
pub fn is_integer(&self) -> bool {
|
||||||
matches!(self, Value::Number(v) if v.is_integer())
|
matches!(self, Value::Number(v) if v.is_integer())
|
||||||
|
@ -1195,6 +1200,16 @@ impl Value {
|
||||||
match self {
|
match self {
|
||||||
// Allow any boolean value
|
// Allow any boolean value
|
||||||
Value::Bool(boolean) => Ok(boolean),
|
Value::Bool(boolean) => Ok(boolean),
|
||||||
|
// Attempt to convert a string value
|
||||||
|
Value::Strand(ref v) => match v.parse::<bool>() {
|
||||||
|
// The string can be represented as a Float
|
||||||
|
Ok(v) => Ok(v),
|
||||||
|
// Ths string is not a float
|
||||||
|
_ => Err(Error::ConvertTo {
|
||||||
|
from: self,
|
||||||
|
into: "bool".into(),
|
||||||
|
}),
|
||||||
|
},
|
||||||
// Anything else raises an error
|
// Anything else raises an error
|
||||||
_ => Err(Error::ConvertTo {
|
_ => Err(Error::ConvertTo {
|
||||||
from: self,
|
from: self,
|
||||||
|
|
File diff suppressed because it is too large
Load diff
Loading…
Reference in a new issue