allow marking return value from lua as containing macro
This commit is contained in:
parent
8a9db43424
commit
6dde4b41c1
|
@ -10,8 +10,8 @@ mod enabled {
|
||||||
base::{self, source_file::SourceElement, Handler},
|
base::{self, source_file::SourceElement, Handler},
|
||||||
syntax::syntax_tree::expression::LuaCode,
|
syntax::syntax_tree::expression::LuaCode,
|
||||||
transpile::{
|
transpile::{
|
||||||
error::{LuaRuntimeError, TranspileError, TranspileResult},
|
error::{LuaRuntimeError, MismatchedTypes, TranspileError, TranspileResult},
|
||||||
expression::ComptimeValue,
|
expression::{ComptimeValue, ExpectedType},
|
||||||
Scope, VariableData,
|
Scope, VariableData,
|
||||||
},
|
},
|
||||||
};
|
};
|
||||||
|
@ -124,7 +124,7 @@ mod enabled {
|
||||||
table.set("path", lua.create_string(path)?)?;
|
table.set("path", lua.create_string(path)?)?;
|
||||||
Value::Table(table)
|
Value::Table(table)
|
||||||
}
|
}
|
||||||
Some(_) => todo!("allow other types"),
|
Some(_) => todo!("allow other variable types"),
|
||||||
None => todo!("throw correct error"),
|
None => todo!("throw correct error"),
|
||||||
};
|
};
|
||||||
globals.set(name, value)?;
|
globals.set(name, value)?;
|
||||||
|
@ -155,8 +155,68 @@ mod enabled {
|
||||||
handler,
|
handler,
|
||||||
),
|
),
|
||||||
Value::Boolean(boolean) => Ok(Some(ComptimeValue::Boolean(boolean))),
|
Value::Boolean(boolean) => Ok(Some(ComptimeValue::Boolean(boolean))),
|
||||||
|
Value::Table(table) => match table.get::<Value>("value") {
|
||||||
|
Ok(Value::Nil) => {
|
||||||
|
let err = TranspileError::LuaRuntimeError(LuaRuntimeError {
|
||||||
|
code_block: self.span(),
|
||||||
|
error_message: "return table must contain non-nil 'value'".to_string(),
|
||||||
|
});
|
||||||
|
handler.receive(err.clone());
|
||||||
|
Err(err)
|
||||||
|
}
|
||||||
|
Ok(value) => {
|
||||||
|
let value = match self.handle_lua_result(value, handler)? {
|
||||||
|
Some(ComptimeValue::String(s)) => {
|
||||||
|
let contains_macro = match table.get::<Value>("contains_macro") {
|
||||||
|
Ok(Value::Boolean(boolean)) => Ok(boolean),
|
||||||
|
Ok(value) => {
|
||||||
|
if let Some(ComptimeValue::Boolean(boolean)) =
|
||||||
|
self.handle_lua_result(value, handler)?
|
||||||
|
{
|
||||||
|
Ok(boolean)
|
||||||
|
} else {
|
||||||
|
let err =
|
||||||
|
TranspileError::MismatchedTypes(MismatchedTypes {
|
||||||
|
expression: self.span(),
|
||||||
|
expected_type: ExpectedType::Boolean,
|
||||||
|
});
|
||||||
|
handler.receive(err.clone());
|
||||||
|
Err(err)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
let err =
|
||||||
|
TranspileError::MismatchedTypes(MismatchedTypes {
|
||||||
|
expression: self.span(),
|
||||||
|
expected_type: ExpectedType::Boolean,
|
||||||
|
});
|
||||||
|
handler.receive(err.clone());
|
||||||
|
Err(err)
|
||||||
|
}
|
||||||
|
}?;
|
||||||
|
|
||||||
|
if contains_macro {
|
||||||
|
Some(ComptimeValue::MacroString(
|
||||||
|
s.parse().expect("parsing cannot fail"),
|
||||||
|
))
|
||||||
|
} else {
|
||||||
|
Some(ComptimeValue::String(s))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
value => value,
|
||||||
|
};
|
||||||
|
Ok(value)
|
||||||
|
}
|
||||||
|
Err(err) => {
|
||||||
|
let err = TranspileError::LuaRuntimeError(LuaRuntimeError::from_lua_err(
|
||||||
|
&err,
|
||||||
|
self.span(),
|
||||||
|
));
|
||||||
|
handler.receive(err.clone());
|
||||||
|
Err(err)
|
||||||
|
}
|
||||||
|
},
|
||||||
Value::Error(_)
|
Value::Error(_)
|
||||||
| Value::Table(_)
|
|
||||||
| Value::Thread(_)
|
| Value::Thread(_)
|
||||||
| Value::UserData(_)
|
| Value::UserData(_)
|
||||||
| Value::LightUserData(_)
|
| Value::LightUserData(_)
|
||||||
|
|
|
@ -1,6 +1,6 @@
|
||||||
//! Utility methods for transpiling
|
//! Utility methods for transpiling
|
||||||
|
|
||||||
use std::fmt::Display;
|
use std::{fmt::Display, str::FromStr};
|
||||||
|
|
||||||
use crate::{
|
use crate::{
|
||||||
lexical::token::{MacroStringLiteral, MacroStringLiteralPart},
|
lexical::token::{MacroStringLiteral, MacroStringLiteralPart},
|
||||||
|
@ -114,6 +114,61 @@ where
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
impl FromStr for MacroString {
|
||||||
|
type Err = ();
|
||||||
|
|
||||||
|
fn from_str(s: &str) -> Result<Self, Self::Err> {
|
||||||
|
let pos = s.find("$(");
|
||||||
|
if pos.is_some_and(|pos| s[pos..].contains(')')) {
|
||||||
|
let mut parts = Vec::new();
|
||||||
|
let mut s = s;
|
||||||
|
while let Some(pos) = s.find("$(") {
|
||||||
|
let (before, after) = s.split_at(pos);
|
||||||
|
|
||||||
|
let last_macro_index = after
|
||||||
|
.char_indices()
|
||||||
|
.skip(2)
|
||||||
|
.take_while(|&(_, c)| c.is_ascii_alphanumeric() || c == '_')
|
||||||
|
.map(|(i, _)| i)
|
||||||
|
.last();
|
||||||
|
|
||||||
|
match last_macro_index {
|
||||||
|
Some(last_macro_index) if after[last_macro_index + 1..].starts_with(')') => {
|
||||||
|
if !before.is_empty() {
|
||||||
|
parts.push(MacroStringPart::String(before.to_string()));
|
||||||
|
}
|
||||||
|
parts.push(MacroStringPart::MacroUsage(
|
||||||
|
after[2..=last_macro_index].to_string(),
|
||||||
|
));
|
||||||
|
s = &after[last_macro_index + 2..];
|
||||||
|
if s.is_empty() {
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
_ => {
|
||||||
|
parts.push(MacroStringPart::String(s.to_string()));
|
||||||
|
s = "";
|
||||||
|
break;
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
if !s.is_empty() {
|
||||||
|
parts.push(MacroStringPart::String(s.to_string()));
|
||||||
|
}
|
||||||
|
if parts
|
||||||
|
.iter()
|
||||||
|
.any(|p| matches!(p, MacroStringPart::MacroUsage(_)))
|
||||||
|
{
|
||||||
|
Ok(Self::MacroString(parts))
|
||||||
|
} else {
|
||||||
|
Ok(Self::String(s.to_string()))
|
||||||
|
}
|
||||||
|
} else {
|
||||||
|
Ok(Self::String(s.to_string()))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
impl<S> From<S> for MacroString
|
impl<S> From<S> for MacroString
|
||||||
where
|
where
|
||||||
S: Into<String>,
|
S: Into<String>,
|
||||||
|
@ -172,3 +227,39 @@ impl From<MacroStringLiteral> for MacroString {
|
||||||
Self::from(&value)
|
Self::from(&value)
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod tests {
|
||||||
|
use super::*;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn test_parse_macro_string() {
|
||||||
|
assert_eq!(
|
||||||
|
MacroString::from_str("Hello, $(world)!").unwrap(),
|
||||||
|
MacroString::MacroString(vec![
|
||||||
|
MacroStringPart::String("Hello, ".to_string()),
|
||||||
|
MacroStringPart::MacroUsage("world".to_string()),
|
||||||
|
MacroStringPart::String("!".to_string())
|
||||||
|
])
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
MacroString::from_str("Hello, $(world)! $(world").unwrap(),
|
||||||
|
MacroString::MacroString(vec![
|
||||||
|
MacroStringPart::String("Hello, ".to_string()),
|
||||||
|
MacroStringPart::MacroUsage("world".to_string()),
|
||||||
|
MacroStringPart::String("! $(world".to_string()),
|
||||||
|
])
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
MacroString::from_str("Hello $(a) from $(b) and $(c)").unwrap(),
|
||||||
|
MacroString::MacroString(vec![
|
||||||
|
MacroStringPart::String("Hello ".to_string()),
|
||||||
|
MacroStringPart::MacroUsage("a".to_string()),
|
||||||
|
MacroStringPart::String(" from ".to_string()),
|
||||||
|
MacroStringPart::MacroUsage("b".to_string()),
|
||||||
|
MacroStringPart::String(" and ".to_string()),
|
||||||
|
MacroStringPart::MacroUsage("c".to_string()),
|
||||||
|
])
|
||||||
|
);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
|
@ -129,6 +129,8 @@ impl<'a> Scope<'a> {
|
||||||
}
|
}
|
||||||
|
|
||||||
/// Gets the number of times a variable has been shadowed.
|
/// Gets the number of times a variable has been shadowed.
|
||||||
|
///
|
||||||
|
///
|
||||||
pub fn get_variable_shadow_count(&self, name: &str) -> usize {
|
pub fn get_variable_shadow_count(&self, name: &str) -> usize {
|
||||||
let count = self
|
let count = self
|
||||||
.shadowed
|
.shadowed
|
||||||
|
@ -138,9 +140,7 @@ impl<'a> Scope<'a> {
|
||||||
.copied()
|
.copied()
|
||||||
.unwrap_or(0);
|
.unwrap_or(0);
|
||||||
self.parent.as_ref().map_or(count, |parent| {
|
self.parent.as_ref().map_or(count, |parent| {
|
||||||
count
|
count.saturating_sub(1) + parent.get_variable_shadow_count(name)
|
||||||
+ parent.get_variable_shadow_count(name)
|
|
||||||
+ usize::from(parent.get_variable(name).is_some())
|
|
||||||
})
|
})
|
||||||
}
|
}
|
||||||
|
|
||||||
|
|
Loading…
Reference in New Issue