use crate::cnf::ID_CHARS; use crate::dbs::Iterator; use crate::dbs::Options; use crate::dbs::Runtime; use crate::dbs::Transaction; use crate::err::Error; use crate::key; use crate::key::thing; use crate::sql::array::Array; use crate::sql::model::Model; use crate::sql::table::Table; use crate::sql::thing::Thing; use crate::sql::value::Value; use async_recursion::async_recursion; use nanoid::nanoid; impl Value { #[cfg_attr(feature = "parallel", async_recursion)] #[cfg_attr(not(feature = "parallel"), async_recursion(?Send))] pub async fn iterate( self, ctx: &Runtime, opt: &Options, txn: &Transaction, ite: &mut Iterator, ) -> Result<(), Error> { if ctx.is_ok() { match self { Value::Array(v) => v.iterate(ctx, opt, txn, ite).await?, Value::Model(v) => v.iterate(ctx, opt, txn, ite).await?, Value::Thing(v) => v.iterate(ctx, opt, txn, ite).await?, Value::Table(v) => v.iterate(ctx, opt, txn, ite).await?, v => ite.process(ctx, opt, txn, None, v).await, } } Ok(()) } } impl Array { #[cfg_attr(feature = "parallel", async_recursion)] #[cfg_attr(not(feature = "parallel"), async_recursion(?Send))] pub async fn iterate( self, ctx: &Runtime, opt: &Options, txn: &Transaction, ite: &mut Iterator, ) -> Result<(), Error> { for v in self.value.into_iter() { if ctx.is_ok() { match v { Value::Array(v) => v.iterate(ctx, opt, txn, ite).await?, Value::Model(v) => v.iterate(ctx, opt, txn, ite).await?, Value::Thing(v) => v.iterate(ctx, opt, txn, ite).await?, Value::Table(v) => v.iterate(ctx, opt, txn, ite).await?, v => ite.process(ctx, opt, txn, None, v).await, } } } Ok(()) } } impl Model { pub async fn iterate( self, ctx: &Runtime, opt: &Options, txn: &Transaction, ite: &mut Iterator, ) -> Result<(), Error> { if ctx.is_ok() { if let Some(c) = self.count { for _ in 0..c { Thing { tb: self.table.to_string(), id: nanoid!(20, &ID_CHARS), } .iterate(ctx, opt, txn, ite) .await?; } } if let Some(r) = self.range { for x in r.0..r.1 { Thing { tb: self.table.to_string(), id: x.to_string(), } .iterate(ctx, opt, txn, ite) .await?; } } } Ok(()) } } impl Thing { pub async fn iterate( self, ctx: &Runtime, opt: &Options, txn: &Transaction, ite: &mut Iterator, ) -> Result<(), Error> { if ctx.is_ok() { let key = thing::new(opt.ns(), opt.db(), &self.tb, &self.id); let val = txn.clone().lock().await.get(key).await?; let val = match val { Some(v) => Value::from(v), None => Value::None, }; ite.process(ctx, opt, txn, Some(self), val).await; } Ok(()) } } impl Table { pub async fn iterate( self, ctx: &Runtime, opt: &Options, txn: &Transaction, ite: &mut Iterator, ) -> Result<(), Error> { if ctx.is_ok() { let beg = thing::new(opt.ns(), opt.db(), &self.name, key::PREFIX); let end = thing::new(opt.ns(), opt.db(), &self.name, key::SUFFIX); let mut nxt: Option> = None; loop { if ctx.is_ok() { let res = match nxt { None => { let min = beg.encode()?; let max = end.encode()?; txn.clone().lock().await.scan(min..max, 1000).await? } Some(ref mut beg) => { beg.push(0x00); let min = beg.clone(); let max = end.encode()?; txn.clone().lock().await.scan(min..max, 1000).await? } }; if !res.is_empty() { // Get total results let n = res.len() - 1; // Loop over results for (i, (k, v)) in res.into_iter().enumerate() { if ctx.is_ok() { // Ready the next if i == n { nxt = Some(k.clone()); } // Parse the key-value let k: crate::key::thing::Thing = (&k).into(); let v: crate::sql::value::Value = (&v).into(); let t = Thing::from((k.tb, k.id)); // Process the record ite.process(ctx, opt, txn, Some(t), v).await; } } continue; } } break; } } Ok(()) } }