2023-03-13 15:02:19 -07:00
|
|
|
mod decl_lib_tests {
|
2024-02-06 22:39:08 +00:00
|
|
|
use flesh::ast::{eval, lex, Ctr, SymTable};
|
|
|
|
|
use flesh::stdlib::{dynamic_stdlib, static_stdlib};
|
2023-01-30 23:37:28 -08:00
|
|
|
|
|
|
|
|
#[test]
|
2023-02-28 11:57:59 -08:00
|
|
|
fn test_variable_def_and_lookup() {
|
2023-03-05 22:18:49 -08:00
|
|
|
let doc1 = "(def test 'my test var' 1)";
|
2023-03-17 11:42:36 -07:00
|
|
|
let doc2 = "test";
|
2023-02-28 11:20:00 -08:00
|
|
|
let result = "(1)";
|
2023-02-17 22:10:54 -08:00
|
|
|
|
2023-02-27 22:53:54 -08:00
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-01-30 23:37:28 -08:00
|
|
|
|
2023-03-17 11:42:36 -07:00
|
|
|
eval(&lex(&doc1.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
let res = *eval(&lex(&doc2.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
assert_eq!(res.to_string(), result);
|
|
|
|
|
}
|
2023-01-30 23:37:28 -08:00
|
|
|
|
2023-03-17 11:42:36 -07:00
|
|
|
#[test]
|
|
|
|
|
fn test_variable_def_and_lookup_list() {
|
|
|
|
|
let doc1 = "(def test 'my test var' (1))";
|
|
|
|
|
let doc2 = "test";
|
|
|
|
|
let result = "((1))";
|
|
|
|
|
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-17 11:42:36 -07:00
|
|
|
|
|
|
|
|
eval(&lex(&doc1.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
let res = *eval(&lex(&doc2.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
assert_eq!(res.to_string(), result);
|
2023-01-30 23:37:28 -08:00
|
|
|
}
|
2023-02-28 11:57:59 -08:00
|
|
|
|
2023-03-02 12:15:42 -08:00
|
|
|
#[test]
|
|
|
|
|
fn test_func_def_and_lookup() {
|
2023-03-05 22:18:49 -08:00
|
|
|
let doc1 = "(def test 'my test func' (hello) hello)";
|
2023-03-17 11:42:36 -07:00
|
|
|
let doc2 = "(test 1)";
|
2023-03-02 12:15:42 -08:00
|
|
|
let result = "1";
|
|
|
|
|
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-02 12:15:42 -08:00
|
|
|
|
2023-03-17 11:42:36 -07:00
|
|
|
eval(&lex(&doc1.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
let res = *eval(&lex(&doc2.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
assert_eq!(res.to_string(), result);
|
2023-03-02 12:15:42 -08:00
|
|
|
}
|
|
|
|
|
|
2023-02-28 11:57:59 -08:00
|
|
|
#[test]
|
|
|
|
|
fn test_variable_def_redef_and_lookup() {
|
2023-03-05 22:18:49 -08:00
|
|
|
let doc1 = "(def test 'my test var' 1)";
|
|
|
|
|
let doc2 = "(def test 'my test var' '2')";
|
2023-02-28 11:57:59 -08:00
|
|
|
let doc3 = "(test)";
|
|
|
|
|
let result = "('2')";
|
|
|
|
|
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-02-28 11:57:59 -08:00
|
|
|
|
2023-03-17 11:42:36 -07:00
|
|
|
eval(&lex(&doc1.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
eval(&lex(&doc2.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
let res = *eval(&lex(&doc3.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
assert_eq!(res.to_string(), result);
|
2023-02-28 11:57:59 -08:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_variable_def_undef_and_lookup_fail() {
|
2023-03-05 22:18:49 -08:00
|
|
|
let doc1 = "(def test 'my test var' 1)";
|
2023-02-28 11:57:59 -08:00
|
|
|
let doc2 = "(def test)";
|
|
|
|
|
let doc3 = "(test)";
|
|
|
|
|
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-02-28 11:57:59 -08:00
|
|
|
|
2023-03-17 11:42:36 -07:00
|
|
|
eval(&lex(&doc1.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
eval(&lex(&doc2.to_string()).unwrap(), &mut syms).unwrap();
|
2023-02-28 11:57:59 -08:00
|
|
|
|
2023-03-17 11:42:36 -07:00
|
|
|
let eval_result = eval(&lex(&doc3.to_string()).unwrap(), &mut syms);
|
|
|
|
|
if let Err(s) = eval_result {
|
|
|
|
|
assert_eq!(
|
2023-05-23 22:06:11 +00:00
|
|
|
s.0.first().unwrap().message,
|
|
|
|
|
"(is an undefined symbol)".to_string()
|
2023-03-17 11:42:36 -07:00
|
|
|
);
|
2023-02-28 11:57:59 -08:00
|
|
|
} else {
|
|
|
|
|
assert!(false);
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-03-05 22:18:49 -08:00
|
|
|
|
2023-03-17 12:21:42 -07:00
|
|
|
#[test]
|
|
|
|
|
fn test_variable_def_redef_via_reference_and_lookup() {
|
|
|
|
|
let doc1 = "(def test 'my test var' 1)";
|
|
|
|
|
let doc2 = "(def ref 'references test' (quote test))";
|
2023-06-08 17:11:10 -07:00
|
|
|
let doc3 = "(def ref 'my test var' '2')";
|
|
|
|
|
let test = "(test)";
|
|
|
|
|
let res1 = "(1)";
|
|
|
|
|
let doc4 = "(def (eval ref) 'my test var' '2')";
|
|
|
|
|
let res2 = "('2')";
|
2023-03-17 12:21:42 -07:00
|
|
|
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-17 12:21:42 -07:00
|
|
|
|
|
|
|
|
eval(&lex(&doc1.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
eval(&lex(&doc2.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
eval(&lex(&doc3.to_string()).unwrap(), &mut syms).unwrap();
|
2023-06-08 17:11:10 -07:00
|
|
|
let r1 = *eval(&lex(&test.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
assert_eq!(r1.to_string(), res1);
|
|
|
|
|
eval(&lex(&doc2.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
eval(&lex(&doc4.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
let r2 = *eval(&lex(&test.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
assert_eq!(r2.to_string(), res2);
|
2023-03-17 12:21:42 -07:00
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_variable_doc_dynamic() {
|
|
|
|
|
let doc1 = "(def test-doc 'docs for test' 'test tests tests test')";
|
|
|
|
|
let doc2 = "(def test test-doc 'one')";
|
|
|
|
|
let doc3 = "(eq? (and
|
2023-03-17 13:06:27 -07:00
|
|
|
(eq? (get-doc (quote test)) test-doc)
|
2023-03-17 12:21:42 -07:00
|
|
|
(eq? test 'one')))";
|
|
|
|
|
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-17 12:21:42 -07:00
|
|
|
|
|
|
|
|
eval(&lex(&doc1.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
eval(&lex(&doc2.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
if let Ctr::Bool(b) = *eval(&lex(&doc3.to_string()).unwrap(), &mut syms).unwrap() {
|
|
|
|
|
assert!(b);
|
|
|
|
|
} else {
|
|
|
|
|
assert!(false);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-05 22:18:49 -08:00
|
|
|
#[test]
|
|
|
|
|
fn test_func_def_no_args() {
|
2023-03-17 11:42:36 -07:00
|
|
|
let doc1 = "(def test 'my test func' () 1)";
|
2023-03-05 22:18:49 -08:00
|
|
|
let doc2 = "(test)";
|
|
|
|
|
let result = "1";
|
|
|
|
|
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-05 22:18:49 -08:00
|
|
|
|
2023-03-17 11:42:36 -07:00
|
|
|
eval(&lex(&doc1.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
let res = *eval(&lex(&doc2.to_string()).unwrap(), &mut syms).unwrap();
|
|
|
|
|
assert_eq!(res.to_string(), result);
|
2023-03-05 22:18:49 -08:00
|
|
|
}
|
2023-03-06 15:25:23 -08:00
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_isset_true() {
|
|
|
|
|
let doc1 = "(def test '' 1)";
|
|
|
|
|
let doc2 = "(set? test)";
|
2023-03-17 11:42:36 -07:00
|
|
|
|
2023-03-06 15:25:23 -08:00
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-17 11:42:36 -07:00
|
|
|
|
2023-03-06 15:25:23 -08:00
|
|
|
let def_tree = lex(&doc1.to_string()).unwrap();
|
|
|
|
|
let set_tree = lex(&doc2.to_string()).unwrap();
|
|
|
|
|
eval(&def_tree, &mut syms).unwrap();
|
|
|
|
|
if let Ctr::Bool(b) = *eval(&set_tree, &mut syms).unwrap() {
|
|
|
|
|
assert!(b);
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_isset_false() {
|
|
|
|
|
let doc = "(set? test)";
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-06 15:25:23 -08:00
|
|
|
let set_tree = lex(&doc.to_string()).unwrap();
|
|
|
|
|
if let Ctr::Bool(b) = *eval(&set_tree, &mut syms).unwrap() {
|
|
|
|
|
assert!(!b);
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-03-06 15:50:02 -08:00
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_env_doesnt_lose_elements() {
|
|
|
|
|
let doc1 = "(def t '' 1)";
|
|
|
|
|
let doc2 = "(env)";
|
|
|
|
|
let doc3 = "t";
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-06 15:50:02 -08:00
|
|
|
let set_tree = lex(&doc1.to_string()).unwrap();
|
|
|
|
|
let env_tree = lex(&doc2.to_string()).unwrap();
|
|
|
|
|
let tst_tree = lex(&doc3.to_string()).unwrap();
|
|
|
|
|
eval(&set_tree, &mut syms).unwrap();
|
|
|
|
|
eval(&env_tree, &mut syms).unwrap();
|
|
|
|
|
eval(&tst_tree, &mut syms).unwrap();
|
|
|
|
|
}
|
2023-03-11 22:04:46 -08:00
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_quote() {
|
|
|
|
|
let document = "(quote (add 1 2))";
|
|
|
|
|
let result = "(add 1 2)";
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-11 22:04:46 -08:00
|
|
|
assert_eq!(
|
|
|
|
|
*eval(&lex(&document.to_string()).unwrap(), &mut syms)
|
|
|
|
|
.unwrap()
|
|
|
|
|
.to_string(),
|
|
|
|
|
result.to_string(),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_eval() {
|
|
|
|
|
let document = "
|
|
|
|
|
(let ((stored-tree (quote (add 1 2))))
|
|
|
|
|
(eval stored-tree)))";
|
|
|
|
|
let result = "3";
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-11 22:04:46 -08:00
|
|
|
assert_eq!(
|
|
|
|
|
*eval(&lex(&document.to_string()).unwrap(), &mut syms)
|
|
|
|
|
.unwrap()
|
|
|
|
|
.to_string(),
|
|
|
|
|
result.to_string(),
|
|
|
|
|
);
|
|
|
|
|
}
|
2023-03-13 15:02:19 -07:00
|
|
|
|
2023-03-17 11:42:36 -07:00
|
|
|
#[test]
|
|
|
|
|
fn test_eval_basic() {
|
|
|
|
|
let document = "(eval (1 2 3))";
|
|
|
|
|
let result = "(1 2 3)";
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-17 11:42:36 -07:00
|
|
|
assert_eq!(
|
|
|
|
|
*eval(&lex(&document.to_string()).unwrap(), &mut syms)
|
|
|
|
|
.unwrap()
|
|
|
|
|
.to_string(),
|
|
|
|
|
result.to_string(),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-13 15:02:19 -07:00
|
|
|
#[test]
|
|
|
|
|
fn test_lambda_str_equivalency_list() {
|
|
|
|
|
let document = "(lambda (x y) (add x y))";
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-13 15:02:19 -07:00
|
|
|
assert_eq!(
|
|
|
|
|
*eval(&lex(&document.to_string()).unwrap(), &mut syms)
|
|
|
|
|
.unwrap()
|
|
|
|
|
.to_string(),
|
|
|
|
|
document.to_string(),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_lambda_str_equivalency_no_args() {
|
|
|
|
|
let document = "(lambda () (add 1 2))";
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-13 15:02:19 -07:00
|
|
|
assert_eq!(
|
|
|
|
|
*eval(&lex(&document.to_string()).unwrap(), &mut syms)
|
|
|
|
|
.unwrap()
|
|
|
|
|
.to_string(),
|
|
|
|
|
document.to_string(),
|
|
|
|
|
);
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_lambda_inline_call() {
|
|
|
|
|
let document = "((lambda (x y) (add x y)) 1 2)";
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-13 15:02:19 -07:00
|
|
|
let it = *eval(
|
|
|
|
|
&lex(&document.to_string()).unwrap(),
|
|
|
|
|
&mut syms).unwrap();
|
|
|
|
|
if let Ctr::Integer(i) = it {
|
|
|
|
|
assert_eq!(i, 3)
|
|
|
|
|
} else {
|
|
|
|
|
panic!()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_lambda_let_call() {
|
|
|
|
|
let document = "(let ((adder (lambda (x y) (add x y))))
|
|
|
|
|
(adder 1 2))";
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-13 15:02:19 -07:00
|
|
|
let it = *eval(
|
|
|
|
|
&lex(&document.to_string()).unwrap(),
|
|
|
|
|
&mut syms).unwrap();
|
|
|
|
|
if let Ctr::Integer(i) = it {
|
|
|
|
|
assert_eq!(i, 3)
|
|
|
|
|
} else {
|
|
|
|
|
panic!()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
|
fn test_lambda_var_bound_call() {
|
|
|
|
|
let document = "(let (())
|
|
|
|
|
(def adder 'my adder' (lambda (x y) (add x y)))
|
|
|
|
|
(adder 1 2))";
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-13 15:02:19 -07:00
|
|
|
let it = *eval(
|
|
|
|
|
&lex(&document.to_string()).unwrap(),
|
|
|
|
|
&mut syms).unwrap();
|
|
|
|
|
if let Ctr::Integer(i) = it {
|
|
|
|
|
assert_eq!(i, 3)
|
|
|
|
|
} else {
|
|
|
|
|
panic!()
|
|
|
|
|
}
|
|
|
|
|
}
|
2023-03-17 11:42:36 -07:00
|
|
|
|
2023-03-20 16:22:51 -07:00
|
|
|
#[test]
|
|
|
|
|
fn test_lambda_arg_call() {
|
|
|
|
|
let document = "(let (())
|
|
|
|
|
(def appl '' (func item) (func item))
|
|
|
|
|
(def adder 'my adder' (lambda (x) (add x 1)))
|
|
|
|
|
(appl adder 2))";
|
|
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-20 16:22:51 -07:00
|
|
|
let it = *eval(
|
|
|
|
|
&lex(&document.to_string()).unwrap(),
|
|
|
|
|
&mut syms).unwrap();
|
|
|
|
|
if let Ctr::Integer(i) = it {
|
|
|
|
|
assert_eq!(i, 3)
|
|
|
|
|
} else {
|
|
|
|
|
println!("bad result: {}", it);
|
|
|
|
|
panic!()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-03-17 11:42:36 -07:00
|
|
|
#[test]
|
|
|
|
|
fn test_setget_doc_string() {
|
2023-03-17 13:06:27 -07:00
|
|
|
let highly_inadvisable = "(set-doc (q help) 'help')";
|
|
|
|
|
let document = "(get-doc (q help))";
|
2023-03-17 11:42:36 -07:00
|
|
|
let mut syms = SymTable::new();
|
2023-05-26 06:41:18 +00:00
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
2023-03-17 11:42:36 -07:00
|
|
|
let _ = *eval(
|
|
|
|
|
&lex(&highly_inadvisable.to_string()).unwrap(),
|
|
|
|
|
&mut syms).unwrap();
|
|
|
|
|
let it = *eval(
|
|
|
|
|
&lex(&document.to_string()).unwrap(),
|
|
|
|
|
&mut syms).unwrap();
|
|
|
|
|
if let Ctr::String(i) = it {
|
|
|
|
|
assert_eq!(i, "help".to_string())
|
|
|
|
|
} else {
|
|
|
|
|
panic!()
|
|
|
|
|
}
|
|
|
|
|
}
|
|
|
|
|
|
2023-06-08 17:11:10 -07:00
|
|
|
#[test]
|
|
|
|
|
fn test_eval_quote() {
|
|
|
|
|
let doc = "(eval (quote (add 1 1)))";
|
|
|
|
|
let mut syms = SymTable::new();
|
|
|
|
|
static_stdlib(&mut syms);
|
|
|
|
|
dynamic_stdlib(&mut syms, None);
|
|
|
|
|
assert_eq!(
|
|
|
|
|
*eval(&lex(&doc.to_string()).unwrap(), &mut syms).unwrap().to_string(),
|
|
|
|
|
2.to_string()
|
|
|
|
|
)
|
|
|
|
|
}
|
2023-01-30 23:37:28 -08:00
|
|
|
}
|