Skip to content

Commit 3c2fb9c

Browse files
authored
Properly handle mutability for awaited futures (#239)
* Properly handle mutability for awaited futures * Simplify and add tests * Implement review
1 parent 61a7007 commit 3c2fb9c

File tree

6 files changed

+106
-6
lines changed

6 files changed

+106
-6
lines changed

CHANGELOG.md

+2
Original file line numberDiff line numberDiff line change
@@ -12,6 +12,8 @@
1212
[`Sync`](https://doc.rust-lang.org/std/marker/trait.Sync.html) to prevent UB
1313
when tests are executed in parallel. (see [#235](https://github.com/la10736/rstest/issues/235)
1414
for more details)
15+
- `#[future(awt)]` and `#[awt]` now properly handle mutable (`mut`) parameters by treating futures as immutable and
16+
treating the awaited rebinding as mutable.
1517

1618
## [0.18.2] 2023/8/13
1719

Original file line numberDiff line numberDiff line change
@@ -0,0 +1,24 @@
1+
use rstest::*;
2+
3+
#[rstest]
4+
#[case::pass(async { 3 })]
5+
#[awt]
6+
async fn my_mut_test_global_awt(
7+
#[future]
8+
#[case]
9+
mut a: i32,
10+
) {
11+
a = 4;
12+
assert_eq!(a, 4);
13+
}
14+
15+
#[rstest]
16+
#[case::pass(async { 3 })]
17+
async fn my_mut_test_local_awt(
18+
#[future(awt)]
19+
#[case]
20+
mut a: i32,
21+
) {
22+
a = 4;
23+
assert_eq!(a, 4);
24+
}

rstest/tests/rstest/mod.rs

+13
Original file line numberDiff line numberDiff line change
@@ -534,6 +534,19 @@ mod cases {
534534
.assert(output);
535535
}
536536

537+
#[rstest]
538+
fn should_run_async_mut() {
539+
let prj = prj(res("async_awt_mut.rs"));
540+
prj.add_dependency("async-std", r#"{version="*", features=["attributes"]}"#);
541+
542+
let output = prj.run_tests().unwrap();
543+
544+
TestResults::new()
545+
.ok("my_mut_test_global_awt::case_1_pass")
546+
.ok("my_mut_test_local_awt::case_1_pass")
547+
.assert(output);
548+
}
549+
537550
#[test]
538551
fn should_use_injected_test_attr() {
539552
let prj = prj(res("inject.rs"));

rstest_macros/src/refident.rs

+32
Original file line numberDiff line numberDiff line change
@@ -86,3 +86,35 @@ impl MaybeIdent for crate::parse::Attribute {
8686
}
8787
}
8888
}
89+
90+
pub trait MaybePatIdent {
91+
fn maybe_patident(&self) -> Option<&syn::PatIdent>;
92+
}
93+
94+
impl MaybePatIdent for FnArg {
95+
fn maybe_patident(&self) -> Option<&syn::PatIdent> {
96+
match self {
97+
FnArg::Typed(PatType { pat, .. }) => match pat.as_ref() {
98+
Pat::Ident(ident) => Some(ident),
99+
_ => None,
100+
},
101+
_ => None,
102+
}
103+
}
104+
}
105+
106+
pub trait RemoveMutability {
107+
fn remove_mutability(&mut self);
108+
}
109+
110+
impl RemoveMutability for FnArg {
111+
fn remove_mutability(&mut self) {
112+
match self {
113+
FnArg::Typed(PatType { pat, .. }) => match pat.as_mut() {
114+
Pat::Ident(ident) => ident.mutability = None,
115+
_ => {}
116+
},
117+
_ => {}
118+
};
119+
}
120+
}

rstest_macros/src/render/apply_argumets.rs

+27-6
Original file line numberDiff line numberDiff line change
@@ -3,7 +3,7 @@ use syn::{parse_quote, FnArg, Generics, Ident, ItemFn, Lifetime, Signature, Type
33

44
use crate::{
55
parse::{arguments::ArgumentsInfo, future::MaybeFutureImplType},
6-
refident::MaybeIdent,
6+
refident::{MaybeIdent, MaybePatIdent, RemoveMutability},
77
};
88

99
pub(crate) trait ApplyArgumets<R: Sized = ()> {
@@ -59,17 +59,20 @@ impl ApplyArgumets for Signature {
5959

6060
impl ApplyArgumets for ItemFn {
6161
fn apply_argumets(&mut self, arguments: &ArgumentsInfo) {
62-
let awaited_args = self
62+
let rebound_awaited_args = self
6363
.sig
6464
.inputs
6565
.iter()
66-
.filter_map(|a| a.maybe_ident())
67-
.filter(|&a| arguments.is_future_await(a))
68-
.cloned();
66+
.filter_map(|a| a.maybe_patident())
67+
.filter(|p| arguments.is_future_await(&p.ident))
68+
.map(|p| {
69+
let a = &p.ident;
70+
quote::quote! { let #p = #a.await; }
71+
});
6972
let orig_block_impl = self.block.clone();
7073
self.block = parse_quote! {
7174
{
72-
#(let #awaited_args = #awaited_args.await;)*
75+
#(#rebound_awaited_args)*
7376
#orig_block_impl
7477
}
7578
};
@@ -90,6 +93,7 @@ impl ImplFutureArg for FnArg {
9093
*ty = parse_quote! {
9194
impl std::future::Future<Output = #ty>
9295
};
96+
self.remove_mutability();
9397
lifetime
9498
}
9599
None => None,
@@ -154,6 +158,11 @@ mod should {
154158
&["a"],
155159
"fn f<S: AsRef<str>>(a: impl std::future::Future<Output = S>) {}"
156160
)]
161+
#[case::remove_mut(
162+
"fn f(mut a: u32) {}",
163+
&["a"],
164+
r#"fn f(a: impl std::future::Future<Output = u32>) {}"#
165+
)]
157166
fn replace_future_basic_type(
158167
#[case] item_fn: &str,
159168
#[case] futures: &[&str],
@@ -245,5 +254,17 @@ mod should {
245254
assert_in!(code, await_argument_code_string("b"));
246255
assert_not_in!(code, await_argument_code_string("c"));
247256
}
257+
258+
#[test]
259+
fn with_mut_await() {
260+
let mut item_fn: ItemFn = r#"fn test(mut a: i32) {} "#.ast();
261+
let mut arguments: ArgumentsInfo = Default::default();
262+
arguments.set_future(ident("a"), FutureArg::Await);
263+
264+
item_fn.apply_argumets(&arguments);
265+
266+
let code = item_fn.block.display_code();
267+
assert_in!(code, mut_await_argument_code_string("a"));
268+
}
248269
}
249270
}

rstest_macros/src/test.rs

+8
Original file line numberDiff line numberDiff line change
@@ -319,3 +319,11 @@ pub(crate) fn await_argument_code_string(arg_name: &str) -> String {
319319
};
320320
statment.display_code()
321321
}
322+
323+
pub(crate) fn mut_await_argument_code_string(arg_name: &str) -> String {
324+
let arg_name = ident(arg_name);
325+
let statement: Stmt = parse_quote! {
326+
let mut #arg_name = #arg_name.await;
327+
};
328+
statement.display_code()
329+
}

0 commit comments

Comments
 (0)