surrealdb_core/sql/statements/
show.rsuse crate::ctx::Context;
use crate::dbs::{Options, Transaction};
use crate::doc::CursorDoc;
use crate::err::Error;
use crate::iam::{Action, ResourceKind};
use crate::sql::{Base, Datetime, Table, Value};
use crate::vs::{conv, Versionstamp};
use derive::Store;
use revision::revisioned;
use serde::{Deserialize, Serialize};
use std::fmt;
#[revisioned(revision = 1)]
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Hash)]
#[cfg_attr(feature = "arbitrary", derive(arbitrary::Arbitrary))]
pub enum ShowSince {
Timestamp(Datetime),
Versionstamp(u64),
}
impl ShowSince {
pub fn versionstamp(vs: &Versionstamp) -> ShowSince {
ShowSince::Versionstamp(conv::versionstamp_to_u64(vs))
}
pub fn as_versionstamp(&self) -> Option<Versionstamp> {
match self {
ShowSince::Timestamp(_) => None,
ShowSince::Versionstamp(v) => Some(conv::u64_to_versionstamp(*v)),
}
}
}
#[revisioned(revision = 1)]
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Serialize, Deserialize, Store, Hash)]
#[cfg_attr(feature = "arbitrary", derive(arbitrary::Arbitrary))]
pub struct ShowStatement {
pub table: Option<Table>,
pub since: ShowSince,
pub limit: Option<u32>,
}
impl ShowStatement {
pub(crate) async fn compute(
&self,
_ctx: &Context<'_>,
opt: &Options,
txn: &Transaction,
_doc: Option<&CursorDoc<'_>>,
) -> Result<Value, Error> {
opt.is_allowed(Action::View, ResourceKind::Table, &Base::Db)?;
let txn = txn.clone();
let mut run = txn.lock().await;
let tb = self.table.as_deref();
let r = crate::cf::read(
&mut run,
opt.ns(),
opt.db(),
tb.map(|x| x.as_str()),
self.since.clone(),
self.limit,
)
.await?;
let mut a = Vec::<Value>::new();
for r in r.iter() {
let v: Value = r.clone().into_value();
a.push(v);
}
let v: Value = Value::Array(crate::sql::array::Array(a));
Ok(v)
}
}
impl fmt::Display for ShowStatement {
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
write!(f, "SHOW CHANGES FOR")?;
match self.table {
Some(ref v) => write!(f, " TABLE {}", v)?,
None => write!(f, " DATABASE")?,
}
match self.since {
ShowSince::Timestamp(ref v) => write!(f, " SINCE {}", v)?,
ShowSince::Versionstamp(ref v) => write!(f, " SINCE {}", v)?,
}
if let Some(ref v) = self.limit {
write!(f, " LIMIT {}", v)?
}
Ok(())
}
}
#[cfg(test)]
mod test {
use crate::sql::Datetime;
#[test]
fn timestamps_are_not_versionstamps() {
let sql_dt = Datetime::try_from("2020-01-01T00:00:00Z").unwrap();
let since = super::ShowSince::Timestamp(sql_dt);
assert_eq!(since.as_versionstamp(), None);
}
#[test]
fn versionstamp_can_be_converted() {
let versionstamp = crate::vs::conv::u64_to_versionstamp(1234567890);
let since = super::ShowSince::Versionstamp(1234567890);
let converted = since.as_versionstamp().unwrap();
assert_eq!(converted, versionstamp);
}
}