2021-03-29 15:43:37 +00:00
|
|
|
use crate::ctx::Context;
|
2023-07-06 14:57:42 +00:00
|
|
|
use crate::dbs::{Options, Transaction};
|
|
|
|
use crate::doc::CursorDoc;
|
2021-03-29 15:43:37 +00:00
|
|
|
use crate::err::Error;
|
2024-04-24 13:30:58 +00:00
|
|
|
use crate::sql::statements::rebuild::RebuildStatement;
|
2023-11-18 13:56:13 +00:00
|
|
|
use crate::sql::statements::{
|
|
|
|
CreateStatement, DefineStatement, DeleteStatement, IfelseStatement, InsertStatement,
|
|
|
|
OutputStatement, RelateStatement, RemoveStatement, SelectStatement, UpdateStatement,
|
|
|
|
};
|
|
|
|
use crate::sql::value::Value;
|
2024-04-18 15:51:47 +00:00
|
|
|
use reblessive::tree::Stk;
|
2023-08-17 18:03:46 +00:00
|
|
|
use revision::revisioned;
|
2020-06-29 15:36:01 +00:00
|
|
|
use serde::{Deserialize, Serialize};
|
2021-03-29 15:43:37 +00:00
|
|
|
use std::cmp::Ordering;
|
2022-10-04 21:51:18 +00:00
|
|
|
use std::fmt::{self, Display, Formatter};
|
2020-06-29 15:36:01 +00:00
|
|
|
|
2023-03-30 10:41:44 +00:00
|
|
|
pub(crate) const TOKEN: &str = "$surrealdb::private::sql::Subquery";
|
|
|
|
|
2024-04-24 13:30:58 +00:00
|
|
|
#[revisioned(revision = 2)]
|
2023-04-29 20:50:25 +00:00
|
|
|
#[derive(Clone, Debug, Eq, PartialEq, Serialize, Deserialize, Hash)]
|
|
|
|
#[serde(rename = "$surrealdb::private::sql::Subquery")]
|
2024-01-09 15:34:52 +00:00
|
|
|
#[cfg_attr(feature = "arbitrary", derive(arbitrary::Arbitrary))]
|
2024-04-02 20:12:08 +00:00
|
|
|
#[non_exhaustive]
|
2020-06-29 15:36:01 +00:00
|
|
|
pub enum Subquery {
|
2022-01-13 17:36:41 +00:00
|
|
|
Value(Value),
|
2020-06-29 15:36:01 +00:00
|
|
|
Ifelse(IfelseStatement),
|
2023-02-19 19:41:20 +00:00
|
|
|
Output(OutputStatement),
|
2022-05-13 20:46:56 +00:00
|
|
|
Select(SelectStatement),
|
|
|
|
Create(CreateStatement),
|
|
|
|
Update(UpdateStatement),
|
|
|
|
Delete(DeleteStatement),
|
|
|
|
Relate(RelateStatement),
|
|
|
|
Insert(InsertStatement),
|
2023-08-23 16:05:50 +00:00
|
|
|
Define(DefineStatement),
|
|
|
|
Remove(RemoveStatement),
|
2024-04-24 13:30:58 +00:00
|
|
|
#[revision(start = 2)]
|
|
|
|
Rebuild(RebuildStatement),
|
2023-04-29 20:50:25 +00:00
|
|
|
// Add new variants here
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
|
2021-03-29 15:43:37 +00:00
|
|
|
impl PartialOrd for Subquery {
|
|
|
|
#[inline]
|
|
|
|
fn partial_cmp(&self, _: &Self) -> Option<Ordering> {
|
2022-03-23 11:56:39 +00:00
|
|
|
None
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-01-14 08:12:56 +00:00
|
|
|
impl Subquery {
|
2023-05-09 22:17:29 +00:00
|
|
|
/// Check if we require a writeable transaction
|
2022-05-14 21:30:49 +00:00
|
|
|
pub(crate) fn writeable(&self) -> bool {
|
|
|
|
match self {
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::Value(v) => v.writeable(),
|
|
|
|
Self::Ifelse(v) => v.writeable(),
|
2023-02-19 19:41:20 +00:00
|
|
|
Self::Output(v) => v.writeable(),
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::Select(v) => v.writeable(),
|
|
|
|
Self::Create(v) => v.writeable(),
|
|
|
|
Self::Update(v) => v.writeable(),
|
|
|
|
Self::Delete(v) => v.writeable(),
|
|
|
|
Self::Relate(v) => v.writeable(),
|
|
|
|
Self::Insert(v) => v.writeable(),
|
2023-08-23 16:05:50 +00:00
|
|
|
Self::Define(v) => v.writeable(),
|
|
|
|
Self::Remove(v) => v.writeable(),
|
2024-04-24 13:30:58 +00:00
|
|
|
Self::Rebuild(v) => v.writeable(),
|
2022-05-14 21:30:49 +00:00
|
|
|
}
|
|
|
|
}
|
2023-05-09 22:17:29 +00:00
|
|
|
/// Process this type returning a computed simple Value
|
2023-07-06 14:57:42 +00:00
|
|
|
pub(crate) async fn compute(
|
|
|
|
&self,
|
2024-04-18 15:51:47 +00:00
|
|
|
stk: &mut Stk,
|
2023-07-06 14:57:42 +00:00
|
|
|
ctx: &Context<'_>,
|
|
|
|
opt: &Options,
|
|
|
|
txn: &Transaction,
|
|
|
|
doc: Option<&CursorDoc<'_>>,
|
|
|
|
) -> Result<Value, Error> {
|
2023-09-05 15:22:13 +00:00
|
|
|
// Duplicate context
|
|
|
|
let mut ctx = Context::new(ctx);
|
|
|
|
// Add parent document
|
|
|
|
if let Some(doc) = doc {
|
|
|
|
ctx.add_value("parent", doc.doc.as_ref());
|
|
|
|
}
|
2022-10-06 16:35:03 +00:00
|
|
|
// Process the subquery
|
2021-03-29 15:43:37 +00:00
|
|
|
match self {
|
2024-04-18 15:51:47 +00:00
|
|
|
Self::Value(ref v) => v.compute(stk, &ctx, opt, txn, doc).await,
|
|
|
|
Self::Ifelse(ref v) => v.compute(stk, &ctx, opt, txn, doc).await,
|
|
|
|
Self::Output(ref v) => v.compute(stk, &ctx, opt, txn, doc).await,
|
|
|
|
Self::Define(ref v) => v.compute(stk, &ctx, opt, txn, doc).await,
|
2024-04-24 13:30:58 +00:00
|
|
|
Self::Rebuild(ref v) => v.compute(stk, &ctx, opt, txn, doc).await,
|
2023-09-05 15:22:13 +00:00
|
|
|
Self::Remove(ref v) => v.compute(&ctx, opt, txn, doc).await,
|
2024-04-18 15:51:47 +00:00
|
|
|
Self::Select(ref v) => v.compute(stk, &ctx, opt, txn, doc).await,
|
|
|
|
Self::Create(ref v) => v.compute(stk, &ctx, opt, txn, doc).await,
|
|
|
|
Self::Update(ref v) => v.compute(stk, &ctx, opt, txn, doc).await,
|
|
|
|
Self::Delete(ref v) => v.compute(stk, &ctx, opt, txn, doc).await,
|
|
|
|
Self::Relate(ref v) => v.compute(stk, &ctx, opt, txn, doc).await,
|
|
|
|
Self::Insert(ref v) => v.compute(stk, &ctx, opt, txn, doc).await,
|
2021-03-29 15:43:37 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-10-04 21:51:18 +00:00
|
|
|
impl Display for Subquery {
|
|
|
|
fn fmt(&self, f: &mut Formatter) -> fmt::Result {
|
2022-01-14 08:12:56 +00:00
|
|
|
match self {
|
2023-02-03 11:47:07 +00:00
|
|
|
Self::Value(v) => write!(f, "({v})"),
|
2023-02-19 19:41:20 +00:00
|
|
|
Self::Output(v) => write!(f, "({v})"),
|
2023-02-03 11:47:07 +00:00
|
|
|
Self::Select(v) => write!(f, "({v})"),
|
|
|
|
Self::Create(v) => write!(f, "({v})"),
|
|
|
|
Self::Update(v) => write!(f, "({v})"),
|
|
|
|
Self::Delete(v) => write!(f, "({v})"),
|
|
|
|
Self::Relate(v) => write!(f, "({v})"),
|
|
|
|
Self::Insert(v) => write!(f, "({v})"),
|
2023-08-23 16:05:50 +00:00
|
|
|
Self::Define(v) => write!(f, "({v})"),
|
|
|
|
Self::Remove(v) => write!(f, "({v})"),
|
2024-04-24 13:30:58 +00:00
|
|
|
Self::Rebuild(v) => write!(f, "({v})"),
|
2022-10-04 21:51:18 +00:00
|
|
|
Self::Ifelse(v) => Display::fmt(v, f),
|
2022-01-14 08:12:56 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|