|
@@ -3,6 +3,11 @@ use rand::Rng;
|
|
use std::collections::HashMap;
|
|
use std::collections::HashMap;
|
|
use std::fmt;
|
|
use std::fmt;
|
|
|
|
|
|
|
|
+macro_rules! bail {
|
|
|
|
+ ($fmt:expr) => { return Err(Error { message: format!($fmt), }) };
|
|
|
|
+ ($fmt:expr, $($e:expr),*) => { return Err(Error { message: format!($fmt, $($e),*), }) }
|
|
|
|
+}
|
|
|
|
+
|
|
#[derive(Debug)]
|
|
#[derive(Debug)]
|
|
pub enum Value {
|
|
pub enum Value {
|
|
Lit(Literal),
|
|
Lit(Literal),
|
|
@@ -10,6 +15,41 @@ pub enum Value {
|
|
Builtin(&'static BuiltinFunc),
|
|
Builtin(&'static BuiltinFunc),
|
|
}
|
|
}
|
|
|
|
|
|
|
|
+impl Value {
|
|
|
|
+ fn as_num(&self) -> Result<i64, Error> {
|
|
|
|
+ match self {
|
|
|
|
+ Value::Lit(Literal::Num(n)) => Ok(*n),
|
|
|
|
+ _ => self.with_str(|s| bail!("Expected number, got {}", s)),
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn with_str<U>(&self, f: impl Fn(&str) -> U) -> U {
|
|
|
|
+ match self {
|
|
|
|
+ Value::Lit(Literal::Str(s)) => f(&s),
|
|
|
|
+ Value::Lit(Literal::Atom(s)) => f(&s),
|
|
|
|
+ Value::Lit(Literal::Num(n)) => f(&format!("{}", n)),
|
|
|
|
+ Value::Tup(values) => {
|
|
|
|
+ let mut buf = String::new();
|
|
|
|
+ buf.push_str("<");
|
|
|
|
+ for (i, val) in values.iter().enumerate() {
|
|
|
|
+ if i > 0 {
|
|
|
|
+ buf.push_str(", ");
|
|
|
|
+ }
|
|
|
|
+ buf.push_str(&val.to_string());
|
|
|
|
+ }
|
|
|
|
+ buf.push_str(">");
|
|
|
|
+ f(&buf)
|
|
|
|
+ }
|
|
|
|
+ Value::Builtin(func) =>
|
|
|
|
+ f(&format!("#<builtin {}>", func.name)),
|
|
|
|
+ }
|
|
|
|
+ }
|
|
|
|
+
|
|
|
|
+ fn to_string(&self) -> String {
|
|
|
|
+ self.with_str(|s| s.to_string())
|
|
|
|
+ }
|
|
|
|
+}
|
|
|
|
+
|
|
pub struct BuiltinFunc {
|
|
pub struct BuiltinFunc {
|
|
name: &'static str,
|
|
name: &'static str,
|
|
callback: &'static dyn Fn(&mut State, &Expr) -> Result<Value, Error>,
|
|
callback: &'static dyn Fn(&mut State, &Expr) -> Result<Value, Error>,
|
|
@@ -28,11 +68,6 @@ impl fmt::Display for Error {
|
|
|
|
|
|
impl std::error::Error for Error {}
|
|
impl std::error::Error for Error {}
|
|
|
|
|
|
-macro_rules! bail {
|
|
|
|
- ($fmt:expr) => { return Err(Error { message: format!($fmt), }) };
|
|
|
|
- ($fmt:expr, $($e:expr),*) => { return Err(Error { message: format!($fmt, $($e),*), }) }
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
const BUILTINS: &[BuiltinFunc] = &[
|
|
const BUILTINS: &[BuiltinFunc] = &[
|
|
BuiltinFunc {
|
|
BuiltinFunc {
|
|
name: "rep",
|
|
name: "rep",
|
|
@@ -61,30 +96,6 @@ impl fmt::Debug for BuiltinFunc {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
|
|
-impl Value {
|
|
|
|
- fn to_string(&self) -> String {
|
|
|
|
- match self {
|
|
|
|
- Value::Lit(Literal::Str(s)) => s.clone(),
|
|
|
|
- Value::Lit(Literal::Atom(s)) => s.clone(),
|
|
|
|
- Value::Lit(Literal::Num(n)) => format!("{}", n),
|
|
|
|
- Value::Tup(values) => {
|
|
|
|
- let mut buf = String::new();
|
|
|
|
- buf.push_str("<");
|
|
|
|
- for (i, val) in values.iter().enumerate() {
|
|
|
|
- if i > 0 {
|
|
|
|
- buf.push_str(", ");
|
|
|
|
- }
|
|
|
|
- buf.push_str(&val.to_string());
|
|
|
|
- }
|
|
|
|
- buf.push_str(">");
|
|
|
|
- buf
|
|
|
|
- }
|
|
|
|
- Value::Builtin(func) =>
|
|
|
|
- format!("#<builtin {}>", func.name),
|
|
|
|
- }
|
|
|
|
- }
|
|
|
|
-}
|
|
|
|
-
|
|
|
|
enum NamedItem {
|
|
enum NamedItem {
|
|
Expr(Expr),
|
|
Expr(Expr),
|
|
Builtin(&'static BuiltinFunc),
|
|
Builtin(&'static BuiltinFunc),
|
|
@@ -172,14 +183,8 @@ impl State {
|
|
}
|
|
}
|
|
}
|
|
}
|
|
Expr::Range(from, to) => {
|
|
Expr::Range(from, to) => {
|
|
- let from = match self.eval(from)? {
|
|
|
|
- Value::Lit(Literal::Num(n)) => n,
|
|
|
|
- e => bail!("bad start in range: {}", e.to_string()),
|
|
|
|
- };
|
|
|
|
- let to = match self.eval(to)? {
|
|
|
|
- Value::Lit(Literal::Num(n)) => n,
|
|
|
|
- e => bail!("bad end in range: {}", e.to_string()),
|
|
|
|
- };
|
|
|
|
|
|
+ let from = self.eval(from)?.as_num()?;
|
|
|
|
+ let to = self.eval(to)?.as_num()?;
|
|
Ok(Value::Lit(Literal::Num(self.rand.gen_range(from..=to))))
|
|
Ok(Value::Lit(Literal::Num(self.rand.gen_range(from..=to))))
|
|
}
|
|
}
|
|
_ => bail!("unimplemented: {:?}", expr),
|
|
_ => bail!("unimplemented: {:?}", expr),
|