parse path template and generate URL assembly code
This commit is contained in:
parent
445b96056e
commit
25ed132e2e
|
@ -0,0 +1,143 @@
|
||||||
|
use anyhow::{Result, bail};
|
||||||
|
|
||||||
|
#[derive(Eq, PartialEq, Clone, Debug)]
|
||||||
|
enum Component {
|
||||||
|
Constant(String),
|
||||||
|
Parameter(String),
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Eq, PartialEq, Clone, Debug)]
|
||||||
|
pub struct Template {
|
||||||
|
components: Vec<Component>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Template {
|
||||||
|
pub fn compile(&self) -> String {
|
||||||
|
let mut out = " let url = format!(\"{}".to_string();
|
||||||
|
for c in self.components.iter() {
|
||||||
|
out.push('/');
|
||||||
|
match c {
|
||||||
|
Component::Constant(n) => out.push_str(n),
|
||||||
|
Component::Parameter(_) => out.push_str("{}"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
out.push_str("\",\n");
|
||||||
|
out.push_str(" self.baseurl,\n");
|
||||||
|
for c in self.components.iter() {
|
||||||
|
if let Component::Parameter(n) = &c {
|
||||||
|
out.push_str(&format!(" {},\n", n));
|
||||||
|
}
|
||||||
|
}
|
||||||
|
out.push_str(" );\n");
|
||||||
|
out
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn parse(t: &str) -> Result<Template> {
|
||||||
|
enum State {
|
||||||
|
Start,
|
||||||
|
ConstantOrParameter,
|
||||||
|
Parameter,
|
||||||
|
Constant,
|
||||||
|
}
|
||||||
|
|
||||||
|
let mut s = State::Start;
|
||||||
|
let mut a = String::new();
|
||||||
|
let mut components = Vec::new();
|
||||||
|
|
||||||
|
for c in t.chars() {
|
||||||
|
match s {
|
||||||
|
State::Start => {
|
||||||
|
if c == '/' {
|
||||||
|
s = State::ConstantOrParameter;
|
||||||
|
} else {
|
||||||
|
bail!("path must start with a slash");
|
||||||
|
}
|
||||||
|
}
|
||||||
|
State::ConstantOrParameter => {
|
||||||
|
if c == '/' || c == '}' {
|
||||||
|
bail!("expected a constant or parameter");
|
||||||
|
} else if c == '{' {
|
||||||
|
s = State::Parameter;
|
||||||
|
} else {
|
||||||
|
s = State::Constant;
|
||||||
|
a.push(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
State::Constant => {
|
||||||
|
if c == '/' {
|
||||||
|
components.push(Component::Constant(a));
|
||||||
|
a = String::new();
|
||||||
|
s = State::ConstantOrParameter;
|
||||||
|
} else if c == '{' || c == '}' {
|
||||||
|
bail!("unexpected parameter");
|
||||||
|
} else {
|
||||||
|
a.push(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
State::Parameter => {
|
||||||
|
if c == '}' {
|
||||||
|
components.push(Component::Parameter(a));
|
||||||
|
a = String::new();
|
||||||
|
s = State::ConstantOrParameter;
|
||||||
|
} else if c == '/' || c == '{' {
|
||||||
|
bail!("expected parameter");
|
||||||
|
} else {
|
||||||
|
a.push(c);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
match s {
|
||||||
|
State::Start => bail!("empty path"),
|
||||||
|
State::ConstantOrParameter => (),
|
||||||
|
State::Constant => components.push(Component::Constant(a)),
|
||||||
|
State::Parameter => bail!("unterminated parameter"),
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(Template { components, })
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test {
|
||||||
|
use anyhow::{anyhow, Context, Result};
|
||||||
|
use super::{parse, Template, Component};
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn basic() -> Result<()> {
|
||||||
|
let trials = vec![
|
||||||
|
("/info", Template {
|
||||||
|
components: vec![
|
||||||
|
Component::Constant("info".into()),
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
("/measure/{number}", Template {
|
||||||
|
components: vec![
|
||||||
|
Component::Constant("measure".into()),
|
||||||
|
Component::Parameter("number".into()),
|
||||||
|
],
|
||||||
|
}),
|
||||||
|
];
|
||||||
|
|
||||||
|
for (path, want) in trials.iter() {
|
||||||
|
let t = parse(path)
|
||||||
|
.with_context(|| anyhow!("path {}", path))?;
|
||||||
|
assert_eq!(&t, want);
|
||||||
|
}
|
||||||
|
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn compile() -> Result<()> {
|
||||||
|
let t = parse("/measure/{number}")?;
|
||||||
|
let out = t.compile();
|
||||||
|
let want = " let url = format!(\"{}/measure/{}\",\
|
||||||
|
\n self.baseurl,\
|
||||||
|
\n number,\
|
||||||
|
\n );\n";
|
||||||
|
assert_eq!(want, &out);
|
||||||
|
Ok(())
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue