From baa304cf6d8914dfd693683d82b9b61c0d9c757a Mon Sep 17 00:00:00 2001 From: Viet Dinh <54ckb0y789@gmail.com> Date: Sat, 20 Jan 2024 15:41:09 -0500 Subject: [PATCH] test: add fixtures to test function codegen --- Cargo.lock | 23 +++++++++++++ frb_codegen/Cargo.toml | 1 + .../library/codegen/generator/api_dart/mod.rs | 6 ++++ .../api_dart/spec_generator/class/field.rs | 22 ++++++++---- frb_codegen/src/library/codegen/mod.rs | 2 +- .../src/library/commands/cargo_expand.rs | 34 ++++++++++++------- .../src/library/commands/command_runner.rs | 2 +- .../src/library/integration/integrator.rs | 4 +-- frb_codegen/src/library/utils/test_utils.rs | 30 ++++++++++++---- .../api_dart/mod/functions/Cargo.toml | 11 ++++++ .../api_dart/mod/functions/expect_output.dart | 14 ++++++++ .../mod/functions/flutter_rust_bridge.yaml | 3 ++ .../api_dart/mod/functions/pubspec.yaml | 1 + .../api_dart/mod/functions/src/api.rs | 4 +++ .../api_dart/mod/functions/src/lib.rs | 1 + .../lib/src/rust/api/attribute.dart | 2 +- .../attribute_twin_rust_async.dart | 3 +- .../pseudo_manual/attribute_twin_sync.dart | 2 +- .../lib/src/rust/frb_generated.dart | 13 +++---- frb_macros/src/lib.rs | 20 ++++++----- justfile | 2 +- 21 files changed, 150 insertions(+), 50 deletions(-) create mode 100644 frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/Cargo.toml create mode 100644 frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/expect_output.dart create mode 100644 frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/flutter_rust_bridge.yaml create mode 100644 frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/pubspec.yaml create mode 100644 frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/src/api.rs create mode 100644 frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/src/lib.rs diff --git a/Cargo.lock b/Cargo.lock index d72441fe68..d082fc5e34 100644 --- a/Cargo.lock +++ b/Cargo.lock @@ -456,6 +456,12 @@ dependencies = [ "syn 1.0.109", ] +[[package]] +name = "diff" +version = "0.1.13" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "56254986775e3233ffa9c4d7d3faaf6d36a2c09d30b20687e9f88bc8bafc16c8" + [[package]] name = "either" version = "1.9.0" @@ -617,6 +623,7 @@ dependencies = [ "notify-debouncer-mini", "paste", "pathdiff", + "pretty_assertions", "proc-macro2", "quote", "regex", @@ -1146,6 +1153,16 @@ version = "1.6.0" source = "registry+https://github.com/rust-lang/crates.io-index" checksum = "7170ef9988bc169ba16dd36a7fa041e5c4cbeb6a35b76d4c03daded371eae7c0" +[[package]] +name = "pretty_assertions" +version = "1.4.0" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "af7cee1a6c8a5b9208b3cb1061f10c0cb689087b3d8ce85fb9d2dd7a29b6ba66" +dependencies = [ + "diff", + "yansi", +] + [[package]] name = "proc-macro2" version = "1.0.66" @@ -1853,3 +1870,9 @@ checksum = "56c1936c4cc7a1c9ab21a1ebb602eb942ba868cbd44a99cb7cdc5892335e1c85" dependencies = [ "linked-hash-map", ] + +[[package]] +name = "yansi" +version = "0.5.1" +source = "registry+https://github.com/rust-lang/crates.io-index" +checksum = "09041cd90cf85f7f8b2df60c646f853b7f535ce68f85244eb6731cf89fa498ec" diff --git a/frb_codegen/Cargo.toml b/frb_codegen/Cargo.toml index 87d1d34ed3..cd3bd50a1a 100644 --- a/frb_codegen/Cargo.toml +++ b/frb_codegen/Cargo.toml @@ -55,6 +55,7 @@ notify = "6.1.1" notify-debouncer-mini = "0.4.1" [dev-dependencies] +pretty_assertions = "1.4.0" semver = "1.0.12" [package.metadata.binstall] diff --git a/frb_codegen/src/library/codegen/generator/api_dart/mod.rs b/frb_codegen/src/library/codegen/generator/api_dart/mod.rs index ff6b21469e..b58353481c 100644 --- a/frb_codegen/src/library/codegen/generator/api_dart/mod.rs +++ b/frb_codegen/src/library/codegen/generator/api_dart/mod.rs @@ -56,6 +56,12 @@ mod tests { body("library/codegen/generator/api_dart/mod/simple") } + #[test] + #[serial] + fn test_functions() -> anyhow::Result<()> { + body("library/codegen/generator/api_dart/mod/functions") + } + fn body(fixture_name: &str) -> anyhow::Result<()> { configure_opinionated_test_logging(); let test_fixture_dir = get_test_fixture_dir(fixture_name); diff --git a/frb_codegen/src/library/codegen/generator/api_dart/spec_generator/class/field.rs b/frb_codegen/src/library/codegen/generator/api_dart/spec_generator/class/field.rs index 8610576e90..edc6bcc9fd 100644 --- a/frb_codegen/src/library/codegen/generator/api_dart/spec_generator/class/field.rs +++ b/frb_codegen/src/library/codegen/generator/api_dart/spec_generator/class/field.rs @@ -51,11 +51,19 @@ fn default_value_maybe_to_dart_style(value: &str, enable: bool) -> Cow { } fn default_value_to_dart_style(value: &str) -> String { - let mut split = value.split('.'); - let enum_name = split.next().unwrap(); - - let variant_name = split.next().unwrap().to_string(); - let variant_name = make_string_keyword_safe(variant_name.to_case(Case::Camel)); - - format!("{enum_name}.{variant_name}") + match value.split_once('.') { + // If the user is explicitly calling an enum variant's constructor + // i.e. `const Foo.bar()` instead of `Foo.Bar`, we trust that they + // really mean it and don't convert. + Some((enum_name, variant_name)) + if !enum_name.starts_with("const ") && !variant_name.contains('(') => + { + format!( + "{}.{}", + enum_name, + make_string_keyword_safe(variant_name.to_string()).to_case(Case::Camel) + ) + } + _ => value.to_string(), + } } diff --git a/frb_codegen/src/library/codegen/mod.rs b/frb_codegen/src/library/codegen/mod.rs index 18565bf2e5..39da61a197 100644 --- a/frb_codegen/src/library/codegen/mod.rs +++ b/frb_codegen/src/library/codegen/mod.rs @@ -16,7 +16,7 @@ use crate::codegen::dumper::Dumper; use crate::codegen::misc::GeneratorProgressBarPack; use crate::codegen::parser::reader::CachedRustReader; pub use config::config::{Config, MetaConfig}; -pub use config::config_parser::*; + pub use dumper::internal_config::ConfigDumpContent; use log::debug; diff --git a/frb_codegen/src/library/commands/cargo_expand.rs b/frb_codegen/src/library/commands/cargo_expand.rs index b1a62dc6a7..8ad567b5f8 100644 --- a/frb_codegen/src/library/commands/cargo_expand.rs +++ b/frb_codegen/src/library/commands/cargo_expand.rs @@ -2,14 +2,17 @@ use crate::codegen::dumper::Dumper; use crate::codegen::ConfigDumpContent; use crate::command_args; use crate::library::commands::command_runner::execute_command; + use anyhow::{bail, Context, Result}; use itertools::Itertools; +use lazy_static::lazy_static; use log::{debug, info, warn}; use regex::Regex; + +use std::borrow::Cow; use std::collections::hash_map::Entry::{Occupied, Vacant}; use std::collections::HashMap; use std::path::{Path, PathBuf}; -use std::sync::OnceLock; use std::{env, fs}; #[derive(Default)] @@ -41,7 +44,10 @@ impl CachedCargoExpand { let expanded = match self.cache.entry(rust_crate_dir.to_owned()) { Occupied(entry) => entry.into_mut(), - Vacant(entry) => entry.insert(run_cargo_expand(rust_crate_dir, dumper, true)?), + Vacant(entry) => entry.insert( + unwrap_frb_attrs_in_doc(&run_cargo_expand(rust_crate_dir, dumper, true)?) + .into_owned(), + ), }; extract_module(expanded, module) @@ -93,7 +99,7 @@ fn run_cargo_expand( "--theme=none", "--ugly", "--config", - "build.rustflags=\"--cfg frb_expand\"" + r#"build.rustflags="--cfg frb_expand""# ); let output = execute_command("cargo", &args, Some(rust_crate_dir), None) @@ -114,20 +120,22 @@ fn run_cargo_expand( // frb-coverage:ignore-end } - let mut stdout_lines = stdout.lines(); - stdout_lines.next(); - let ans = stdout_lines.join("\n"); - static PATTERN: OnceLock = OnceLock::new(); - let pattern = PATTERN.get_or_init(|| { - Regex::new(r####"#\[doc =[\s\n]*r###"frb_marker: ([\s\S]*?)"###]"####).unwrap() - }); - let ans = pattern.replace_all(&ans, "$1").into_owned(); - + let ans = stdout.lines().skip(1).join("\n"); dumper.dump_str(ConfigDumpContent::Source, "cargo_expand.rs", &ans)?; - Ok(ans) } +/// Turns `#[doc = "frb_marker: .."]` back into `#[frb(..)]`, usually produced +/// as a side-effect of cargo-expand. +// NOTE: The amount of pounds must match exactly with the implementation in frb_macros +fn unwrap_frb_attrs_in_doc(code: &str) -> Cow { + lazy_static! { + static ref PATTERN: Regex = + Regex::new(r####"#\[doc =[\s\n]*r###"frb_marker: ([\s\S]*?)"###]"####).unwrap(); + } + PATTERN.replace_all(code, "$1") +} + fn install_cargo_expand() -> Result<()> { execute_command( "cargo", diff --git a/frb_codegen/src/library/commands/command_runner.rs b/frb_codegen/src/library/commands/command_runner.rs index 16f795f122..3fca2c934a 100644 --- a/frb_codegen/src/library/commands/command_runner.rs +++ b/frb_codegen/src/library/commands/command_runner.rs @@ -38,7 +38,7 @@ macro_rules! command_run { }}; } -/// Formats a list of [`PathBuf`]s using the syntax detailed in [`run`]. +/// Formats a list of [`PathBuf`]s using the syntax detailed in [`command_run`]. #[doc(hidden)] #[macro_export] macro_rules! command_args { diff --git a/frb_codegen/src/library/integration/integrator.rs b/frb_codegen/src/library/integration/integrator.rs index 8edcae4997..8102f3a95e 100644 --- a/frb_codegen/src/library/integration/integrator.rs +++ b/frb_codegen/src/library/integration/integrator.rs @@ -63,7 +63,7 @@ pub fn integrate(config: IntegrateConfig) -> Result<()> { fn modify_permissions(dart_root: &Path) -> Result<()> { let dir_cargokit = dart_root.join("rust_builder").join("cargokit"); - #[cfg(not(windows))] + #[cfg(unix)] { set_permission_executable(&dir_cargokit.join("build_pod.sh"))?; set_permission_executable(&dir_cargokit.join("run_build_tool.sh"))?; @@ -82,7 +82,7 @@ fn setup_cargokit_dependencies(dart_root: &Path) -> Result<()> { flutter_pub_get(&build_tool_dir) } -#[cfg(not(windows))] +#[cfg(unix)] fn set_permission_executable(path: &Path) -> Result<()> { use std::os::unix::fs::PermissionsExt; diff --git a/frb_codegen/src/library/utils/test_utils.rs b/frb_codegen/src/library/utils/test_utils.rs index a43c5230ab..5d7fe895d3 100644 --- a/frb_codegen/src/library/utils/test_utils.rs +++ b/frb_codegen/src/library/utils/test_utils.rs @@ -28,16 +28,28 @@ pub(crate) fn json_golden_test( let actual: Value = serde_json::from_str(&actual_str)?; debug!("json_golden_test sanitizers={sanitizers:?} actual:\n{actual_str}"); - raw_golden_test(actual, &actual_str, matcher_path, |x| { - Ok(serde_json::from_str(&x)?) - }) + raw_golden_test( + actual, + &actual_str, + matcher_path, + |x| Ok(serde_json::from_str(&x)?), + None, + ) } pub(crate) fn text_golden_test(actual: String, matcher_path: &Path) -> anyhow::Result<()> { - raw_golden_test(actual.clone(), &actual, matcher_path, |x| { + raw_golden_test( + actual.clone(), + &actual, + matcher_path, // Otherwise tests in macos/linux passes but fails on windows - Ok(x.replace("\r\n", "\n")) - }) + |x| Ok(x.replace("\r\n", "\n")), + Some(|expect, actual| { + #[cfg(test)] + use pretty_assertions::assert_str_eq as assert_eq; + assert_eq!(expect, actual); + }), + ) } fn raw_golden_test( @@ -45,11 +57,14 @@ fn raw_golden_test( actual_str: &str, matcher_path: &Path, deserializer: F, + asserter: Option, ) -> anyhow::Result<()> where T: Eq + Debug, F: Fn(String) -> anyhow::Result, { + #[cfg(test)] + use pretty_assertions::assert_eq; // This is *test* utils, not a part of real codegen, so no need to consider coverage // frb-coverage:ignore-start let expect = deserializer(if matcher_path.exists() { @@ -64,7 +79,8 @@ where fs::write(matcher_path, actual_str)?; } } else { - assert_eq!(actual, expect); + let asserter = asserter.unwrap_or(|expect, actual| assert_eq!(expect, actual)); + asserter(&expect, &actual); } Ok(()) diff --git a/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/Cargo.toml b/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/Cargo.toml new file mode 100644 index 0000000000..c91a146840 --- /dev/null +++ b/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/Cargo.toml @@ -0,0 +1,11 @@ +[package] +name = "example" +version = "0.1.0" +edition = "2018" + +# See more keys and their definitions at https://doc.rust-lang.org/cargo/reference/manifest.html + +[dependencies] +flutter_rust_bridge_macros.path = "../../../../../../../../frb_macros" + +[workspace] diff --git a/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/expect_output.dart b/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/expect_output.dart new file mode 100644 index 0000000000..81ba392b73 --- /dev/null +++ b/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/expect_output.dart @@ -0,0 +1,14 @@ + + // This file is automatically generated, so please do not edit it. +// Generated by `flutter_rust_bridge`@ {VERSION}. + +// ignore_for_file: invalid_use_of_internal_member, unused_import, unnecessary_import + +import 'frb_generated.dart'; +import 'package:flutter_rust_bridge/flutter_rust_bridge_for_generated.dart'; + + + Future fnWithDefaultArg({int foo = 1, dynamic hint}) => RustLib.instance.api.fnWithDefaultArg(foo: foo, hint: hint); + + + \ No newline at end of file diff --git a/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/flutter_rust_bridge.yaml b/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/flutter_rust_bridge.yaml new file mode 100644 index 0000000000..eb15b30ea2 --- /dev/null +++ b/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/flutter_rust_bridge.yaml @@ -0,0 +1,3 @@ +rust_input: src/api.rs +dart_output: . +c_output: frb_generated.h diff --git a/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/pubspec.yaml b/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/pubspec.yaml new file mode 100644 index 0000000000..14dd5aa312 --- /dev/null +++ b/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/pubspec.yaml @@ -0,0 +1 @@ +name: fake_dart_package diff --git a/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/src/api.rs b/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/src/api.rs new file mode 100644 index 0000000000..290c9b26bf --- /dev/null +++ b/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/src/api.rs @@ -0,0 +1,4 @@ +#[flutter_rust_bridge_macros::frb] +pub fn fn_with_default_arg(#[frb(default = 1)] foo: i32) { + drop(foo); +} diff --git a/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/src/lib.rs b/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/src/lib.rs new file mode 100644 index 0000000000..b32f9e29f1 --- /dev/null +++ b/frb_codegen/test_fixtures/library/codegen/generator/api_dart/mod/functions/src/lib.rs @@ -0,0 +1 @@ +mod api; diff --git a/frb_example/pure_dart_pde/lib/src/rust/api/attribute.dart b/frb_example/pure_dart_pde/lib/src/rust/api/attribute.dart index 0e47c807cb..4c38db0575 100644 --- a/frb_example/pure_dart_pde/lib/src/rust/api/attribute.dart +++ b/frb_example/pure_dart_pde/lib/src/rust/api/attribute.dart @@ -14,7 +14,7 @@ Future handleCustomizedStructTwinNormal( RustLib.instance.api.handleCustomizedStructTwinNormal(val: val, hint: hint); Future nextUserIdTwinNormal( - {required UserIdTwinNormal userId, dynamic hint}) => + {UserIdTwinNormal userId = const UserIdTwinNormal(), dynamic hint}) => RustLib.instance.api.nextUserIdTwinNormal(userId: userId, hint: hint); class CustomizedTwinNormal { diff --git a/frb_example/pure_dart_pde/lib/src/rust/api/pseudo_manual/attribute_twin_rust_async.dart b/frb_example/pure_dart_pde/lib/src/rust/api/pseudo_manual/attribute_twin_rust_async.dart index c991817f36..03d442f2ff 100644 --- a/frb_example/pure_dart_pde/lib/src/rust/api/pseudo_manual/attribute_twin_rust_async.dart +++ b/frb_example/pure_dart_pde/lib/src/rust/api/pseudo_manual/attribute_twin_rust_async.dart @@ -15,7 +15,8 @@ Future handleCustomizedStructTwinRustAsync( .handleCustomizedStructTwinRustAsync(val: val, hint: hint); Future nextUserIdTwinRustAsync( - {required UserIdTwinRustAsync userId, dynamic hint}) => + {UserIdTwinRustAsync userId = const UserIdTwinRustAsync(), + dynamic hint}) => RustLib.instance.api.nextUserIdTwinRustAsync(userId: userId, hint: hint); class CustomizedTwinRustAsync { diff --git a/frb_example/pure_dart_pde/lib/src/rust/api/pseudo_manual/attribute_twin_sync.dart b/frb_example/pure_dart_pde/lib/src/rust/api/pseudo_manual/attribute_twin_sync.dart index 8c9b6ae598..d13b9e480c 100644 --- a/frb_example/pure_dart_pde/lib/src/rust/api/pseudo_manual/attribute_twin_sync.dart +++ b/frb_example/pure_dart_pde/lib/src/rust/api/pseudo_manual/attribute_twin_sync.dart @@ -14,7 +14,7 @@ void handleCustomizedStructTwinSync( RustLib.instance.api.handleCustomizedStructTwinSync(val: val, hint: hint); UserIdTwinSync nextUserIdTwinSync( - {required UserIdTwinSync userId, dynamic hint}) => + {UserIdTwinSync userId = const UserIdTwinSync(), dynamic hint}) => RustLib.instance.api.nextUserIdTwinSync(userId: userId, hint: hint); class CustomizedTwinSync { diff --git a/frb_example/pure_dart_pde/lib/src/rust/frb_generated.dart b/frb_example/pure_dart_pde/lib/src/rust/frb_generated.dart index 1c2dd1e165..a7a31d0ba1 100644 --- a/frb_example/pure_dart_pde/lib/src/rust/frb_generated.dart +++ b/frb_example/pure_dart_pde/lib/src/rust/frb_generated.dart @@ -219,7 +219,7 @@ abstract class RustLibApi extends BaseApi { {required CustomizedTwinNormal val, dynamic hint}); Future nextUserIdTwinNormal( - {required UserIdTwinNormal userId, dynamic hint}); + {UserIdTwinNormal userId = const UserIdTwinNormal(), dynamic hint}); void benchmarkVoidSemiSerialize({dynamic hint}); @@ -735,13 +735,13 @@ abstract class RustLibApi extends BaseApi { {required CustomizedTwinRustAsync val, dynamic hint}); Future nextUserIdTwinRustAsync( - {required UserIdTwinRustAsync userId, dynamic hint}); + {UserIdTwinRustAsync userId = const UserIdTwinRustAsync(), dynamic hint}); void handleCustomizedStructTwinSync( {required CustomizedTwinSync val, dynamic hint}); UserIdTwinSync nextUserIdTwinSync( - {required UserIdTwinSync userId, dynamic hint}); + {UserIdTwinSync userId = const UserIdTwinSync(), dynamic hint}); Future exampleBasicTypeBasicGeneralEnumTwinNormalTwinNormal( @@ -3715,7 +3715,7 @@ class RustLibApiImpl extends RustLibApiImplPlatform implements RustLibApi { @override Future nextUserIdTwinNormal( - {required UserIdTwinNormal userId, dynamic hint}) { + {UserIdTwinNormal userId = const UserIdTwinNormal(), dynamic hint}) { return handler.executeNormal(NormalTask( callFfi: (port_) { final serializer = SseSerializer(generalizedFrbRustBinding); @@ -8498,7 +8498,8 @@ class RustLibApiImpl extends RustLibApiImplPlatform implements RustLibApi { @override Future nextUserIdTwinRustAsync( - {required UserIdTwinRustAsync userId, dynamic hint}) { + {UserIdTwinRustAsync userId = const UserIdTwinRustAsync(), + dynamic hint}) { return handler.executeNormal(NormalTask( callFfi: (port_) { final serializer = SseSerializer(generalizedFrbRustBinding); @@ -8550,7 +8551,7 @@ class RustLibApiImpl extends RustLibApiImplPlatform implements RustLibApi { @override UserIdTwinSync nextUserIdTwinSync( - {required UserIdTwinSync userId, dynamic hint}) { + {UserIdTwinSync userId = const UserIdTwinSync(), dynamic hint}) { return handler.executeSync(SyncTask( callFfi: () { final serializer = SseSerializer(generalizedFrbRustBinding); diff --git a/frb_macros/src/lib.rs b/frb_macros/src/lib.rs index a055e6ba78..b9d4ff5399 100644 --- a/frb_macros/src/lib.rs +++ b/frb_macros/src/lib.rs @@ -10,10 +10,10 @@ use proc_macro::*; // frb-coverage:ignore-start #[proc_macro_attribute] pub fn frb(attribute: TokenStream, item: TokenStream) -> TokenStream { - let mut attribute = format_frb_attribute(format!("#[frb({attribute})]")); + let mut output = format_frb_attribute(format!("#[frb({attribute})]")); let item = strip_frb_attr(item); - attribute.extend(item); - attribute + output.extend(item); + output } fn strip_frb_attr(item: TokenStream) -> TokenStream { @@ -30,14 +30,16 @@ fn strip_frb_attr(item: TokenStream) -> TokenStream { Some(format_frb_attribute(format!("#[{}]", group.stream()))) } (_, T::Group(group)) => Some( - pound - .take() - .into_iter() - .chain(Some(T::Group(Group::new( + [ + pound.take(), + Some(T::Group(Group::new( group.delimiter(), strip_frb_attr(group.stream()), - )))) - .collect(), + ))), + ] + .into_iter() + .flatten() + .collect(), ), _ => Some(tok.into()), } diff --git a/justfile b/justfile index 765e5b0fff..e91616d743 100644 --- a/justfile +++ b/justfile @@ -27,5 +27,5 @@ _port_forward_ubuntu: ssh -L 8181:localhost:8181 ubuntu -N [no-cd] -expand *args: +_expand *args: cargo expand --config 'build.rustflags="--cfg frb_expand"' {{args}}