2024-10-29 23:11:55 -04:00
|
|
|
use dust_lang::*;
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn allow_access_to_parent_scope() {
|
|
|
|
let source = r#"
|
|
|
|
let x = 1;
|
|
|
|
{
|
|
|
|
x
|
|
|
|
}
|
|
|
|
"#;
|
|
|
|
|
2024-11-27 19:43:50 -05:00
|
|
|
assert_eq!(run(source), Ok(Some(ConcreteValue::Integer(1))));
|
2024-10-29 23:11:55 -04:00
|
|
|
}
|
|
|
|
|
2024-11-04 08:08:56 -05:00
|
|
|
#[test]
|
|
|
|
fn block_scope() {
|
|
|
|
let source = "
|
|
|
|
let a = 0;
|
|
|
|
{
|
|
|
|
let b = 42;
|
|
|
|
{
|
|
|
|
let c = 1;
|
|
|
|
}
|
|
|
|
let d = 2;
|
|
|
|
}
|
|
|
|
let e = 1;
|
|
|
|
";
|
|
|
|
|
|
|
|
assert_eq!(
|
2024-11-06 15:40:37 -05:00
|
|
|
compile(source),
|
2024-11-04 08:08:56 -05:00
|
|
|
Ok(Chunk::with_data(
|
|
|
|
None,
|
2024-11-16 05:16:51 -05:00
|
|
|
FunctionType {
|
|
|
|
type_parameters: None,
|
|
|
|
value_parameters: None,
|
2024-12-10 03:34:41 -05:00
|
|
|
return_type: Type::None,
|
2024-11-16 05:16:51 -05:00
|
|
|
},
|
2024-11-04 08:08:56 -05:00
|
|
|
vec![
|
2024-12-10 03:34:41 -05:00
|
|
|
(Instruction::load_constant(0, 0, false), Span(17, 18)),
|
|
|
|
(Instruction::load_constant(1, 2, false), Span(50, 52)),
|
|
|
|
(Instruction::load_constant(2, 4, false), Span(92, 93)),
|
|
|
|
(Instruction::load_constant(3, 6, false), Span(129, 130)),
|
|
|
|
(Instruction::load_constant(4, 4, false), Span(158, 159)),
|
2024-12-02 05:59:01 -05:00
|
|
|
(Instruction::r#return(false), Span(165, 165))
|
2024-11-04 08:08:56 -05:00
|
|
|
],
|
|
|
|
vec![
|
2024-11-16 01:29:21 -05:00
|
|
|
ConcreteValue::Integer(0),
|
|
|
|
ConcreteValue::string("a"),
|
|
|
|
ConcreteValue::Integer(42),
|
|
|
|
ConcreteValue::string("b"),
|
|
|
|
ConcreteValue::Integer(1),
|
|
|
|
ConcreteValue::string("c"),
|
|
|
|
ConcreteValue::Integer(2),
|
|
|
|
ConcreteValue::string("d"),
|
|
|
|
ConcreteValue::string("e"),
|
2024-11-04 08:08:56 -05:00
|
|
|
],
|
|
|
|
vec![
|
2024-12-10 03:34:41 -05:00
|
|
|
Local::new(1, 0, false, Scope::new(0, 0)),
|
|
|
|
Local::new(3, 2, false, Scope::new(1, 1)),
|
|
|
|
Local::new(5, 4, false, Scope::new(2, 2)),
|
|
|
|
Local::new(7, 6, false, Scope::new(1, 1)),
|
|
|
|
Local::new(8, 7, false, Scope::new(0, 0)),
|
2024-11-04 08:08:56 -05:00
|
|
|
]
|
|
|
|
)),
|
|
|
|
);
|
|
|
|
|
2024-11-27 19:43:50 -05:00
|
|
|
assert_eq!(run(source), Ok(None));
|
2024-11-04 08:08:56 -05:00
|
|
|
}
|
|
|
|
|
2024-11-05 12:44:16 -05:00
|
|
|
#[test]
|
|
|
|
fn multiple_block_scopes() {
|
|
|
|
let source = "
|
|
|
|
let a = 0;
|
|
|
|
{
|
|
|
|
let b = 42;
|
|
|
|
{
|
|
|
|
let c = 1;
|
|
|
|
}
|
2024-11-17 20:32:53 -05:00
|
|
|
let d = b;
|
2024-11-05 12:44:16 -05:00
|
|
|
}
|
2024-11-17 20:32:53 -05:00
|
|
|
let q = a;
|
2024-11-05 12:44:16 -05:00
|
|
|
{
|
|
|
|
let b = 42;
|
|
|
|
{
|
|
|
|
let c = 1;
|
|
|
|
}
|
2024-11-17 20:32:53 -05:00
|
|
|
let d = b;
|
2024-11-05 12:44:16 -05:00
|
|
|
}
|
2024-11-17 20:32:53 -05:00
|
|
|
let e = a;
|
2024-11-05 12:44:16 -05:00
|
|
|
";
|
|
|
|
|
|
|
|
assert_eq!(
|
2024-11-06 15:40:37 -05:00
|
|
|
compile(source),
|
2024-11-05 12:44:16 -05:00
|
|
|
Ok(Chunk::with_data(
|
|
|
|
None,
|
2024-11-16 05:16:51 -05:00
|
|
|
FunctionType {
|
|
|
|
type_parameters: None,
|
|
|
|
value_parameters: None,
|
2024-12-10 03:34:41 -05:00
|
|
|
return_type: Type::None,
|
2024-11-16 05:16:51 -05:00
|
|
|
},
|
2024-11-05 12:44:16 -05:00
|
|
|
vec![
|
2024-12-10 03:34:41 -05:00
|
|
|
(Instruction::load_constant(0, 0, false), Span(17, 18)),
|
|
|
|
(Instruction::load_constant(1, 2, false), Span(50, 52)),
|
|
|
|
(Instruction::load_constant(2, 4, false), Span(92, 93)),
|
|
|
|
(Instruction::get_local(3, 1), Span(129, 130)),
|
|
|
|
(Instruction::get_local(4, 0), Span(158, 159)),
|
|
|
|
(Instruction::load_constant(5, 2, false), Span(191, 193)),
|
|
|
|
(Instruction::load_constant(4, 4, false), Span(233, 234)),
|
|
|
|
(Instruction::get_local(7, 5), Span(270, 271)),
|
|
|
|
(Instruction::get_local(8, 0), Span(299, 300)),
|
2024-12-02 05:59:01 -05:00
|
|
|
(Instruction::r#return(false), Span(306, 306))
|
2024-11-05 12:44:16 -05:00
|
|
|
],
|
|
|
|
vec![
|
2024-11-16 01:29:21 -05:00
|
|
|
ConcreteValue::Integer(0),
|
|
|
|
ConcreteValue::string("a"),
|
|
|
|
ConcreteValue::Integer(42),
|
|
|
|
ConcreteValue::string("b"),
|
|
|
|
ConcreteValue::Integer(1),
|
|
|
|
ConcreteValue::string("c"),
|
|
|
|
ConcreteValue::string("d"),
|
|
|
|
ConcreteValue::string("q"),
|
|
|
|
ConcreteValue::string("e"),
|
2024-11-05 12:44:16 -05:00
|
|
|
],
|
|
|
|
vec![
|
2024-12-10 03:34:41 -05:00
|
|
|
Local::new(1, 0, false, Scope::new(0, 0)),
|
|
|
|
Local::new(3, 2, false, Scope::new(1, 1)),
|
|
|
|
Local::new(5, 4, false, Scope::new(2, 2)),
|
|
|
|
Local::new(6, 5, false, Scope::new(1, 1)),
|
|
|
|
Local::new(7, 6, false, Scope::new(0, 0)),
|
|
|
|
Local::new(3, 1, false, Scope::new(1, 3)),
|
|
|
|
Local::new(5, 1, false, Scope::new(2, 4)),
|
|
|
|
Local::new(6, 1, false, Scope::new(1, 3)),
|
|
|
|
Local::new(8, 1, false, Scope::new(0, 0)),
|
2024-11-05 12:44:16 -05:00
|
|
|
]
|
|
|
|
)),
|
|
|
|
);
|
|
|
|
|
2024-11-27 19:43:50 -05:00
|
|
|
assert_eq!(run(source), Ok(None));
|
2024-11-05 12:44:16 -05:00
|
|
|
}
|
2024-11-05 16:07:51 -05:00
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn disallow_access_to_child_scope() {
|
|
|
|
let source = r#"
|
|
|
|
{
|
|
|
|
let x = 1;
|
|
|
|
}
|
|
|
|
x
|
|
|
|
"#;
|
|
|
|
|
|
|
|
assert_eq!(
|
2024-11-27 19:43:50 -05:00
|
|
|
run(source),
|
2024-12-10 03:34:41 -05:00
|
|
|
Err(DustError::Compile {
|
2024-11-06 15:40:37 -05:00
|
|
|
error: CompileError::VariableOutOfScope {
|
2024-11-05 16:07:51 -05:00
|
|
|
identifier: "x".to_string(),
|
|
|
|
position: Span(52, 53),
|
|
|
|
variable_scope: Scope::new(1, 1),
|
|
|
|
access_scope: Scope::new(0, 0),
|
|
|
|
},
|
|
|
|
source
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn disallow_access_to_child_scope_nested() {
|
|
|
|
let source = r#"
|
|
|
|
{
|
|
|
|
{
|
|
|
|
let x = 1;
|
|
|
|
}
|
|
|
|
x
|
|
|
|
}
|
|
|
|
"#;
|
|
|
|
|
|
|
|
assert_eq!(
|
2024-11-27 19:43:50 -05:00
|
|
|
run(source),
|
2024-12-10 03:34:41 -05:00
|
|
|
Err(DustError::Compile {
|
2024-11-06 15:40:37 -05:00
|
|
|
error: CompileError::VariableOutOfScope {
|
2024-11-05 16:07:51 -05:00
|
|
|
identifier: "x".to_string(),
|
|
|
|
position: Span(78, 79),
|
|
|
|
variable_scope: Scope::new(2, 2),
|
|
|
|
access_scope: Scope::new(1, 1),
|
|
|
|
},
|
|
|
|
source
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn disallow_access_to_sibling_scope() {
|
|
|
|
let source = r#"
|
|
|
|
{
|
|
|
|
let x = 1;
|
|
|
|
}
|
|
|
|
{
|
|
|
|
x
|
|
|
|
}
|
|
|
|
"#;
|
|
|
|
|
|
|
|
assert_eq!(
|
2024-11-27 19:43:50 -05:00
|
|
|
run(source),
|
2024-12-10 03:34:41 -05:00
|
|
|
Err(DustError::Compile {
|
2024-11-06 15:40:37 -05:00
|
|
|
error: CompileError::VariableOutOfScope {
|
2024-11-05 16:07:51 -05:00
|
|
|
identifier: "x".to_string(),
|
|
|
|
variable_scope: Scope::new(1, 1),
|
|
|
|
access_scope: Scope::new(1, 2),
|
|
|
|
position: Span(66, 67),
|
|
|
|
},
|
|
|
|
source
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|
|
|
|
|
|
|
|
#[test]
|
|
|
|
fn disallow_access_to_sibling_scope_nested() {
|
|
|
|
let source = r#"
|
|
|
|
{
|
|
|
|
{
|
|
|
|
let x = 1;
|
|
|
|
}
|
|
|
|
{
|
|
|
|
x
|
|
|
|
}
|
|
|
|
}
|
|
|
|
"#;
|
|
|
|
|
|
|
|
assert_eq!(
|
2024-11-27 19:43:50 -05:00
|
|
|
run(source),
|
2024-12-10 03:34:41 -05:00
|
|
|
Err(DustError::Compile {
|
2024-11-06 15:40:37 -05:00
|
|
|
error: CompileError::VariableOutOfScope {
|
2024-11-05 16:07:51 -05:00
|
|
|
identifier: "x".to_string(),
|
|
|
|
variable_scope: Scope::new(2, 2),
|
|
|
|
access_scope: Scope::new(2, 3),
|
|
|
|
position: Span(96, 97),
|
|
|
|
},
|
|
|
|
source
|
|
|
|
})
|
|
|
|
);
|
|
|
|
}
|