Merge pull request #51 from jhwgh1968/plugins
Fix CI and reorganize tests
This commit is contained in:
commit
44b581c61c
@ -553,522 +553,3 @@ impl ExportedFn {
|
|||||||
}
|
}
|
||||||
}
|
}
|
||||||
}
|
}
|
||||||
|
|
||||||
#[cfg(test)]
|
|
||||||
mod function_tests {
|
|
||||||
use super::ExportedFn;
|
|
||||||
|
|
||||||
use proc_macro2::TokenStream;
|
|
||||||
use quote::quote;
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn minimal_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn do_nothing() { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert_eq!(&item_fn.name().to_string(), "do_nothing");
|
|
||||||
assert!(!item_fn.mutable_receiver());
|
|
||||||
assert!(item_fn.is_public());
|
|
||||||
assert!(item_fn.return_type().is_none());
|
|
||||||
assert_eq!(item_fn.arg_list().count(), 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn one_arg_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn do_something(x: usize) { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert_eq!(&item_fn.name().to_string(), "do_something");
|
|
||||||
assert_eq!(item_fn.arg_list().count(), 1);
|
|
||||||
assert!(!item_fn.mutable_receiver());
|
|
||||||
assert!(item_fn.is_public());
|
|
||||||
assert!(item_fn.return_type().is_none());
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
item_fn.arg_list().next().unwrap(),
|
|
||||||
&syn::parse2::<syn::FnArg>(quote! { x: usize }).unwrap()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn two_arg_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn do_something(x: usize, y: f32) { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert_eq!(&item_fn.name().to_string(), "do_something");
|
|
||||||
assert_eq!(item_fn.arg_list().count(), 2);
|
|
||||||
assert!(!item_fn.mutable_receiver());
|
|
||||||
assert!(item_fn.is_public());
|
|
||||||
assert!(item_fn.return_type().is_none());
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
item_fn.arg_list().next().unwrap(),
|
|
||||||
&syn::parse2::<syn::FnArg>(quote! { x: usize }).unwrap()
|
|
||||||
);
|
|
||||||
assert_eq!(
|
|
||||||
item_fn.arg_list().nth(1).unwrap(),
|
|
||||||
&syn::parse2::<syn::FnArg>(quote! { y: f32 }).unwrap()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn usize_returning_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn get_magic_number() -> usize { 42 }
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert_eq!(&item_fn.name().to_string(), "get_magic_number");
|
|
||||||
assert!(!item_fn.mutable_receiver());
|
|
||||||
assert!(item_fn.is_public());
|
|
||||||
assert_eq!(item_fn.arg_list().count(), 0);
|
|
||||||
assert_eq!(
|
|
||||||
item_fn.return_type().unwrap(),
|
|
||||||
&syn::Type::Path(syn::TypePath {
|
|
||||||
qself: None,
|
|
||||||
path: syn::parse2::<syn::Path>(quote! { usize }).unwrap()
|
|
||||||
})
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn ref_returning_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn get_magic_phrase() -> &'static str { "open sesame" }
|
|
||||||
};
|
|
||||||
|
|
||||||
let err = syn::parse2::<ExportedFn>(input_tokens).unwrap_err();
|
|
||||||
assert_eq!(format!("{}", err), "cannot return a reference to Rhai");
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn ptr_returning_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn get_magic_phrase() -> *const str { "open sesame" }
|
|
||||||
};
|
|
||||||
|
|
||||||
let err = syn::parse2::<ExportedFn>(input_tokens).unwrap_err();
|
|
||||||
assert_eq!(format!("{}", err), "cannot return a pointer to Rhai");
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn ref_arg_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn greet(who: &Person) { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let err = syn::parse2::<ExportedFn>(input_tokens).unwrap_err();
|
|
||||||
assert_eq!(
|
|
||||||
format!("{}", err),
|
|
||||||
"references from Rhai in this position must be mutable"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn ref_second_arg_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn greet(count: usize, who: &Person) { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let err = syn::parse2::<ExportedFn>(input_tokens).unwrap_err();
|
|
||||||
assert_eq!(
|
|
||||||
format!("{}", err),
|
|
||||||
"this type in this position passes from Rhai by value"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn mut_ref_second_arg_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn give(item_name: &str, who: &mut Person) { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let err = syn::parse2::<ExportedFn>(input_tokens).unwrap_err();
|
|
||||||
assert_eq!(
|
|
||||||
format!("{}", err),
|
|
||||||
"this type in this position passes from Rhai by value"
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn str_arg_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn log(message: &str) { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert_eq!(&item_fn.name().to_string(), "log");
|
|
||||||
assert_eq!(item_fn.arg_list().count(), 1);
|
|
||||||
assert!(!item_fn.mutable_receiver());
|
|
||||||
assert!(item_fn.is_public());
|
|
||||||
assert!(item_fn.return_type().is_none());
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
item_fn.arg_list().next().unwrap(),
|
|
||||||
&syn::parse2::<syn::FnArg>(quote! { message: &str }).unwrap()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn str_second_arg_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn log(level: usize, message: &str) { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert_eq!(&item_fn.name().to_string(), "log");
|
|
||||||
assert_eq!(item_fn.arg_list().count(), 2);
|
|
||||||
assert!(!item_fn.mutable_receiver());
|
|
||||||
assert!(item_fn.is_public());
|
|
||||||
assert!(item_fn.return_type().is_none());
|
|
||||||
|
|
||||||
assert_eq!(
|
|
||||||
item_fn.arg_list().next().unwrap(),
|
|
||||||
&syn::parse2::<syn::FnArg>(quote! { level: usize }).unwrap()
|
|
||||||
);
|
|
||||||
assert_eq!(
|
|
||||||
item_fn.arg_list().nth(1).unwrap(),
|
|
||||||
&syn::parse2::<syn::FnArg>(quote! { message: &str }).unwrap()
|
|
||||||
);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn private_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
fn do_nothing() { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert_eq!(&item_fn.name().to_string(), "do_nothing");
|
|
||||||
assert!(!item_fn.mutable_receiver());
|
|
||||||
assert!(!item_fn.is_public());
|
|
||||||
assert!(item_fn.return_type().is_none());
|
|
||||||
assert_eq!(item_fn.arg_list().count(), 0);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn receiver_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn act_upon(&mut self) { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert_eq!(&item_fn.name().to_string(), "act_upon");
|
|
||||||
assert!(item_fn.mutable_receiver());
|
|
||||||
assert!(item_fn.is_public());
|
|
||||||
assert!(item_fn.return_type().is_none());
|
|
||||||
assert_eq!(item_fn.arg_list().count(), 1);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn immutable_receiver_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn act_upon(&self) { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert_eq!(&item_fn.name().to_string(), "act_upon");
|
|
||||||
assert!(item_fn.mutable_receiver());
|
|
||||||
assert!(item_fn.is_public());
|
|
||||||
assert!(item_fn.return_type().is_none());
|
|
||||||
assert_eq!(item_fn.arg_list().count(), 1);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
|
||||||
#[cfg(test)]
|
|
||||||
mod generate_tests {
|
|
||||||
use super::ExportedFn;
|
|
||||||
|
|
||||||
use proc_macro2::TokenStream;
|
|
||||||
use quote::quote;
|
|
||||||
|
|
||||||
fn assert_streams_eq(actual: TokenStream, expected: TokenStream) {
|
|
||||||
let actual = actual.to_string();
|
|
||||||
let expected = expected.to_string();
|
|
||||||
if &actual != &expected {
|
|
||||||
let mut counter = 0;
|
|
||||||
let iter = actual
|
|
||||||
.chars()
|
|
||||||
.zip(expected.chars())
|
|
||||||
.inspect(|_| counter += 1)
|
|
||||||
.skip_while(|(a, e)| *a == *e);
|
|
||||||
let (actual_diff, expected_diff) = {
|
|
||||||
let mut actual_diff = String::new();
|
|
||||||
let mut expected_diff = String::new();
|
|
||||||
for (a, e) in iter.take(50) {
|
|
||||||
actual_diff.push(a);
|
|
||||||
expected_diff.push(e);
|
|
||||||
}
|
|
||||||
(actual_diff, expected_diff)
|
|
||||||
};
|
|
||||||
eprintln!("actual != expected, diverge at char {}", counter);
|
|
||||||
}
|
|
||||||
assert_eq!(actual, expected);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn minimal_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn do_nothing() { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let expected_tokens = quote! {
|
|
||||||
#[allow(unused)]
|
|
||||||
pub mod rhai_fn_do_nothing {
|
|
||||||
use super::*;
|
|
||||||
struct Token();
|
|
||||||
impl PluginFunction for Token {
|
|
||||||
fn call(&self,
|
|
||||||
args: &mut [&mut Dynamic], pos: Position
|
|
||||||
) -> Result<Dynamic, Box<EvalAltResult>> {
|
|
||||||
debug_assert_eq!(args.len(), 0usize,
|
|
||||||
"wrong arg count: {} != {}", args.len(), 0usize);
|
|
||||||
Ok(Dynamic::from(do_nothing()))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn is_method_call(&self) -> bool { false }
|
|
||||||
fn is_varadic(&self) -> bool { false }
|
|
||||||
fn clone_boxed(&self) -> Box<dyn PluginFunction> { Box::new(Token()) }
|
|
||||||
fn input_types(&self) -> Box<[TypeId]> {
|
|
||||||
new_vec![].into_boxed_slice()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pub fn token_callable() -> CallableFunction {
|
|
||||||
CallableFunction::from_plugin(Token())
|
|
||||||
}
|
|
||||||
pub fn token_input_types() -> Box<[TypeId]> {
|
|
||||||
Token().input_types()
|
|
||||||
}
|
|
||||||
type EvalBox = Box<EvalAltResult>;
|
|
||||||
pub fn dynamic_result_fn() -> Result<Dynamic, EvalBox> {
|
|
||||||
Ok(Dynamic::from(super::do_nothing()))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert_streams_eq(item_fn.generate(), expected_tokens);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn one_arg_usize_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn do_something(x: usize) { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let expected_tokens = quote! {
|
|
||||||
#[allow(unused)]
|
|
||||||
pub mod rhai_fn_do_something {
|
|
||||||
use super::*;
|
|
||||||
struct Token();
|
|
||||||
impl PluginFunction for Token {
|
|
||||||
fn call(&self,
|
|
||||||
args: &mut [&mut Dynamic], pos: Position
|
|
||||||
) -> Result<Dynamic, Box<EvalAltResult>> {
|
|
||||||
debug_assert_eq!(args.len(), 1usize,
|
|
||||||
"wrong arg count: {} != {}", args.len(), 1usize);
|
|
||||||
let arg0 = mem::take(args[0usize]).clone().cast::<usize>();
|
|
||||||
Ok(Dynamic::from(do_something(arg0)))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn is_method_call(&self) -> bool { false }
|
|
||||||
fn is_varadic(&self) -> bool { false }
|
|
||||||
fn clone_boxed(&self) -> Box<dyn PluginFunction> { Box::new(Token()) }
|
|
||||||
fn input_types(&self) -> Box<[TypeId]> {
|
|
||||||
new_vec![TypeId::of::<usize>()].into_boxed_slice()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pub fn token_callable() -> CallableFunction {
|
|
||||||
CallableFunction::from_plugin(Token())
|
|
||||||
}
|
|
||||||
pub fn token_input_types() -> Box<[TypeId]> {
|
|
||||||
Token().input_types()
|
|
||||||
}
|
|
||||||
type EvalBox = Box<EvalAltResult>;
|
|
||||||
pub fn dynamic_result_fn(x: usize) -> Result<Dynamic, EvalBox> {
|
|
||||||
Ok(Dynamic::from(super::do_something(x)))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert_streams_eq(item_fn.generate(), expected_tokens);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn one_arg_usize_fn_impl() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn do_something(x: usize) { }
|
|
||||||
};
|
|
||||||
|
|
||||||
let expected_tokens = quote! {
|
|
||||||
impl PluginFunction for MyType {
|
|
||||||
fn call(&self,
|
|
||||||
args: &mut [&mut Dynamic], pos: Position
|
|
||||||
) -> Result<Dynamic, Box<EvalAltResult>> {
|
|
||||||
debug_assert_eq!(args.len(), 1usize,
|
|
||||||
"wrong arg count: {} != {}", args.len(), 1usize);
|
|
||||||
let arg0 = mem::take(args[0usize]).clone().cast::<usize>();
|
|
||||||
Ok(Dynamic::from(do_something(arg0)))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn is_method_call(&self) -> bool { false }
|
|
||||||
fn is_varadic(&self) -> bool { false }
|
|
||||||
fn clone_boxed(&self) -> Box<dyn PluginFunction> { Box::new(MyType()) }
|
|
||||||
fn input_types(&self) -> Box<[TypeId]> {
|
|
||||||
new_vec![TypeId::of::<usize>()].into_boxed_slice()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert_streams_eq(item_fn.generate_impl("MyType"), expected_tokens);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn two_arg_returning_usize_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn add_together(x: usize, y: usize) -> usize { x + y }
|
|
||||||
};
|
|
||||||
|
|
||||||
let expected_tokens = quote! {
|
|
||||||
#[allow(unused)]
|
|
||||||
pub mod rhai_fn_add_together {
|
|
||||||
use super::*;
|
|
||||||
struct Token();
|
|
||||||
impl PluginFunction for Token {
|
|
||||||
fn call(&self,
|
|
||||||
args: &mut [&mut Dynamic], pos: Position
|
|
||||||
) -> Result<Dynamic, Box<EvalAltResult>> {
|
|
||||||
debug_assert_eq!(args.len(), 2usize,
|
|
||||||
"wrong arg count: {} != {}", args.len(), 2usize);
|
|
||||||
let arg0 = mem::take(args[0usize]).clone().cast::<usize>();
|
|
||||||
let arg1 = mem::take(args[1usize]).clone().cast::<usize>();
|
|
||||||
Ok(Dynamic::from(add_together(arg0, arg1)))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn is_method_call(&self) -> bool { false }
|
|
||||||
fn is_varadic(&self) -> bool { false }
|
|
||||||
fn clone_boxed(&self) -> Box<dyn PluginFunction> { Box::new(Token()) }
|
|
||||||
fn input_types(&self) -> Box<[TypeId]> {
|
|
||||||
new_vec![TypeId::of::<usize>(),
|
|
||||||
TypeId::of::<usize>()].into_boxed_slice()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pub fn token_callable() -> CallableFunction {
|
|
||||||
CallableFunction::from_plugin(Token())
|
|
||||||
}
|
|
||||||
pub fn token_input_types() -> Box<[TypeId]> {
|
|
||||||
Token().input_types()
|
|
||||||
}
|
|
||||||
type EvalBox = Box<EvalAltResult>;
|
|
||||||
pub fn dynamic_result_fn(x: usize, y: usize) -> Result<Dynamic, EvalBox> {
|
|
||||||
Ok(Dynamic::from(super::add_together(x, y)))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert_streams_eq(item_fn.generate(), expected_tokens);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn mut_arg_usize_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn increment(x: &mut usize, y: usize) { *x += y; }
|
|
||||||
};
|
|
||||||
|
|
||||||
let expected_tokens = quote! {
|
|
||||||
#[allow(unused)]
|
|
||||||
pub mod rhai_fn_increment {
|
|
||||||
use super::*;
|
|
||||||
struct Token();
|
|
||||||
impl PluginFunction for Token {
|
|
||||||
fn call(&self,
|
|
||||||
args: &mut [&mut Dynamic], pos: Position
|
|
||||||
) -> Result<Dynamic, Box<EvalAltResult>> {
|
|
||||||
debug_assert_eq!(args.len(), 2usize,
|
|
||||||
"wrong arg count: {} != {}", args.len(), 2usize);
|
|
||||||
let arg1 = mem::take(args[1usize]).clone().cast::<usize>();
|
|
||||||
let arg0: &mut _ = &mut args[0usize].write_lock::<usize>().unwrap();
|
|
||||||
Ok(Dynamic::from(increment(arg0, arg1)))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn is_method_call(&self) -> bool { true }
|
|
||||||
fn is_varadic(&self) -> bool { false }
|
|
||||||
fn clone_boxed(&self) -> Box<dyn PluginFunction> { Box::new(Token()) }
|
|
||||||
fn input_types(&self) -> Box<[TypeId]> {
|
|
||||||
new_vec![TypeId::of::<usize>(),
|
|
||||||
TypeId::of::<usize>()].into_boxed_slice()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pub fn token_callable() -> CallableFunction {
|
|
||||||
CallableFunction::from_plugin(Token())
|
|
||||||
}
|
|
||||||
pub fn token_input_types() -> Box<[TypeId]> {
|
|
||||||
Token().input_types()
|
|
||||||
}
|
|
||||||
type EvalBox = Box<EvalAltResult>;
|
|
||||||
pub fn dynamic_result_fn(x: &mut usize, y: usize) -> Result<Dynamic, EvalBox> {
|
|
||||||
Ok(Dynamic::from(super::increment(x, y)))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert!(item_fn.mutable_receiver());
|
|
||||||
assert_streams_eq(item_fn.generate(), expected_tokens);
|
|
||||||
}
|
|
||||||
|
|
||||||
#[test]
|
|
||||||
fn str_arg_fn() {
|
|
||||||
let input_tokens: TokenStream = quote! {
|
|
||||||
pub fn special_print(message: &str) { eprintln!("----{}----", message); }
|
|
||||||
};
|
|
||||||
|
|
||||||
let expected_tokens = quote! {
|
|
||||||
#[allow(unused)]
|
|
||||||
pub mod rhai_fn_special_print {
|
|
||||||
use super::*;
|
|
||||||
struct Token();
|
|
||||||
impl PluginFunction for Token {
|
|
||||||
fn call(&self,
|
|
||||||
args: &mut [&mut Dynamic], pos: Position
|
|
||||||
) -> Result<Dynamic, Box<EvalAltResult>> {
|
|
||||||
debug_assert_eq!(args.len(), 1usize,
|
|
||||||
"wrong arg count: {} != {}", args.len(), 1usize);
|
|
||||||
let arg0 = mem::take(args[0usize]).clone().cast::<ImmutableString>();
|
|
||||||
Ok(Dynamic::from(special_print(&arg0)))
|
|
||||||
}
|
|
||||||
|
|
||||||
fn is_method_call(&self) -> bool { false }
|
|
||||||
fn is_varadic(&self) -> bool { false }
|
|
||||||
fn clone_boxed(&self) -> Box<dyn PluginFunction> { Box::new(Token()) }
|
|
||||||
fn input_types(&self) -> Box<[TypeId]> {
|
|
||||||
new_vec![TypeId::of::<ImmutableString>()].into_boxed_slice()
|
|
||||||
}
|
|
||||||
}
|
|
||||||
pub fn token_callable() -> CallableFunction {
|
|
||||||
CallableFunction::from_plugin(Token())
|
|
||||||
}
|
|
||||||
pub fn token_input_types() -> Box<[TypeId]> {
|
|
||||||
Token().input_types()
|
|
||||||
}
|
|
||||||
type EvalBox = Box<EvalAltResult>;
|
|
||||||
pub fn dynamic_result_fn(message: &str) -> Result<Dynamic, EvalBox> {
|
|
||||||
Ok(Dynamic::from(super::special_print(message)))
|
|
||||||
}
|
|
||||||
}
|
|
||||||
};
|
|
||||||
|
|
||||||
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
|
||||||
assert!(!item_fn.mutable_receiver());
|
|
||||||
assert_streams_eq(item_fn.generate(), expected_tokens);
|
|
||||||
}
|
|
||||||
}
|
|
||||||
|
@ -102,6 +102,9 @@ mod module;
|
|||||||
mod register;
|
mod register;
|
||||||
mod rhai_module;
|
mod rhai_module;
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod test;
|
||||||
|
|
||||||
#[proc_macro_attribute]
|
#[proc_macro_attribute]
|
||||||
pub fn export_fn(
|
pub fn export_fn(
|
||||||
args: proc_macro::TokenStream,
|
args: proc_macro::TokenStream,
|
||||||
|
File diff suppressed because it is too large
Load Diff
518
codegen/src/test/function.rs
Normal file
518
codegen/src/test/function.rs
Normal file
@ -0,0 +1,518 @@
|
|||||||
|
#[cfg(test)]
|
||||||
|
mod function_tests {
|
||||||
|
use crate::function::ExportedFn;
|
||||||
|
|
||||||
|
use proc_macro2::TokenStream;
|
||||||
|
use quote::quote;
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn minimal_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn do_nothing() { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert_eq!(&item_fn.name().to_string(), "do_nothing");
|
||||||
|
assert!(!item_fn.mutable_receiver());
|
||||||
|
assert!(item_fn.is_public());
|
||||||
|
assert!(item_fn.return_type().is_none());
|
||||||
|
assert_eq!(item_fn.arg_list().count(), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn one_arg_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn do_something(x: usize) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert_eq!(&item_fn.name().to_string(), "do_something");
|
||||||
|
assert_eq!(item_fn.arg_list().count(), 1);
|
||||||
|
assert!(!item_fn.mutable_receiver());
|
||||||
|
assert!(item_fn.is_public());
|
||||||
|
assert!(item_fn.return_type().is_none());
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
item_fn.arg_list().next().unwrap(),
|
||||||
|
&syn::parse2::<syn::FnArg>(quote! { x: usize }).unwrap()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn two_arg_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn do_something(x: usize, y: f32) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert_eq!(&item_fn.name().to_string(), "do_something");
|
||||||
|
assert_eq!(item_fn.arg_list().count(), 2);
|
||||||
|
assert!(!item_fn.mutable_receiver());
|
||||||
|
assert!(item_fn.is_public());
|
||||||
|
assert!(item_fn.return_type().is_none());
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
item_fn.arg_list().next().unwrap(),
|
||||||
|
&syn::parse2::<syn::FnArg>(quote! { x: usize }).unwrap()
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
item_fn.arg_list().nth(1).unwrap(),
|
||||||
|
&syn::parse2::<syn::FnArg>(quote! { y: f32 }).unwrap()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn usize_returning_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn get_magic_number() -> usize { 42 }
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert_eq!(&item_fn.name().to_string(), "get_magic_number");
|
||||||
|
assert!(!item_fn.mutable_receiver());
|
||||||
|
assert!(item_fn.is_public());
|
||||||
|
assert_eq!(item_fn.arg_list().count(), 0);
|
||||||
|
assert_eq!(
|
||||||
|
item_fn.return_type().unwrap(),
|
||||||
|
&syn::Type::Path(syn::TypePath {
|
||||||
|
qself: None,
|
||||||
|
path: syn::parse2::<syn::Path>(quote! { usize }).unwrap()
|
||||||
|
})
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn ref_returning_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn get_magic_phrase() -> &'static str { "open sesame" }
|
||||||
|
};
|
||||||
|
|
||||||
|
let err = syn::parse2::<ExportedFn>(input_tokens).unwrap_err();
|
||||||
|
assert_eq!(format!("{}", err), "cannot return a reference to Rhai");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn ptr_returning_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn get_magic_phrase() -> *const str { "open sesame" }
|
||||||
|
};
|
||||||
|
|
||||||
|
let err = syn::parse2::<ExportedFn>(input_tokens).unwrap_err();
|
||||||
|
assert_eq!(format!("{}", err), "cannot return a pointer to Rhai");
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn ref_arg_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn greet(who: &Person) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let err = syn::parse2::<ExportedFn>(input_tokens).unwrap_err();
|
||||||
|
assert_eq!(
|
||||||
|
format!("{}", err),
|
||||||
|
"references from Rhai in this position must be mutable"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn ref_second_arg_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn greet(count: usize, who: &Person) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let err = syn::parse2::<ExportedFn>(input_tokens).unwrap_err();
|
||||||
|
assert_eq!(
|
||||||
|
format!("{}", err),
|
||||||
|
"this type in this position passes from Rhai by value"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn mut_ref_second_arg_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn give(item_name: &str, who: &mut Person) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let err = syn::parse2::<ExportedFn>(input_tokens).unwrap_err();
|
||||||
|
assert_eq!(
|
||||||
|
format!("{}", err),
|
||||||
|
"this type in this position passes from Rhai by value"
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn str_arg_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn log(message: &str) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert_eq!(&item_fn.name().to_string(), "log");
|
||||||
|
assert_eq!(item_fn.arg_list().count(), 1);
|
||||||
|
assert!(!item_fn.mutable_receiver());
|
||||||
|
assert!(item_fn.is_public());
|
||||||
|
assert!(item_fn.return_type().is_none());
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
item_fn.arg_list().next().unwrap(),
|
||||||
|
&syn::parse2::<syn::FnArg>(quote! { message: &str }).unwrap()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn str_second_arg_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn log(level: usize, message: &str) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert_eq!(&item_fn.name().to_string(), "log");
|
||||||
|
assert_eq!(item_fn.arg_list().count(), 2);
|
||||||
|
assert!(!item_fn.mutable_receiver());
|
||||||
|
assert!(item_fn.is_public());
|
||||||
|
assert!(item_fn.return_type().is_none());
|
||||||
|
|
||||||
|
assert_eq!(
|
||||||
|
item_fn.arg_list().next().unwrap(),
|
||||||
|
&syn::parse2::<syn::FnArg>(quote! { level: usize }).unwrap()
|
||||||
|
);
|
||||||
|
assert_eq!(
|
||||||
|
item_fn.arg_list().nth(1).unwrap(),
|
||||||
|
&syn::parse2::<syn::FnArg>(quote! { message: &str }).unwrap()
|
||||||
|
);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn private_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
fn do_nothing() { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert_eq!(&item_fn.name().to_string(), "do_nothing");
|
||||||
|
assert!(!item_fn.mutable_receiver());
|
||||||
|
assert!(!item_fn.is_public());
|
||||||
|
assert!(item_fn.return_type().is_none());
|
||||||
|
assert_eq!(item_fn.arg_list().count(), 0);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn receiver_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn act_upon(&mut self) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert_eq!(&item_fn.name().to_string(), "act_upon");
|
||||||
|
assert!(item_fn.mutable_receiver());
|
||||||
|
assert!(item_fn.is_public());
|
||||||
|
assert!(item_fn.return_type().is_none());
|
||||||
|
assert_eq!(item_fn.arg_list().count(), 1);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn immutable_receiver_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn act_upon(&self) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert_eq!(&item_fn.name().to_string(), "act_upon");
|
||||||
|
assert!(item_fn.mutable_receiver());
|
||||||
|
assert!(item_fn.is_public());
|
||||||
|
assert!(item_fn.return_type().is_none());
|
||||||
|
assert_eq!(item_fn.arg_list().count(), 1);
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
#[cfg(test)]
|
||||||
|
mod generate_tests {
|
||||||
|
use crate::function::ExportedFn;
|
||||||
|
|
||||||
|
use proc_macro2::TokenStream;
|
||||||
|
use quote::quote;
|
||||||
|
|
||||||
|
fn assert_streams_eq(actual: TokenStream, expected: TokenStream) {
|
||||||
|
let actual = actual.to_string();
|
||||||
|
let expected = expected.to_string();
|
||||||
|
if &actual != &expected {
|
||||||
|
let mut counter = 0;
|
||||||
|
let iter = actual
|
||||||
|
.chars()
|
||||||
|
.zip(expected.chars())
|
||||||
|
.inspect(|_| counter += 1)
|
||||||
|
.skip_while(|(a, e)| *a == *e);
|
||||||
|
let (actual_diff, expected_diff) = {
|
||||||
|
let mut actual_diff = String::new();
|
||||||
|
let mut expected_diff = String::new();
|
||||||
|
for (a, e) in iter.take(50) {
|
||||||
|
actual_diff.push(a);
|
||||||
|
expected_diff.push(e);
|
||||||
|
}
|
||||||
|
(actual_diff, expected_diff)
|
||||||
|
};
|
||||||
|
eprintln!("actual != expected, diverge at char {}", counter);
|
||||||
|
}
|
||||||
|
assert_eq!(actual, expected);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn minimal_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn do_nothing() { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let expected_tokens = quote! {
|
||||||
|
#[allow(unused)]
|
||||||
|
pub mod rhai_fn_do_nothing {
|
||||||
|
use super::*;
|
||||||
|
struct Token();
|
||||||
|
impl PluginFunction for Token {
|
||||||
|
fn call(&self,
|
||||||
|
args: &mut [&mut Dynamic], pos: Position
|
||||||
|
) -> Result<Dynamic, Box<EvalAltResult>> {
|
||||||
|
debug_assert_eq!(args.len(), 0usize,
|
||||||
|
"wrong arg count: {} != {}", args.len(), 0usize);
|
||||||
|
Ok(Dynamic::from(do_nothing()))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_method_call(&self) -> bool { false }
|
||||||
|
fn is_varadic(&self) -> bool { false }
|
||||||
|
fn clone_boxed(&self) -> Box<dyn PluginFunction> { Box::new(Token()) }
|
||||||
|
fn input_types(&self) -> Box<[TypeId]> {
|
||||||
|
new_vec![].into_boxed_slice()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn token_callable() -> CallableFunction {
|
||||||
|
CallableFunction::from_plugin(Token())
|
||||||
|
}
|
||||||
|
pub fn token_input_types() -> Box<[TypeId]> {
|
||||||
|
Token().input_types()
|
||||||
|
}
|
||||||
|
type EvalBox = Box<EvalAltResult>;
|
||||||
|
pub fn dynamic_result_fn() -> Result<Dynamic, EvalBox> {
|
||||||
|
Ok(Dynamic::from(super::do_nothing()))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert_streams_eq(item_fn.generate(), expected_tokens);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn one_arg_usize_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn do_something(x: usize) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let expected_tokens = quote! {
|
||||||
|
#[allow(unused)]
|
||||||
|
pub mod rhai_fn_do_something {
|
||||||
|
use super::*;
|
||||||
|
struct Token();
|
||||||
|
impl PluginFunction for Token {
|
||||||
|
fn call(&self,
|
||||||
|
args: &mut [&mut Dynamic], pos: Position
|
||||||
|
) -> Result<Dynamic, Box<EvalAltResult>> {
|
||||||
|
debug_assert_eq!(args.len(), 1usize,
|
||||||
|
"wrong arg count: {} != {}", args.len(), 1usize);
|
||||||
|
let arg0 = mem::take(args[0usize]).clone().cast::<usize>();
|
||||||
|
Ok(Dynamic::from(do_something(arg0)))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_method_call(&self) -> bool { false }
|
||||||
|
fn is_varadic(&self) -> bool { false }
|
||||||
|
fn clone_boxed(&self) -> Box<dyn PluginFunction> { Box::new(Token()) }
|
||||||
|
fn input_types(&self) -> Box<[TypeId]> {
|
||||||
|
new_vec![TypeId::of::<usize>()].into_boxed_slice()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn token_callable() -> CallableFunction {
|
||||||
|
CallableFunction::from_plugin(Token())
|
||||||
|
}
|
||||||
|
pub fn token_input_types() -> Box<[TypeId]> {
|
||||||
|
Token().input_types()
|
||||||
|
}
|
||||||
|
type EvalBox = Box<EvalAltResult>;
|
||||||
|
pub fn dynamic_result_fn(x: usize) -> Result<Dynamic, EvalBox> {
|
||||||
|
Ok(Dynamic::from(super::do_something(x)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert_streams_eq(item_fn.generate(), expected_tokens);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn one_arg_usize_fn_impl() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn do_something(x: usize) { }
|
||||||
|
};
|
||||||
|
|
||||||
|
let expected_tokens = quote! {
|
||||||
|
impl PluginFunction for MyType {
|
||||||
|
fn call(&self,
|
||||||
|
args: &mut [&mut Dynamic], pos: Position
|
||||||
|
) -> Result<Dynamic, Box<EvalAltResult>> {
|
||||||
|
debug_assert_eq!(args.len(), 1usize,
|
||||||
|
"wrong arg count: {} != {}", args.len(), 1usize);
|
||||||
|
let arg0 = mem::take(args[0usize]).clone().cast::<usize>();
|
||||||
|
Ok(Dynamic::from(do_something(arg0)))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_method_call(&self) -> bool { false }
|
||||||
|
fn is_varadic(&self) -> bool { false }
|
||||||
|
fn clone_boxed(&self) -> Box<dyn PluginFunction> { Box::new(MyType()) }
|
||||||
|
fn input_types(&self) -> Box<[TypeId]> {
|
||||||
|
new_vec![TypeId::of::<usize>()].into_boxed_slice()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert_streams_eq(item_fn.generate_impl("MyType"), expected_tokens);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn two_arg_returning_usize_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn add_together(x: usize, y: usize) -> usize { x + y }
|
||||||
|
};
|
||||||
|
|
||||||
|
let expected_tokens = quote! {
|
||||||
|
#[allow(unused)]
|
||||||
|
pub mod rhai_fn_add_together {
|
||||||
|
use super::*;
|
||||||
|
struct Token();
|
||||||
|
impl PluginFunction for Token {
|
||||||
|
fn call(&self,
|
||||||
|
args: &mut [&mut Dynamic], pos: Position
|
||||||
|
) -> Result<Dynamic, Box<EvalAltResult>> {
|
||||||
|
debug_assert_eq!(args.len(), 2usize,
|
||||||
|
"wrong arg count: {} != {}", args.len(), 2usize);
|
||||||
|
let arg0 = mem::take(args[0usize]).clone().cast::<usize>();
|
||||||
|
let arg1 = mem::take(args[1usize]).clone().cast::<usize>();
|
||||||
|
Ok(Dynamic::from(add_together(arg0, arg1)))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_method_call(&self) -> bool { false }
|
||||||
|
fn is_varadic(&self) -> bool { false }
|
||||||
|
fn clone_boxed(&self) -> Box<dyn PluginFunction> { Box::new(Token()) }
|
||||||
|
fn input_types(&self) -> Box<[TypeId]> {
|
||||||
|
new_vec![TypeId::of::<usize>(),
|
||||||
|
TypeId::of::<usize>()].into_boxed_slice()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn token_callable() -> CallableFunction {
|
||||||
|
CallableFunction::from_plugin(Token())
|
||||||
|
}
|
||||||
|
pub fn token_input_types() -> Box<[TypeId]> {
|
||||||
|
Token().input_types()
|
||||||
|
}
|
||||||
|
type EvalBox = Box<EvalAltResult>;
|
||||||
|
pub fn dynamic_result_fn(x: usize, y: usize) -> Result<Dynamic, EvalBox> {
|
||||||
|
Ok(Dynamic::from(super::add_together(x, y)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert_streams_eq(item_fn.generate(), expected_tokens);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn mut_arg_usize_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn increment(x: &mut usize, y: usize) { *x += y; }
|
||||||
|
};
|
||||||
|
|
||||||
|
let expected_tokens = quote! {
|
||||||
|
#[allow(unused)]
|
||||||
|
pub mod rhai_fn_increment {
|
||||||
|
use super::*;
|
||||||
|
struct Token();
|
||||||
|
impl PluginFunction for Token {
|
||||||
|
fn call(&self,
|
||||||
|
args: &mut [&mut Dynamic], pos: Position
|
||||||
|
) -> Result<Dynamic, Box<EvalAltResult>> {
|
||||||
|
debug_assert_eq!(args.len(), 2usize,
|
||||||
|
"wrong arg count: {} != {}", args.len(), 2usize);
|
||||||
|
let arg1 = mem::take(args[1usize]).clone().cast::<usize>();
|
||||||
|
let arg0: &mut _ = &mut args[0usize].write_lock::<usize>().unwrap();
|
||||||
|
Ok(Dynamic::from(increment(arg0, arg1)))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_method_call(&self) -> bool { true }
|
||||||
|
fn is_varadic(&self) -> bool { false }
|
||||||
|
fn clone_boxed(&self) -> Box<dyn PluginFunction> { Box::new(Token()) }
|
||||||
|
fn input_types(&self) -> Box<[TypeId]> {
|
||||||
|
new_vec![TypeId::of::<usize>(),
|
||||||
|
TypeId::of::<usize>()].into_boxed_slice()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn token_callable() -> CallableFunction {
|
||||||
|
CallableFunction::from_plugin(Token())
|
||||||
|
}
|
||||||
|
pub fn token_input_types() -> Box<[TypeId]> {
|
||||||
|
Token().input_types()
|
||||||
|
}
|
||||||
|
type EvalBox = Box<EvalAltResult>;
|
||||||
|
pub fn dynamic_result_fn(x: &mut usize, y: usize) -> Result<Dynamic, EvalBox> {
|
||||||
|
Ok(Dynamic::from(super::increment(x, y)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert!(item_fn.mutable_receiver());
|
||||||
|
assert_streams_eq(item_fn.generate(), expected_tokens);
|
||||||
|
}
|
||||||
|
|
||||||
|
#[test]
|
||||||
|
fn str_arg_fn() {
|
||||||
|
let input_tokens: TokenStream = quote! {
|
||||||
|
pub fn special_print(message: &str) { eprintln!("----{}----", message); }
|
||||||
|
};
|
||||||
|
|
||||||
|
let expected_tokens = quote! {
|
||||||
|
#[allow(unused)]
|
||||||
|
pub mod rhai_fn_special_print {
|
||||||
|
use super::*;
|
||||||
|
struct Token();
|
||||||
|
impl PluginFunction for Token {
|
||||||
|
fn call(&self,
|
||||||
|
args: &mut [&mut Dynamic], pos: Position
|
||||||
|
) -> Result<Dynamic, Box<EvalAltResult>> {
|
||||||
|
debug_assert_eq!(args.len(), 1usize,
|
||||||
|
"wrong arg count: {} != {}", args.len(), 1usize);
|
||||||
|
let arg0 = mem::take(args[0usize]).clone().cast::<ImmutableString>();
|
||||||
|
Ok(Dynamic::from(special_print(&arg0)))
|
||||||
|
}
|
||||||
|
|
||||||
|
fn is_method_call(&self) -> bool { false }
|
||||||
|
fn is_varadic(&self) -> bool { false }
|
||||||
|
fn clone_boxed(&self) -> Box<dyn PluginFunction> { Box::new(Token()) }
|
||||||
|
fn input_types(&self) -> Box<[TypeId]> {
|
||||||
|
new_vec![TypeId::of::<ImmutableString>()].into_boxed_slice()
|
||||||
|
}
|
||||||
|
}
|
||||||
|
pub fn token_callable() -> CallableFunction {
|
||||||
|
CallableFunction::from_plugin(Token())
|
||||||
|
}
|
||||||
|
pub fn token_input_types() -> Box<[TypeId]> {
|
||||||
|
Token().input_types()
|
||||||
|
}
|
||||||
|
type EvalBox = Box<EvalAltResult>;
|
||||||
|
pub fn dynamic_result_fn(message: &str) -> Result<Dynamic, EvalBox> {
|
||||||
|
Ok(Dynamic::from(super::special_print(message)))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
};
|
||||||
|
|
||||||
|
let item_fn = syn::parse2::<ExportedFn>(input_tokens).unwrap();
|
||||||
|
assert!(!item_fn.mutable_receiver());
|
||||||
|
assert_streams_eq(item_fn.generate(), expected_tokens);
|
||||||
|
}
|
||||||
|
}
|
2
codegen/src/test/mod.rs
Normal file
2
codegen/src/test/mod.rs
Normal file
@ -0,0 +1,2 @@
|
|||||||
|
mod function;
|
||||||
|
mod module;
|
1243
codegen/src/test/module.rs
Normal file
1243
codegen/src/test/module.rs
Normal file
File diff suppressed because it is too large
Load Diff
@ -1,5 +1,5 @@
|
|||||||
error[E0277]: the trait bound `NonClonable: std::clone::Clone` is not satisfied
|
error[E0277]: the trait bound `NonClonable: Clone` is not satisfied
|
||||||
--> $DIR/non_clonable.rs:11:23
|
--> $DIR/non_clonable.rs:11:23
|
||||||
|
|
|
|
||||||
11 | pub fn test_fn(input: NonClonable) -> bool {
|
11 | pub fn test_fn(input: NonClonable) -> bool {
|
||||||
| ^^^^^^^^^^^ the trait `std::clone::Clone` is not implemented for `NonClonable`
|
| ^^^^^^^^^^^ the trait `Clone` is not implemented for `NonClonable`
|
||||||
|
@ -1,5 +1,5 @@
|
|||||||
error[E0277]: the trait bound `NonClonable: std::clone::Clone` is not satisfied
|
error[E0277]: the trait bound `NonClonable: Clone` is not satisfied
|
||||||
--> $DIR/non_clonable_second.rs:11:27
|
--> $DIR/non_clonable_second.rs:11:27
|
||||||
|
|
|
|
||||||
11 | pub fn test_fn(a: u32, b: NonClonable) -> bool {
|
11 | pub fn test_fn(a: u32, b: NonClonable) -> bool {
|
||||||
| ^^^^^^^^^^^ the trait `std::clone::Clone` is not implemented for `NonClonable`
|
| ^^^^^^^^^^^ the trait `Clone` is not implemented for `NonClonable`
|
||||||
|
Loading…
Reference in New Issue
Block a user