2022-05-14 12:35:08 +00:00
|
|
|
use crate::ctx::Context;
|
2024-05-28 10:43:45 +00:00
|
|
|
use crate::dbs::Operable;
|
2022-02-06 01:14:56 +00:00
|
|
|
use crate::dbs::Statement;
|
2022-05-30 15:32:26 +00:00
|
|
|
use crate::dbs::Workable;
|
2023-07-20 12:56:32 +00:00
|
|
|
use crate::dbs::{Options, Processed};
|
2022-02-06 01:14:56 +00:00
|
|
|
use crate::doc::Document;
|
|
|
|
use crate::err::Error;
|
|
|
|
use crate::sql::value::Value;
|
2022-05-13 20:46:56 +00:00
|
|
|
use channel::Sender;
|
2024-04-18 15:51:47 +00:00
|
|
|
use reblessive::tree::Stk;
|
2022-02-06 01:14:56 +00:00
|
|
|
|
2022-02-13 19:03:00 +00:00
|
|
|
impl<'a> Document<'a> {
|
2022-12-30 08:23:19 +00:00
|
|
|
#[allow(dead_code)]
|
2022-05-01 22:25:53 +00:00
|
|
|
pub(crate) async fn compute(
|
2024-04-18 15:51:47 +00:00
|
|
|
stk: &mut Stk,
|
2022-05-14 12:35:08 +00:00
|
|
|
ctx: &Context<'_>,
|
2022-05-13 20:46:56 +00:00
|
|
|
opt: &Options,
|
|
|
|
stm: &Statement<'_>,
|
2022-02-26 23:30:19 +00:00
|
|
|
chn: Sender<Result<Value, Error>>,
|
2023-08-29 08:46:48 +00:00
|
|
|
mut pro: Processed,
|
2022-02-26 23:30:19 +00:00
|
|
|
) -> Result<(), Error> {
|
2023-08-29 08:46:48 +00:00
|
|
|
// Loop over maximum two times
|
|
|
|
for _ in 0..2 {
|
|
|
|
// Check current context
|
|
|
|
if ctx.is_done() {
|
|
|
|
break;
|
|
|
|
}
|
|
|
|
// Setup a new workable
|
|
|
|
let ins = match pro.val {
|
|
|
|
Operable::Value(v) => (v, Workable::Normal),
|
|
|
|
Operable::Mergeable(v, o) => (v, Workable::Insert(o)),
|
|
|
|
Operable::Relatable(f, v, w) => (v, Workable::Relate(f, w)),
|
|
|
|
};
|
|
|
|
// Setup a new document
|
2024-05-24 13:45:21 +00:00
|
|
|
let mut doc = Document::new(pro.rid.as_ref(), pro.ir.as_ref(), &ins.0, ins.1);
|
2023-08-29 08:46:48 +00:00
|
|
|
// Process the statement
|
|
|
|
let res = match stm {
|
2024-05-28 10:43:45 +00:00
|
|
|
Statement::Select(_) => doc.select(stk, ctx, opt, stm).await,
|
|
|
|
Statement::Create(_) => doc.create(stk, ctx, opt, stm).await,
|
|
|
|
Statement::Update(_) => doc.update(stk, ctx, opt, stm).await,
|
|
|
|
Statement::Relate(_) => doc.relate(stk, ctx, opt, stm).await,
|
|
|
|
Statement::Delete(_) => doc.delete(stk, ctx, opt, stm).await,
|
|
|
|
Statement::Insert(_) => doc.insert(stk, ctx, opt, stm).await,
|
2023-08-29 08:46:48 +00:00
|
|
|
_ => unreachable!(),
|
|
|
|
};
|
|
|
|
// Check the result
|
|
|
|
let res = match res {
|
|
|
|
// We received an error suggesting that we
|
|
|
|
// retry this request using a new ID, so
|
|
|
|
// we load the new record, and reprocess
|
|
|
|
Err(Error::RetryWithId(v)) => {
|
|
|
|
// Fetch the data from the store
|
|
|
|
let key = crate::key::thing::new(opt.ns(), opt.db(), &v.tb, &v.id);
|
2024-05-28 10:43:45 +00:00
|
|
|
let val = ctx.tx_lock().await.get(key).await?;
|
2023-08-29 08:46:48 +00:00
|
|
|
// Parse the data from the store
|
|
|
|
let val = match val {
|
|
|
|
Some(v) => Value::from(v),
|
|
|
|
None => Value::None,
|
|
|
|
};
|
|
|
|
pro = Processed {
|
|
|
|
rid: Some(v),
|
2024-05-24 13:45:21 +00:00
|
|
|
ir: None,
|
2023-08-29 08:46:48 +00:00
|
|
|
val: match doc.extras {
|
|
|
|
Workable::Normal => Operable::Value(val),
|
|
|
|
Workable::Insert(o) => Operable::Mergeable(val, o),
|
|
|
|
Workable::Relate(f, w) => Operable::Relatable(f, val, w),
|
|
|
|
},
|
|
|
|
};
|
|
|
|
// Go to top of loop
|
|
|
|
continue;
|
|
|
|
}
|
|
|
|
// If any other error was received, then let's
|
|
|
|
// pass that error through and return an error
|
|
|
|
Err(e) => Err(e),
|
|
|
|
// Otherwise the record creation succeeded
|
|
|
|
Ok(v) => Ok(v),
|
|
|
|
};
|
|
|
|
// Send back the result
|
|
|
|
let _ = chn.send(res).await;
|
|
|
|
// Break the loop
|
|
|
|
break;
|
|
|
|
}
|
2022-02-26 23:30:19 +00:00
|
|
|
// Everything went ok
|
|
|
|
Ok(())
|
2022-02-06 01:14:56 +00:00
|
|
|
}
|
|
|
|
}
|