2020-06-29 15:36:01 +00:00
|
|
|
use crate::sql::common::take_u64;
|
2022-01-16 20:31:50 +00:00
|
|
|
use crate::sql::error::IResult;
|
2022-05-15 08:34:29 +00:00
|
|
|
use crate::sql::escape::escape_ident;
|
2022-05-30 15:05:05 +00:00
|
|
|
use crate::sql::id::Id;
|
2020-06-29 15:36:01 +00:00
|
|
|
use crate::sql::ident::ident_raw;
|
2022-05-30 15:05:05 +00:00
|
|
|
use crate::sql::thing::Thing;
|
2020-06-29 15:36:01 +00:00
|
|
|
use nom::branch::alt;
|
2022-03-16 23:52:25 +00:00
|
|
|
use nom::character::complete::char;
|
2020-06-29 15:36:01 +00:00
|
|
|
use serde::{Deserialize, Serialize};
|
|
|
|
use std::fmt;
|
|
|
|
|
2022-05-30 15:05:05 +00:00
|
|
|
pub struct IntoIter {
|
|
|
|
model: Model,
|
|
|
|
index: u64,
|
|
|
|
}
|
|
|
|
|
|
|
|
impl Iterator for IntoIter {
|
|
|
|
type Item = Thing;
|
|
|
|
fn next(&mut self) -> Option<Thing> {
|
|
|
|
match &self.model {
|
|
|
|
Model::Count(tb, c) => {
|
|
|
|
if self.index < *c {
|
|
|
|
self.index += 1;
|
|
|
|
Some(Thing {
|
|
|
|
tb: tb.to_string(),
|
|
|
|
id: Id::rand(),
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
Model::Range(tb, b, e) => {
|
|
|
|
if self.index + b <= *e {
|
|
|
|
self.index += 1;
|
|
|
|
Some(Thing {
|
|
|
|
tb: tb.to_string(),
|
|
|
|
id: Id::from(self.index),
|
|
|
|
})
|
|
|
|
} else {
|
|
|
|
None
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2022-05-06 16:02:44 +00:00
|
|
|
#[derive(Clone, Debug, Eq, PartialEq, PartialOrd, Serialize, Deserialize)]
|
|
|
|
pub enum Model {
|
|
|
|
Count(String, u64),
|
|
|
|
Range(String, u64, u64),
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
|
2022-05-30 15:05:05 +00:00
|
|
|
impl IntoIterator for Model {
|
|
|
|
type Item = Thing;
|
|
|
|
type IntoIter = IntoIter;
|
|
|
|
fn into_iter(self) -> Self::IntoIter {
|
|
|
|
IntoIter {
|
|
|
|
model: self,
|
|
|
|
index: 0,
|
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
2020-06-29 15:36:01 +00:00
|
|
|
impl fmt::Display for Model {
|
|
|
|
fn fmt(&self, f: &mut fmt::Formatter) -> fmt::Result {
|
2022-05-06 16:02:44 +00:00
|
|
|
match self {
|
|
|
|
Model::Count(tb, c) => {
|
2022-05-15 08:34:29 +00:00
|
|
|
write!(f, "|{}:{}|", escape_ident(tb), c)
|
2022-05-06 16:02:44 +00:00
|
|
|
}
|
|
|
|
Model::Range(tb, b, e) => {
|
2022-05-15 08:34:29 +00:00
|
|
|
write!(f, "|{}:{}..{}|", escape_ident(tb), b, e)
|
2022-05-06 16:02:44 +00:00
|
|
|
}
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
}
|
|
|
|
}
|
|
|
|
|
|
|
|
pub fn model(i: &str) -> IResult<&str, Model> {
|
|
|
|
alt((model_count, model_range))(i)
|
|
|
|
}
|
|
|
|
|
|
|
|
fn model_count(i: &str) -> IResult<&str, Model> {
|
2022-03-16 23:52:25 +00:00
|
|
|
let (i, _) = char('|')(i)?;
|
2020-06-29 15:36:01 +00:00
|
|
|
let (i, t) = ident_raw(i)?;
|
2022-03-16 23:52:25 +00:00
|
|
|
let (i, _) = char(':')(i)?;
|
2020-06-29 15:36:01 +00:00
|
|
|
let (i, c) = take_u64(i)?;
|
2022-03-16 23:52:25 +00:00
|
|
|
let (i, _) = char('|')(i)?;
|
2022-05-06 16:02:44 +00:00
|
|
|
Ok((i, Model::Count(t, c)))
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
fn model_range(i: &str) -> IResult<&str, Model> {
|
2022-03-16 23:52:25 +00:00
|
|
|
let (i, _) = char('|')(i)?;
|
2020-06-29 15:36:01 +00:00
|
|
|
let (i, t) = ident_raw(i)?;
|
2022-03-16 23:52:25 +00:00
|
|
|
let (i, _) = char(':')(i)?;
|
2020-06-29 15:36:01 +00:00
|
|
|
let (i, b) = take_u64(i)?;
|
2022-03-16 23:52:25 +00:00
|
|
|
let (i, _) = char('.')(i)?;
|
|
|
|
let (i, _) = char('.')(i)?;
|
2020-06-29 15:36:01 +00:00
|
|
|
let (i, e) = take_u64(i)?;
|
2022-03-16 23:52:25 +00:00
|
|
|
let (i, _) = char('|')(i)?;
|
2022-05-06 16:02:44 +00:00
|
|
|
Ok((i, Model::Range(t, b, e)))
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[cfg(test)]
|
|
|
|
mod tests {
|
|
|
|
|
|
|
|
use super::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn model_count() {
|
|
|
|
let sql = "|test:1000|";
|
|
|
|
let res = model(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("|test:1000|", format!("{}", out));
|
2022-05-06 16:02:44 +00:00
|
|
|
assert_eq!(out, Model::Count(String::from("test"), 1000));
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn model_range() {
|
|
|
|
let sql = "|test:1..1000|";
|
|
|
|
let res = model(sql);
|
|
|
|
assert!(res.is_ok());
|
|
|
|
let out = res.unwrap().1;
|
|
|
|
assert_eq!("|test:1..1000|", format!("{}", out));
|
2022-05-06 16:02:44 +00:00
|
|
|
assert_eq!(out, Model::Range(String::from("test"), 1, 1000));
|
2020-06-29 15:36:01 +00:00
|
|
|
}
|
|
|
|
}
|