Skip to content

Commit

Permalink
Merge #164
Browse files Browse the repository at this point in the history
164: Inject scene tree into `#[itest]` r=Bromeon a=Bromeon

Also adds another test for `Gd::eq()` in the case of dead instances, and a stub for testing #23.

Simplifies the proc-macro machinery further.

bors r+

Co-authored-by: Jan Haller <[email protected]>
  • Loading branch information
bors[bot] and Bromeon authored Mar 9, 2023
2 parents ac5f78c + ad5e62c commit 5c17c4e
Show file tree
Hide file tree
Showing 9 changed files with 189 additions and 53 deletions.
5 changes: 5 additions & 0 deletions godot-core/src/obj/gd.rs
Original file line number Diff line number Diff line change
Expand Up @@ -671,3 +671,8 @@ impl<T: GodotClass> VariantMetadata for Gd<T> {
ClassName::of::<T>()
}
}

// Gd unwinding across panics does not invalidate any invariants;
// its mutability is anyway present, in the Godot engine.
impl<T: GodotClass> std::panic::UnwindSafe for Gd<T> {}
impl<T: GodotClass> std::panic::RefUnwindSafe for Gd<T> {}
22 changes: 8 additions & 14 deletions godot-macros/src/gdextension.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,12 +4,14 @@
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
*/

use crate::util::{bail, ident, parse_kv_group, path_is_single, validate_impl, KvValue};
use proc_macro2::TokenStream;
use quote::quote;
use venial::Declaration;

pub fn transform(decl: Declaration) -> Result<TokenStream, venial::Error> {
use crate::util::{bail, ident, validate_impl, KvParser};
use crate::ParseResult;

pub fn transform(decl: Declaration) -> ParseResult<TokenStream> {
let mut impl_decl = match decl {
Declaration::Impl(item) => item,
_ => return bail("#[gdextension] can only be applied to trait impls", &decl),
Expand All @@ -23,18 +25,10 @@ pub fn transform(decl: Declaration) -> Result<TokenStream, venial::Error> {
);
}

let mut entry_point = None;
for attr in impl_decl.attributes.drain(..) {
if path_is_single(&attr.path, "gdextension") {
for (k, v) in parse_kv_group(&attr.value).expect("#[gdextension] has invalid arguments")
{
match (k.as_str(), v) {
("entry_point", KvValue::Ident(f)) => entry_point = Some(f),
_ => return bail(format!("#[gdextension]: invalid argument `{k}`"), attr),
}
}
}
}
let drained_attributes = std::mem::take(&mut impl_decl.attributes);
let mut parser = KvParser::parse_required(&drained_attributes, "gdextension", &impl_decl)?;
let entry_point = parser.handle_ident("entry_point")?;
parser.finish()?;

let entry_point = entry_point.unwrap_or_else(|| ident("gdextension_rust_init"));
let impl_ty = &impl_decl.self_ty;
Expand Down
47 changes: 37 additions & 10 deletions godot-macros/src/itest.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,11 +4,12 @@
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
*/

use crate::util::{bail, KvParser};
use crate::ParseResult;
use proc_macro2::TokenStream;
use quote::quote;
use venial::Declaration;
use quote::{quote, ToTokens};
use venial::{Declaration, Error, FnParam, Function};

use crate::util::{bail, path_ends_with, KvParser};
use crate::ParseResult;

pub fn transform(input_decl: Declaration) -> ParseResult<TokenStream> {
let func = match input_decl {
Expand All @@ -18,14 +19,11 @@ pub fn transform(input_decl: Declaration) -> ParseResult<TokenStream> {

// Note: allow attributes for things like #[rustfmt] or #[clippy]
if func.generic_params.is_some()
|| !func.params.is_empty()
|| func.params.len() > 1
|| func.return_ty.is_some()
|| func.where_clause.is_some()
{
return bail(
format!("#[itest] must be of form: fn {}() {{ ... }}", func.name),
&func,
);
return bad_signature(&func);
}

let mut attr = KvParser::parse_required(&func.attributes, "itest", &func.name)?;
Expand All @@ -42,10 +40,27 @@ pub fn transform(input_decl: Declaration) -> ParseResult<TokenStream> {

let test_name = &func.name;
let test_name_str = func.name.to_string();

// Detect parameter name chosen by user, or unused fallback
let param = if let Some((param, _punct)) = func.params.first() {
if let FnParam::Typed(param) = param {
// Correct parameter type (crude macro check) -> reuse parameter name
if path_ends_with(&param.ty.tokens, "TestContext") {
param.to_token_stream()
} else {
return bad_signature(&func);
}
} else {
return bad_signature(&func);
}
} else {
quote! { __unused_context: &crate::TestContext }
};

let body = &func.body;

Ok(quote! {
pub fn #test_name() {
pub fn #test_name(#param) {
#body
}

Expand All @@ -59,3 +74,15 @@ pub fn transform(input_decl: Declaration) -> ParseResult<TokenStream> {
});
})
}

fn bad_signature(func: &Function) -> Result<TokenStream, Error> {
bail(
format!(
"#[itest] function must have one of these signatures:\
\n fn {f}() {{ ... }}\
\n fn {f}(ctx: &TestContext) {{ ... }}",
f = func.name
),
&func,
)
}
46 changes: 33 additions & 13 deletions godot-macros/src/util.rs
Original file line number Diff line number Diff line change
Expand Up @@ -67,24 +67,22 @@ pub(crate) struct KvParser {
finished: bool,
}

#[allow(dead_code)] // some functions will be used later
impl KvParser {
/// Create a new parser which requires a `#[expected]` attribute.
///
/// `context` is used for the span in error messages.
#[allow(dead_code)] // will be used later
pub fn parse_required(
attributes: &[Attribute],
expected: &str,
context: impl ToTokens,
) -> ParseResult<Self> {
match Self::parse(attributes, expected) {
Ok(Some(result)) => Ok(result),
Ok(None) => {
return bail(
format!("expected attribute #[{expected}], but not present"),
context,
)
}
Ok(None) => bail(
format!("expected attribute #[{expected}], but not present"),
context,
),
Err(e) => Err(e),
}
}
Expand Down Expand Up @@ -147,11 +145,26 @@ impl KvParser {
}
}

/// `#[attr(key="string", key2=123, key3=true)]`, with a given key being required
pub fn handle_ident_required(&mut self, key: &str) -> ParseResult<Ident> {
self.inner_required(key, "ident", Self::handle_ident)
}

/// `#[attr(key="string", key2=123, key3=true)]`, with a given key being required
pub fn handle_lit_required(&mut self, key: &str) -> ParseResult<String> {
match self.handle_lit(key) {
self.inner_required(key, "literal", Self::handle_lit)
}

fn inner_required<T, F>(&mut self, key: &str, context: &str, mut f: F) -> ParseResult<T>
where
F: FnMut(&mut Self, &str) -> ParseResult<Option<T>>,
{
match f(self, key) {
Ok(Some(string)) => Ok(string),
Ok(None) => self.bail_key(key, "expected to have literal value, but is absent"),
Ok(None) => self.bail_key(
key,
&format!("expected to have {context} value, but is absent"),
),
Err(err) => Err(err),
}
}
Expand All @@ -171,21 +184,21 @@ impl KvParser {
let keys = self.map.keys().cloned().collect::<Vec<_>>().join(", ");

let s = if self.map.len() > 1 { "s" } else { "" }; // plural
return bail(
bail(
format!(
"#[{attr}]: unrecognized key{s}: {keys}",
attr = self.attr_name
),
self.span,
);
)
}
}

fn bail_key<R>(&self, key: &str, msg: &str) -> ParseResult<R> {
return bail(
bail(
format!("#[{attr}]: key `{key}` {msg}", attr = self.attr_name),
self.span,
);
)
}
}

Expand Down Expand Up @@ -486,3 +499,10 @@ mod tests {
pub(crate) fn path_is_single(path: &Vec<TokenTree>, expected: &str) -> bool {
path.len() == 1 && path[0].to_string() == expected
}

pub(crate) fn path_ends_with(path: &Vec<TokenTree>, expected: &str) -> bool {
// could also use .as_path()
path.last()
.map(|last| last.to_string() == expected)
.unwrap_or(false)
}
6 changes: 3 additions & 3 deletions itest/godot/.godot/global_script_class_cache.cfg
Original file line number Diff line number Diff line change
@@ -1,13 +1,13 @@
list=[{
list=Array[Dictionary]([{
"base": &"RefCounted",
"class": &"TestStats",
"icon": "",
"language": &"GDScript",
"path": "res://TestStats.gd"
}, {
"base": &"Node",
"base": &"RefCounted",
"class": &"TestSuite",
"icon": "",
"language": &"GDScript",
"path": "res://TestSuite.gd"
}]
}])
7 changes: 6 additions & 1 deletion itest/godot/TestRunner.gd
Original file line number Diff line number Diff line change
Expand Up @@ -33,7 +33,12 @@ func _ready():
if method_name.begins_with("test_"):
gdscript_tests.push_back(GDScriptTestCase.new(suite, method_name))

var success: bool = rust_runner.run_all_tests(gdscript_tests, gdscript_suites.size(), allow_focus)
var success: bool = rust_runner.run_all_tests(
gdscript_tests,
gdscript_suites.size(),
allow_focus,
self,
)

var exit_code: int = 0 if success else 1
get_tree().quit(exit_code)
Expand Down
8 changes: 7 additions & 1 deletion itest/rust/src/lib.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,7 +4,9 @@
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
*/

use godot::engine::Node;
use godot::init::{gdextension, ExtensionLibrary};
use godot::obj::Gd;
use godot::sys;

mod array_test;
Expand Down Expand Up @@ -89,6 +91,10 @@ fn collect_rust_tests() -> (Vec<RustTestCase>, usize, bool) {
(tests, all_files.len(), is_focus_run)
}

pub struct TestContext {
scene_tree: Gd<Node>,
}

#[derive(Copy, Clone)]
struct RustTestCase {
name: &'static str,
Expand All @@ -98,5 +104,5 @@ struct RustTestCase {
focused: bool,
#[allow(dead_code)]
line: u32,
function: fn(),
function: fn(&TestContext),
}
79 changes: 76 additions & 3 deletions itest/rust/src/object_test.rs
Original file line number Diff line number Diff line change
Expand Up @@ -4,18 +4,21 @@
* file, You can obtain one at https://mozilla.org/MPL/2.0/.
*/

use crate::{expect_panic, itest};
use std::cell::RefCell;
use std::mem;
use std::rc::Rc;

use godot::bind::{godot_api, GodotClass, GodotExt};
use godot::builtin::{
FromVariant, GodotString, StringName, ToVariant, Variant, VariantConversionError, Vector3,
};
use godot::engine::node::InternalMode;
use godot::engine::{file_access, Area2D, Camera3D, FileAccess, Node, Node3D, Object, RefCounted};
use godot::obj::{Base, Gd, InstanceId};
use godot::obj::{Inherits, Share};
use godot::sys::GodotFfi;

use std::cell::RefCell;
use std::rc::Rc;
use crate::{expect_panic, itest, TestContext};

// TODO:
// * make sure that ptrcalls are used when possible (ie. when type info available; maybe GDScript integration test)
Expand All @@ -33,6 +36,39 @@ fn object_construct_value() {
assert_eq!(obj.bind().value, 222);
}

// TODO(#23): DerefMut on Gd pointer may be used to break subtyping relations
#[itest(skip)]
fn object_subtype_swap() {
let mut a: Gd<Node> = Node::new_alloc();
let mut b: Gd<Node3D> = Node3D::new_alloc();

/*
let a_id = a.instance_id();
let b_id = b.instance_id();
let a_class = a.get_class();
let b_class = b.get_class();
dbg!(a_id);
dbg!(b_id);
dbg!(&a_class);
dbg!(&b_class);
println!("..swap..");
*/

mem::swap(&mut *a, &mut *b);

/*
dbg!(a_id);
dbg!(b_id);
dbg!(&a_class);
dbg!(&b_class);
*/

// This should not panic
a.free();
b.free();
}

#[itest]
fn object_user_roundtrip_return() {
let value: i16 = 17943;
Expand Down Expand Up @@ -192,6 +228,32 @@ fn object_engine_eq() {
b1.free();
}

#[itest]
fn object_dead_eq() {
let a = Node3D::new_alloc();
let b = Node3D::new_alloc();

// Destroy b1 without consuming it
b.share().free();

{
let lhs = a.share();
let rhs = b.share();
expect_panic("Gd::eq() panics when one operand is dead", move || {
let _ = lhs == rhs;
});
}
{
let lhs = b.share();
let rhs = a.share();
expect_panic("Gd::ne() panics when one operand is dead", move || {
let _ = lhs != rhs;
});
}

a.free();
}

#[itest]
fn object_user_convert_variant() {
let value: i16 = 17943;
Expand Down Expand Up @@ -537,6 +599,17 @@ fn object_call_with_args() {
node.free();
}

#[itest]
fn object_get_scene_tree(ctx: &TestContext) {
let node = Node3D::new_alloc();

let mut tree = ctx.scene_tree.share();
tree.add_child(node.upcast(), false, InternalMode::INTERNAL_MODE_DISABLED);

let count = tree.get_child_count(false);
assert_eq!(count, 1);
} // implicitly tested: node does not leak

// ----------------------------------------------------------------------------------------------------------------------------------------------

#[inline(never)] // force to move "out of scope", can trigger potential dangling pointer errors
Expand Down
Loading

0 comments on commit 5c17c4e

Please sign in to comment.