From f7f620e44d8655aac3514677f2d2dc6974f9d918 Mon Sep 17 00:00:00 2001 From: Kevin Wenner Date: Thu, 28 Sep 2023 09:15:46 +0400 Subject: [PATCH] Fix `@forward` statement altering the scope of the forwarded module (#85) * avoid mutating forwarded module scope * fix by adding scope to ForwardedModule instead - add a test reproducing the issue * fix verbose and no-color sass cli flags --- crates/compiler/src/builtin/modules/mod.rs | 25 ++++++---------- crates/lib/src/lib.rs | 4 +-- crates/lib/src/main.rs | 2 ++ crates/lib/tests/forward.rs | 34 ++++++++++++++++++++++ 4 files changed, 46 insertions(+), 19 deletions(-) diff --git a/crates/compiler/src/builtin/modules/mod.rs b/crates/compiler/src/builtin/modules/mod.rs index 0159e96..d7703bb 100644 --- a/crates/compiler/src/builtin/modules/mod.rs +++ b/crates/compiler/src/builtin/modules/mod.rs @@ -108,6 +108,8 @@ impl ShadowedModule { #[derive(Debug, Clone)] pub(crate) struct ForwardedModule { + scope: ModuleScope, + #[allow(dead_code)] inner: Arc>, #[allow(dead_code)] forward_rule: AstForwardRule, @@ -138,15 +140,16 @@ impl ForwardedModule { rule.hidden_mixins_and_functions.as_ref(), ); - (*module).borrow_mut().set_scope(ModuleScope { + let scope = ModuleScope { variables, mixins, functions, - }); + }; ForwardedModule { inner: module, forward_rule: rule, + scope, } } @@ -362,17 +365,8 @@ impl Module { match self { Self::Builtin { scope } | Self::Environment { scope, .. } + | Self::Forwarded(ForwardedModule { scope, .. }) | Self::Shadowed(ShadowedModule { scope, .. }) => scope.clone(), - Self::Forwarded(forwarded) => (*forwarded.inner).borrow().scope(), - } - } - - fn set_scope(&mut self, new_scope: ModuleScope) { - match self { - Self::Builtin { scope } - | Self::Environment { scope, .. } - | Self::Shadowed(ShadowedModule { scope, .. }) => *scope = new_scope, - Self::Forwarded(forwarded) => (*forwarded.inner).borrow_mut().set_scope(new_scope), } } @@ -402,10 +396,9 @@ impl Module { Self::Builtin { .. } => { return Err(("Cannot modify built-in variable.", name.span).into()) } - Self::Environment { scope, .. } | Self::Shadowed(ShadowedModule { scope, .. }) => { - scope.clone() - } - Self::Forwarded(forwarded) => (*forwarded.inner).borrow_mut().scope(), + Self::Environment { scope, .. } + | Self::Forwarded(ForwardedModule { scope, .. }) + | Self::Shadowed(ShadowedModule { scope, .. }) => scope.clone(), }; if scope.variables.insert(name.node, value).is_none() { diff --git a/crates/lib/src/lib.rs b/crates/lib/src/lib.rs index 18acce6..8ae8c86 100644 --- a/crates/lib/src/lib.rs +++ b/crates/lib/src/lib.rs @@ -72,9 +72,7 @@ pub use grass_compiler::{ /// Include CSS in your binary at compile time from a Sass source file /// -/// ``` -/// static CSS: &str = grass::include!("../static/_index.scss"); -/// ``` +/// `static CSS: &str = grass::include!("../static/_index.scss");` /// /// This requires the `"macro"` feature, which is not enabled by default. /// diff --git a/crates/lib/src/main.rs b/crates/lib/src/main.rs index b511652..8e1ea5a 100644 --- a/crates/lib/src/main.rs +++ b/crates/lib/src/main.rs @@ -168,12 +168,14 @@ fn cli() -> Command { .arg( Arg::new("NO_COLOR") .short('c') + .action(ArgAction::SetTrue) .long("no-color") .hide(true) .help("Whether to use terminal colors for messages.") ) .arg( Arg::new("VERBOSE") + .action(ArgAction::SetTrue) .long("verbose") .hide(true) .help("Print all deprecation warnings even when they're repetitive.") diff --git a/crates/lib/tests/forward.rs b/crates/lib/tests/forward.rs index b7c5702..5cd8de3 100644 --- a/crates/lib/tests/forward.rs +++ b/crates/lib/tests/forward.rs @@ -501,6 +501,40 @@ fn import_forwarded_first_no_use() { ); } +#[test] +fn forward_same_module_with_and_without_prefix() { + let mut fs = TestFs::new(); + + fs.add_file( + "_midstream.scss", + r#" + @forward "upstream"; + @forward "upstream" as b-*; + "#, + ); + fs.add_file( + "_upstream.scss", + r#" + @mixin a() { + c { + d: e + } + } + "#, + ); + + let input = r#" + @use "midstream"; + + @include midstream.a; + "#; + + assert_eq!( + "c {\n d: e;\n}\n", + &grass::from_string(input.to_string(), &grass::Options::default().fs(&fs)).expect(input) + ); +} + error!( after_style_rule, r#"