prepare transpiling variables
This commit is contained in:
parent
38f90e6491
commit
a07f16f283
|
@ -19,6 +19,8 @@ mod transpiler;
|
||||||
#[cfg(feature = "shulkerbox")]
|
#[cfg(feature = "shulkerbox")]
|
||||||
#[cfg_attr(feature = "shulkerbox", doc(inline))]
|
#[cfg_attr(feature = "shulkerbox", doc(inline))]
|
||||||
pub use transpiler::Transpiler;
|
pub use transpiler::Transpiler;
|
||||||
|
#[cfg(feature = "shulkerbox")]
|
||||||
|
mod variables;
|
||||||
|
|
||||||
pub mod util;
|
pub mod util;
|
||||||
|
|
||||||
|
|
|
@ -29,6 +29,7 @@ use crate::{
|
||||||
|
|
||||||
use super::{
|
use super::{
|
||||||
error::{TranspileError, TranspileResult},
|
error::{TranspileError, TranspileResult},
|
||||||
|
variables::{Scope, VariableType},
|
||||||
FunctionData,
|
FunctionData,
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -74,8 +75,10 @@ impl Transpiler {
|
||||||
) -> Result<(), TranspileError> {
|
) -> Result<(), TranspileError> {
|
||||||
tracing::trace!("Transpiling program declarations");
|
tracing::trace!("Transpiling program declarations");
|
||||||
|
|
||||||
|
let scope = Scope::new();
|
||||||
|
|
||||||
for program in programs {
|
for program in programs {
|
||||||
self.transpile_program_declarations(program, handler);
|
self.transpile_program_declarations(program, &scope, handler);
|
||||||
}
|
}
|
||||||
|
|
||||||
let mut always_transpile_functions = Vec::new();
|
let mut always_transpile_functions = Vec::new();
|
||||||
|
@ -98,7 +101,7 @@ impl Transpiler {
|
||||||
);
|
);
|
||||||
|
|
||||||
for identifier_span in always_transpile_functions {
|
for identifier_span in always_transpile_functions {
|
||||||
self.get_or_transpile_function(&identifier_span, None, handler)?;
|
self.get_or_transpile_function(&identifier_span, None, &scope, handler)?;
|
||||||
}
|
}
|
||||||
|
|
||||||
Ok(())
|
Ok(())
|
||||||
|
@ -108,12 +111,13 @@ impl Transpiler {
|
||||||
fn transpile_program_declarations(
|
fn transpile_program_declarations(
|
||||||
&mut self,
|
&mut self,
|
||||||
program: &ProgramFile,
|
program: &ProgramFile,
|
||||||
|
scope: &Scope,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) {
|
) {
|
||||||
let namespace = program.namespace();
|
let namespace = program.namespace();
|
||||||
|
|
||||||
for declaration in program.declarations() {
|
for declaration in program.declarations() {
|
||||||
self.transpile_declaration(declaration, namespace, handler);
|
self.transpile_declaration(declaration, namespace, scope, handler);
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -123,6 +127,7 @@ impl Transpiler {
|
||||||
&mut self,
|
&mut self,
|
||||||
declaration: &Declaration,
|
declaration: &Declaration,
|
||||||
namespace: &Namespace,
|
namespace: &Namespace,
|
||||||
|
_scope: &Scope,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) {
|
) {
|
||||||
let program_identifier = declaration.span().source_file().identifier().clone();
|
let program_identifier = declaration.span().source_file().identifier().clone();
|
||||||
|
@ -143,7 +148,6 @@ impl Transpiler {
|
||||||
)
|
)
|
||||||
})
|
})
|
||||||
.collect();
|
.collect();
|
||||||
#[allow(clippy::significant_drop_tightening)]
|
|
||||||
self.functions.insert(
|
self.functions.insert(
|
||||||
(program_identifier, name),
|
(program_identifier, name),
|
||||||
FunctionData {
|
FunctionData {
|
||||||
|
@ -203,6 +207,7 @@ impl Transpiler {
|
||||||
if tag.replace().is_some() {
|
if tag.replace().is_some() {
|
||||||
sb_tag.set_replace(true);
|
sb_tag.set_replace(true);
|
||||||
}
|
}
|
||||||
|
// TODO: handle global variables
|
||||||
}
|
}
|
||||||
};
|
};
|
||||||
}
|
}
|
||||||
|
@ -215,6 +220,7 @@ impl Transpiler {
|
||||||
&mut self,
|
&mut self,
|
||||||
identifier_span: &Span,
|
identifier_span: &Span,
|
||||||
arguments: Option<&[&Expression]>,
|
arguments: Option<&[&Expression]>,
|
||||||
|
scope: &Scope,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) -> TranspileResult<(String, Option<BTreeMap<String, String>>)> {
|
) -> TranspileResult<(String, Option<BTreeMap<String, String>>)> {
|
||||||
let program_identifier = identifier_span.source_file().identifier();
|
let program_identifier = identifier_span.source_file().identifier();
|
||||||
|
@ -237,6 +243,8 @@ impl Transpiler {
|
||||||
if !already_transpiled {
|
if !already_transpiled {
|
||||||
tracing::trace!("Function not transpiled yet, transpiling.");
|
tracing::trace!("Function not transpiled yet, transpiling.");
|
||||||
|
|
||||||
|
let function_scope = scope.new_child();
|
||||||
|
|
||||||
let statements = {
|
let statements = {
|
||||||
let functions = &self.functions;
|
let functions = &self.functions;
|
||||||
let function_data = functions
|
let function_data = functions
|
||||||
|
@ -257,9 +265,15 @@ impl Transpiler {
|
||||||
error
|
error
|
||||||
})?;
|
})?;
|
||||||
|
|
||||||
|
for (i, param) in function_data.parameters.iter().enumerate() {
|
||||||
|
function_scope.set_variable(param, VariableType::FunctionArgument { index: i });
|
||||||
|
}
|
||||||
|
|
||||||
function_data.statements.clone()
|
function_data.statements.clone()
|
||||||
};
|
};
|
||||||
let commands = self.transpile_function(&statements, program_identifier, handler)?;
|
|
||||||
|
let commands =
|
||||||
|
self.transpile_function(&statements, program_identifier, &function_scope, handler)?;
|
||||||
|
|
||||||
let functions = &self.functions;
|
let functions = &self.functions;
|
||||||
let function_data = functions
|
let function_data = functions
|
||||||
|
@ -369,22 +383,22 @@ impl Transpiler {
|
||||||
})
|
})
|
||||||
.map(|(s, _)| s.to_owned())?;
|
.map(|(s, _)| s.to_owned())?;
|
||||||
|
|
||||||
let arg_count = arguments.iter().flat_map(|x| x.iter()).count();
|
let arg_count = arguments.map(<[&Expression]>::len);
|
||||||
if arg_count != parameters.len() {
|
if arg_count.is_some_and(|arg_count| arg_count != parameters.len()) {
|
||||||
let err = TranspileError::InvalidFunctionArguments(InvalidFunctionArguments {
|
let err = TranspileError::InvalidFunctionArguments(InvalidFunctionArguments {
|
||||||
expected: parameters.len(),
|
expected: parameters.len(),
|
||||||
actual: arg_count,
|
actual: arg_count.expect("checked in if condition"),
|
||||||
span: identifier_span.clone(),
|
span: identifier_span.clone(),
|
||||||
});
|
});
|
||||||
handler.receive(err.clone());
|
handler.receive(err.clone());
|
||||||
Err(err)
|
Err(err)
|
||||||
} else if arg_count > 0 {
|
} else if arg_count.is_some_and(|arg_count| arg_count > 0) {
|
||||||
let mut compiled_args = Vec::new();
|
let mut compiled_args = Vec::new();
|
||||||
let mut errs = Vec::new();
|
let mut errs = Vec::new();
|
||||||
for expression in arguments.iter().flat_map(|x| x.iter()) {
|
for expression in arguments.iter().flat_map(|x| x.iter()) {
|
||||||
let value = match expression {
|
let value = match expression {
|
||||||
Expression::Primary(Primary::FunctionCall(func)) => self
|
Expression::Primary(Primary::FunctionCall(func)) => self
|
||||||
.transpile_function_call(func, handler)
|
.transpile_function_call(func, scope, handler)
|
||||||
.map(|cmd| match cmd {
|
.map(|cmd| match cmd {
|
||||||
Command::Raw(s) => s,
|
Command::Raw(s) => s,
|
||||||
_ => unreachable!("Function call should always return a raw command"),
|
_ => unreachable!("Function call should always return a raw command"),
|
||||||
|
@ -426,13 +440,14 @@ impl Transpiler {
|
||||||
&mut self,
|
&mut self,
|
||||||
statements: &[Statement],
|
statements: &[Statement],
|
||||||
program_identifier: &str,
|
program_identifier: &str,
|
||||||
|
scope: &Scope,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) -> TranspileResult<Vec<Command>> {
|
) -> TranspileResult<Vec<Command>> {
|
||||||
let mut errors = Vec::new();
|
let mut errors = Vec::new();
|
||||||
let commands = statements
|
let commands = statements
|
||||||
.iter()
|
.iter()
|
||||||
.filter_map(|statement| {
|
.filter_map(|statement| {
|
||||||
self.transpile_statement(statement, program_identifier, handler)
|
self.transpile_statement(statement, program_identifier, scope, handler)
|
||||||
.unwrap_or_else(|err| {
|
.unwrap_or_else(|err| {
|
||||||
errors.push(err);
|
errors.push(err);
|
||||||
None
|
None
|
||||||
|
@ -451,6 +466,7 @@ impl Transpiler {
|
||||||
&mut self,
|
&mut self,
|
||||||
statement: &Statement,
|
statement: &Statement,
|
||||||
program_identifier: &str,
|
program_identifier: &str,
|
||||||
|
scope: &Scope,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) -> TranspileResult<Option<Command>> {
|
) -> TranspileResult<Option<Command>> {
|
||||||
match statement {
|
match statement {
|
||||||
|
@ -459,7 +475,7 @@ impl Transpiler {
|
||||||
}
|
}
|
||||||
Statement::Run(run) => match run.expression() {
|
Statement::Run(run) => match run.expression() {
|
||||||
Expression::Primary(Primary::FunctionCall(func)) => {
|
Expression::Primary(Primary::FunctionCall(func)) => {
|
||||||
self.transpile_function_call(func, handler).map(Some)
|
self.transpile_function_call(func, scope, handler).map(Some)
|
||||||
}
|
}
|
||||||
Expression::Primary(Primary::Integer(num)) => {
|
Expression::Primary(Primary::Integer(num)) => {
|
||||||
let error = TranspileError::UnexpectedExpression(UnexpectedExpression(
|
let error = TranspileError::UnexpectedExpression(UnexpectedExpression(
|
||||||
|
@ -489,23 +505,30 @@ impl Transpiler {
|
||||||
unreachable!("Only literal commands are allowed in functions at this time.")
|
unreachable!("Only literal commands are allowed in functions at this time.")
|
||||||
}
|
}
|
||||||
Statement::ExecuteBlock(execute) => {
|
Statement::ExecuteBlock(execute) => {
|
||||||
self.transpile_execute_block(execute, program_identifier, handler)
|
let child_scope = scope.new_child();
|
||||||
|
self.transpile_execute_block(execute, program_identifier, &child_scope, handler)
|
||||||
}
|
}
|
||||||
Statement::DocComment(doccomment) => {
|
Statement::DocComment(doccomment) => {
|
||||||
let content = doccomment.content();
|
let content = doccomment.content();
|
||||||
Ok(Some(Command::Comment(content.to_string())))
|
Ok(Some(Command::Comment(content.to_string())))
|
||||||
}
|
}
|
||||||
Statement::Grouping(group) => {
|
Statement::Grouping(group) => {
|
||||||
|
let child_scope = scope.new_child();
|
||||||
let statements = group.block().statements();
|
let statements = group.block().statements();
|
||||||
let mut errors = Vec::new();
|
let mut errors = Vec::new();
|
||||||
let commands = statements
|
let commands = statements
|
||||||
.iter()
|
.iter()
|
||||||
.filter_map(|statement| {
|
.filter_map(|statement| {
|
||||||
self.transpile_statement(statement, program_identifier, handler)
|
self.transpile_statement(
|
||||||
.unwrap_or_else(|err| {
|
statement,
|
||||||
errors.push(err);
|
program_identifier,
|
||||||
None
|
&child_scope,
|
||||||
})
|
handler,
|
||||||
|
)
|
||||||
|
.unwrap_or_else(|err| {
|
||||||
|
errors.push(err);
|
||||||
|
None
|
||||||
|
})
|
||||||
})
|
})
|
||||||
.collect::<Vec<_>>();
|
.collect::<Vec<_>>();
|
||||||
if !errors.is_empty() {
|
if !errors.is_empty() {
|
||||||
|
@ -521,7 +544,7 @@ impl Transpiler {
|
||||||
#[expect(clippy::match_wildcard_for_single_variants)]
|
#[expect(clippy::match_wildcard_for_single_variants)]
|
||||||
SemicolonStatement::Expression(expr) => match expr {
|
SemicolonStatement::Expression(expr) => match expr {
|
||||||
Expression::Primary(Primary::FunctionCall(func)) => {
|
Expression::Primary(Primary::FunctionCall(func)) => {
|
||||||
self.transpile_function_call(func, handler).map(Some)
|
self.transpile_function_call(func, scope, handler).map(Some)
|
||||||
}
|
}
|
||||||
unexpected => {
|
unexpected => {
|
||||||
let error = TranspileError::UnexpectedExpression(UnexpectedExpression(
|
let error = TranspileError::UnexpectedExpression(UnexpectedExpression(
|
||||||
|
@ -531,8 +554,24 @@ impl Transpiler {
|
||||||
Err(error)
|
Err(error)
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
SemicolonStatement::VariableDeclaration(_) => {
|
SemicolonStatement::VariableDeclaration(decl) => {
|
||||||
todo!("Variable declarations are not yet supported.")
|
// let value = match decl {
|
||||||
|
// VariableDeclaration::Single(single) => {
|
||||||
|
// match single.variable_type().keyword {
|
||||||
|
// KeywordKind::Int => {
|
||||||
|
// VariableType::ScoreboardValue { objective: (), name: () }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// }
|
||||||
|
// TODO: only for demonstration
|
||||||
|
// scope.set_variable(
|
||||||
|
// decl.identifier().span.str(),
|
||||||
|
// VariableType::Tag {
|
||||||
|
// tag_name: "TODO".to_string(),
|
||||||
|
// },
|
||||||
|
// );
|
||||||
|
todo!("Variable declarations are not yet supported: {decl:?}")
|
||||||
}
|
}
|
||||||
},
|
},
|
||||||
}
|
}
|
||||||
|
@ -541,14 +580,19 @@ impl Transpiler {
|
||||||
fn transpile_function_call(
|
fn transpile_function_call(
|
||||||
&mut self,
|
&mut self,
|
||||||
func: &FunctionCall,
|
func: &FunctionCall,
|
||||||
|
scope: &Scope,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) -> TranspileResult<Command> {
|
) -> TranspileResult<Command> {
|
||||||
let arguments = func
|
let arguments = func
|
||||||
.arguments()
|
.arguments()
|
||||||
.as_ref()
|
.as_ref()
|
||||||
.map(|l| l.elements().map(Deref::deref).collect::<Vec<_>>());
|
.map(|l| l.elements().map(Deref::deref).collect::<Vec<_>>());
|
||||||
let (location, arguments) =
|
let (location, arguments) = self.get_or_transpile_function(
|
||||||
self.get_or_transpile_function(&func.identifier().span, arguments.as_deref(), handler)?;
|
&func.identifier().span,
|
||||||
|
arguments.as_deref(),
|
||||||
|
scope,
|
||||||
|
handler,
|
||||||
|
)?;
|
||||||
let mut function_call = format!("function {location}");
|
let mut function_call = format!("function {location}");
|
||||||
if let Some(arguments) = arguments {
|
if let Some(arguments) = arguments {
|
||||||
use std::fmt::Write;
|
use std::fmt::Write;
|
||||||
|
@ -572,9 +616,10 @@ impl Transpiler {
|
||||||
&mut self,
|
&mut self,
|
||||||
execute: &ExecuteBlock,
|
execute: &ExecuteBlock,
|
||||||
program_identifier: &str,
|
program_identifier: &str,
|
||||||
|
scope: &Scope,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) -> TranspileResult<Option<Command>> {
|
) -> TranspileResult<Option<Command>> {
|
||||||
self.transpile_execute_block_internal(execute, program_identifier, handler)
|
self.transpile_execute_block_internal(execute, program_identifier, scope, handler)
|
||||||
.map(|ex| ex.map(Command::Execute))
|
.map(|ex| ex.map(Command::Execute))
|
||||||
}
|
}
|
||||||
|
|
||||||
|
@ -582,6 +627,7 @@ impl Transpiler {
|
||||||
&mut self,
|
&mut self,
|
||||||
execute: &ExecuteBlock,
|
execute: &ExecuteBlock,
|
||||||
program_identifier: &str,
|
program_identifier: &str,
|
||||||
|
scope: &Scope,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) -> TranspileResult<Option<Execute>> {
|
) -> TranspileResult<Option<Execute>> {
|
||||||
match execute {
|
match execute {
|
||||||
|
@ -593,7 +639,7 @@ impl Transpiler {
|
||||||
.statements()
|
.statements()
|
||||||
.iter()
|
.iter()
|
||||||
.filter_map(|s| {
|
.filter_map(|s| {
|
||||||
self.transpile_statement(s, program_identifier, handler)
|
self.transpile_statement(s, program_identifier, scope, handler)
|
||||||
.unwrap_or_else(|err| {
|
.unwrap_or_else(|err| {
|
||||||
errors.push(err);
|
errors.push(err);
|
||||||
None
|
None
|
||||||
|
@ -614,11 +660,12 @@ impl Transpiler {
|
||||||
.transpile_execute_block_internal(
|
.transpile_execute_block_internal(
|
||||||
execute_block,
|
execute_block,
|
||||||
program_identifier,
|
program_identifier,
|
||||||
|
scope,
|
||||||
handler,
|
handler,
|
||||||
),
|
),
|
||||||
}?;
|
}?;
|
||||||
|
|
||||||
self.combine_execute_head_tail(head, tail, program_identifier, handler)
|
self.combine_execute_head_tail(head, tail, program_identifier, scope, handler)
|
||||||
}
|
}
|
||||||
ExecuteBlock::IfElse(cond, block, el) => {
|
ExecuteBlock::IfElse(cond, block, el) => {
|
||||||
let statements = block.statements();
|
let statements = block.statements();
|
||||||
|
@ -629,7 +676,7 @@ impl Transpiler {
|
||||||
let commands = statements
|
let commands = statements
|
||||||
.iter()
|
.iter()
|
||||||
.filter_map(|statement| {
|
.filter_map(|statement| {
|
||||||
self.transpile_statement(statement, program_identifier, handler)
|
self.transpile_statement(statement, program_identifier, scope, handler)
|
||||||
.unwrap_or_else(|err| {
|
.unwrap_or_else(|err| {
|
||||||
errors.push(err);
|
errors.push(err);
|
||||||
None
|
None
|
||||||
|
@ -641,7 +688,7 @@ impl Transpiler {
|
||||||
}
|
}
|
||||||
Some(Execute::Runs(commands))
|
Some(Execute::Runs(commands))
|
||||||
} else {
|
} else {
|
||||||
self.transpile_statement(&statements[0], program_identifier, handler)?
|
self.transpile_statement(&statements[0], program_identifier, scope, handler)?
|
||||||
.map(|cmd| Execute::Run(Box::new(cmd)))
|
.map(|cmd| Execute::Run(Box::new(cmd)))
|
||||||
};
|
};
|
||||||
|
|
||||||
|
@ -653,6 +700,7 @@ impl Transpiler {
|
||||||
then,
|
then,
|
||||||
Some(el),
|
Some(el),
|
||||||
program_identifier,
|
program_identifier,
|
||||||
|
scope,
|
||||||
handler,
|
handler,
|
||||||
)
|
)
|
||||||
},
|
},
|
||||||
|
@ -667,6 +715,7 @@ impl Transpiler {
|
||||||
then: Execute,
|
then: Execute,
|
||||||
el: Option<&Else>,
|
el: Option<&Else>,
|
||||||
program_identifier: &str,
|
program_identifier: &str,
|
||||||
|
scope: &Scope,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) -> TranspileResult<Option<Execute>> {
|
) -> TranspileResult<Option<Execute>> {
|
||||||
let (_, cond) = cond.clone().dissolve();
|
let (_, cond) = cond.clone().dissolve();
|
||||||
|
@ -680,7 +729,7 @@ impl Transpiler {
|
||||||
if statements.is_empty() {
|
if statements.is_empty() {
|
||||||
None
|
None
|
||||||
} else if statements.len() == 1 {
|
} else if statements.len() == 1 {
|
||||||
self.transpile_statement(&statements[0], program_identifier, handler)
|
self.transpile_statement(&statements[0], program_identifier, scope, handler)
|
||||||
.unwrap_or_else(|err| {
|
.unwrap_or_else(|err| {
|
||||||
errors.push(err);
|
errors.push(err);
|
||||||
None
|
None
|
||||||
|
@ -690,7 +739,7 @@ impl Transpiler {
|
||||||
let commands = statements
|
let commands = statements
|
||||||
.iter()
|
.iter()
|
||||||
.filter_map(|statement| {
|
.filter_map(|statement| {
|
||||||
self.transpile_statement(statement, program_identifier, handler)
|
self.transpile_statement(statement, program_identifier, scope, handler)
|
||||||
.unwrap_or_else(|err| {
|
.unwrap_or_else(|err| {
|
||||||
errors.push(err);
|
errors.push(err);
|
||||||
None
|
None
|
||||||
|
@ -718,12 +767,20 @@ impl Transpiler {
|
||||||
head: &ExecuteBlockHead,
|
head: &ExecuteBlockHead,
|
||||||
tail: Option<Execute>,
|
tail: Option<Execute>,
|
||||||
program_identifier: &str,
|
program_identifier: &str,
|
||||||
|
scope: &Scope,
|
||||||
handler: &impl Handler<base::Error>,
|
handler: &impl Handler<base::Error>,
|
||||||
) -> TranspileResult<Option<Execute>> {
|
) -> TranspileResult<Option<Execute>> {
|
||||||
Ok(match head {
|
Ok(match head {
|
||||||
ExecuteBlockHead::Conditional(cond) => {
|
ExecuteBlockHead::Conditional(cond) => {
|
||||||
if let Some(tail) = tail {
|
if let Some(tail) = tail {
|
||||||
self.transpile_conditional(cond, tail, None, program_identifier, handler)?
|
self.transpile_conditional(
|
||||||
|
cond,
|
||||||
|
tail,
|
||||||
|
None,
|
||||||
|
program_identifier,
|
||||||
|
scope,
|
||||||
|
handler,
|
||||||
|
)?
|
||||||
} else {
|
} else {
|
||||||
None
|
None
|
||||||
}
|
}
|
||||||
|
|
|
@ -0,0 +1,117 @@
|
||||||
|
#![expect(unused)]
|
||||||
|
|
||||||
|
use std::{collections::HashMap, sync::RwLock};
|
||||||
|
|
||||||
|
use super::Transpiler;
|
||||||
|
|
||||||
|
#[derive(Debug, Clone)]
|
||||||
|
pub enum VariableType {
|
||||||
|
FunctionArgument {
|
||||||
|
index: usize,
|
||||||
|
},
|
||||||
|
Scoreboard {
|
||||||
|
objective: String,
|
||||||
|
},
|
||||||
|
ScoreboardValue {
|
||||||
|
objective: String,
|
||||||
|
name: String,
|
||||||
|
},
|
||||||
|
ScoreboardArray {
|
||||||
|
objective: String,
|
||||||
|
names: Vec<String>,
|
||||||
|
},
|
||||||
|
Tag {
|
||||||
|
tag_name: String,
|
||||||
|
},
|
||||||
|
BooleanStorage {
|
||||||
|
storage_name: String,
|
||||||
|
path: String,
|
||||||
|
},
|
||||||
|
BooleanStorageArray {
|
||||||
|
storage_name: String,
|
||||||
|
paths: Vec<String>,
|
||||||
|
},
|
||||||
|
}
|
||||||
|
|
||||||
|
#[derive(Debug, Default)]
|
||||||
|
pub struct Scope<'a> {
|
||||||
|
parent: Option<&'a Scope<'a>>,
|
||||||
|
variables: RwLock<HashMap<String, VariableType>>,
|
||||||
|
}
|
||||||
|
|
||||||
|
impl<'a> Scope<'a> {
|
||||||
|
pub fn new() -> Self {
|
||||||
|
Self::default()
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn new_child(&'a self) -> Self {
|
||||||
|
Self {
|
||||||
|
parent: Some(self),
|
||||||
|
variables: RwLock::new(HashMap::new()),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_variable(&self, name: &str) -> Option<VariableType> {
|
||||||
|
let var = self.variables.read().unwrap().get(name).cloned();
|
||||||
|
if var.is_some() {
|
||||||
|
var
|
||||||
|
} else {
|
||||||
|
self.parent.and_then(|parent| parent.get_variable(name))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn set_variable(&self, name: &str, var: VariableType) {
|
||||||
|
self.variables
|
||||||
|
.write()
|
||||||
|
.unwrap()
|
||||||
|
.insert(name.to_string(), var);
|
||||||
|
}
|
||||||
|
|
||||||
|
pub fn get_parent(&self) -> Option<&'a Self> {
|
||||||
|
self.parent
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
impl Transpiler {}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_scope() {
|
||||||
|
let scope = Scope::new();
|
||||||
|
{
|
||||||
|
let mut variables = scope.variables.write().unwrap();
|
||||||
|
variables.insert(
|
||||||
|
"test".to_string(),
|
||||||
|
VariableType::Scoreboard {
|
||||||
|
objective: "test".to_string(),
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
match scope.get_variable("test") {
|
||||||
|
Some(VariableType::Scoreboard { objective }) => assert_eq!(objective, "test"),
|
||||||
|
_ => panic!("Incorrect Variable"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_parent() {
|
||||||
|
let scope = Scope::new();
|
||||||
|
{
|
||||||
|
let mut variables = scope.variables.write().unwrap();
|
||||||
|
variables.insert(
|
||||||
|
"test".to_string(),
|
||||||
|
VariableType::Scoreboard {
|
||||||
|
objective: "test".to_string(),
|
||||||
|
},
|
||||||
|
);
|
||||||
|
}
|
||||||
|
let child = scope.new_child();
|
||||||
|
match child.get_variable("test") {
|
||||||
|
Some(VariableType::Scoreboard { objective }) => assert_eq!(objective, "test"),
|
||||||
|
_ => panic!("Incorrect Variable"),
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
Loading…
Reference in New Issue