From 7030e9cac6f1a13bd4d366fab66f0341edeb3002 Mon Sep 17 00:00:00 2001 From: Casey Rodarmor Date: Thu, 31 Oct 2024 17:54:46 -0700 Subject: [PATCH] Add `&&` and `||` operators (#2444) --- GRAMMAR.md | 8 ++- README.md | 43 ++++++++++-- src/analyzer.rs | 5 +- src/assignment_resolver.rs | 130 +++++++------------------------------ src/ast.rs | 1 + src/evaluator.rs | 65 +++++++++++-------- src/expression.rs | 54 +++++++++++---- src/lexer.rs | 2 + src/node.rs | 43 ++++++------ src/parser.rs | 81 ++++++++++++++++------- src/summary.rs | 28 ++++++-- src/token_kind.rs | 2 + src/unstable_feature.rs | 5 ++ src/variables.rs | 46 +++++++------ tests/assert_success.rs | 1 + tests/conditional.rs | 2 +- tests/ignore_comments.rs | 2 +- tests/lib.rs | 1 + tests/logical_operators.rs | 83 +++++++++++++++++++++++ tests/shell_expansion.rs | 2 +- 20 files changed, 373 insertions(+), 231 deletions(-) create mode 100644 tests/logical_operators.rs diff --git a/GRAMMAR.md b/GRAMMAR.md index e5847f5098..00721f15af 100644 --- a/GRAMMAR.md +++ b/GRAMMAR.md @@ -90,7 +90,13 @@ import : 'import' '?'? string? eol module : 'mod' '?'? NAME string? eol -expression : 'if' condition '{' expression '}' 'else' '{' expression '}' +expression : disjunct || expression + | disjunct + +disjunct : conjunct && disjunct + | conjunct + +conjunct : 'if' condition '{' expression '}' 'else' '{' expression '}' | 'assert' '(' condition ',' expression ')' | '/' expression | value '/' expression diff --git a/README.md b/README.md index ca596854fc..bccf9a1357 100644 --- a/README.md +++ b/README.md @@ -1290,9 +1290,11 @@ Available recipes: test ``` -### Variables and Substitution +### Expressions and Substitutions -Variables, strings, concatenation, path joining, substitution using `{{…}}`, and function calls are supported: +Various operators and function calls are supported in expressions, which may be +used in assignments, default recipe arguments, and inside recipe body `{{…}}` +substitutions. ```just tmpdir := `mktemp -d` @@ -1310,6 +1312,39 @@ publish: rm -rf {{tarball}} {{tardir}} ``` +#### Concatenation + +The `+` operator returns the left-hand argument concatenated with the +right-hand argument: + +```just +foobar := 'foo' + 'bar' +``` + +#### Logical Operators + +The logical operators `&&` and `||` can be used to coalesce string +valuesmaster, similar to Python's `and` and `or`. These operators +consider the empty string `''` to be false, and all other strings to be true. + +These operators are currently unstable. + +The `&&` operator returns the empty string if the left-hand argument is the +empty string, otherwise it returns the right-hand argument: + +```mf +foo := '' && 'goodbye' # '' +bar := 'hello' && 'goodbye' # 'goodbye' +``` + +The `||` operator returns the left-hand argument if it is non-empty, otherwise +it returns the right-hand argument: + +```mf +foo := '' || 'goodbye' # 'goodbye' +bar := 'hello' || 'goodbye' # 'hello' +``` + #### Joining Paths The `/` operator can be used to join two strings with a slash: @@ -2367,8 +2402,8 @@ Testing server:unit… ./test --tests unit server ``` -Default values may be arbitrary expressions, but concatenations or path joins -must be parenthesized: +Default values may be arbitrary expressions, but expressions containing the +`+`, `&&`, `||`, or `/` operators must be parenthesized: ```just arch := "wasm" diff --git a/src/analyzer.rs b/src/analyzer.rs index 9a26ef0455..0929294c5d 100644 --- a/src/analyzer.rs +++ b/src/analyzer.rs @@ -36,12 +36,15 @@ impl<'run, 'src> Analyzer<'run, 'src> { ) -> CompileResult<'src, Justfile<'src>> { let mut definitions = HashMap::new(); let mut imports = HashSet::new(); + let mut unstable_features = BTreeSet::new(); let mut stack = Vec::new(); let ast = asts.get(root).unwrap(); stack.push(ast); while let Some(ast) = stack.pop() { + unstable_features.extend(&ast.unstable_features); + for item in &ast.items { match item { Item::Alias(alias) => { @@ -166,8 +169,6 @@ impl<'run, 'src> Analyzer<'run, 'src> { aliases.insert(Self::resolve_alias(&recipes, alias)?); } - let mut unstable_features = BTreeSet::new(); - for recipe in recipes.values() { for attribute in &recipe.attributes { if let Attribute::Script(_) = attribute { diff --git a/src/assignment_resolver.rs b/src/assignment_resolver.rs index 53863fc9c5..511464d4eb 100644 --- a/src/assignment_resolver.rs +++ b/src/assignment_resolver.rs @@ -31,7 +31,29 @@ impl<'src: 'run, 'run> AssignmentResolver<'src, 'run> { self.stack.push(name); if let Some(assignment) = self.assignments.get(name) { - self.resolve_expression(&assignment.value)?; + for variable in assignment.value.variables() { + let name = variable.lexeme(); + + if self.evaluated.contains(name) || constants().contains_key(name) { + continue; + } + + if self.stack.contains(&name) { + self.stack.push(name); + return Err( + self.assignments[name] + .name + .error(CircularVariableDependency { + variable: name, + circle: self.stack.clone(), + }), + ); + } else if self.assignments.contains_key(name) { + self.resolve_assignment(name)?; + } else { + return Err(variable.error(UndefinedVariable { variable: name })); + } + } self.evaluated.insert(name); } else { let message = format!("attempted to resolve unknown assignment `{name}`"); @@ -51,112 +73,6 @@ impl<'src: 'run, 'run> AssignmentResolver<'src, 'run> { Ok(()) } - - fn resolve_expression(&mut self, expression: &Expression<'src>) -> CompileResult<'src> { - match expression { - Expression::Assert { - condition: Condition { - lhs, - rhs, - operator: _, - }, - error, - } => { - self.resolve_expression(lhs)?; - self.resolve_expression(rhs)?; - self.resolve_expression(error) - } - Expression::Call { thunk } => match thunk { - Thunk::Nullary { .. } => Ok(()), - Thunk::Unary { arg, .. } => self.resolve_expression(arg), - Thunk::UnaryOpt { args: (a, b), .. } => { - self.resolve_expression(a)?; - if let Some(b) = b.as_ref() { - self.resolve_expression(b)?; - } - Ok(()) - } - Thunk::UnaryPlus { - args: (a, rest), .. - } => { - self.resolve_expression(a)?; - for arg in rest { - self.resolve_expression(arg)?; - } - Ok(()) - } - Thunk::Binary { args: [a, b], .. } => { - self.resolve_expression(a)?; - self.resolve_expression(b) - } - Thunk::BinaryPlus { - args: ([a, b], rest), - .. - } => { - self.resolve_expression(a)?; - self.resolve_expression(b)?; - for arg in rest { - self.resolve_expression(arg)?; - } - Ok(()) - } - Thunk::Ternary { - args: [a, b, c], .. - } => { - self.resolve_expression(a)?; - self.resolve_expression(b)?; - self.resolve_expression(c) - } - }, - Expression::Concatenation { lhs, rhs } => { - self.resolve_expression(lhs)?; - self.resolve_expression(rhs) - } - Expression::Conditional { - condition: Condition { - lhs, - rhs, - operator: _, - }, - then, - otherwise, - .. - } => { - self.resolve_expression(lhs)?; - self.resolve_expression(rhs)?; - self.resolve_expression(then)?; - self.resolve_expression(otherwise) - } - Expression::Group { contents } => self.resolve_expression(contents), - Expression::Join { lhs, rhs } => { - if let Some(lhs) = lhs { - self.resolve_expression(lhs)?; - } - self.resolve_expression(rhs) - } - Expression::StringLiteral { .. } | Expression::Backtick { .. } => Ok(()), - Expression::Variable { name } => { - let variable = name.lexeme(); - if self.evaluated.contains(variable) || constants().contains_key(variable) { - Ok(()) - } else if self.stack.contains(&variable) { - self.stack.push(variable); - Err( - self.assignments[variable] - .name - .error(CircularVariableDependency { - variable, - circle: self.stack.clone(), - }), - ) - } else if self.assignments.contains_key(variable) { - self.resolve_assignment(variable) - } else { - Err(name.token.error(UndefinedVariable { variable })) - } - } - } - } } #[cfg(test)] diff --git a/src/ast.rs b/src/ast.rs index f9dd10c9f3..1ad7a8aa22 100644 --- a/src/ast.rs +++ b/src/ast.rs @@ -6,6 +6,7 @@ use super::*; #[derive(Debug, Clone)] pub(crate) struct Ast<'src> { pub(crate) items: Vec>, + pub(crate) unstable_features: BTreeSet, pub(crate) warnings: Vec, pub(crate) working_directory: PathBuf, } diff --git a/src/evaluator.rs b/src/evaluator.rs index 4ed00036d1..d48cece87e 100644 --- a/src/evaluator.rs +++ b/src/evaluator.rs @@ -84,24 +84,31 @@ impl<'src, 'run> Evaluator<'src, 'run> { expression: &Expression<'src>, ) -> RunResult<'src, String> { match expression { - Expression::Variable { name, .. } => { - let variable = name.lexeme(); - if let Some(value) = self.scope.value(variable) { - Ok(value.to_owned()) - } else if let Some(assignment) = self - .assignments - .and_then(|assignments| assignments.get(variable)) - { - Ok(self.evaluate_assignment(assignment)?.to_owned()) + Expression::And { lhs, rhs } => { + let lhs = self.evaluate_expression(lhs)?; + if lhs.is_empty() { + return Ok(String::new()); + } + self.evaluate_expression(rhs) + } + Expression::Assert { condition, error } => { + if self.evaluate_condition(condition)? { + Ok(String::new()) } else { - Err(Error::Internal { - message: format!("attempted to evaluate undefined variable `{variable}`"), + Err(Error::Assert { + message: self.evaluate_expression(error)?, }) } } + Expression::Backtick { contents, token } => { + if self.context.config.dry_run { + Ok(format!("`{contents}`")) + } else { + Ok(self.run_backtick(contents, token)?) + } + } Expression::Call { thunk } => { use Thunk::*; - let result = match thunk { Nullary { function, .. } => function(function::Context::new(self, thunk.name())), Unary { function, arg, .. } => { @@ -118,7 +125,6 @@ impl<'src, 'run> Evaluator<'src, 'run> { Some(b) => Some(self.evaluate_expression(b)?), None => None, }; - function(function::Context::new(self, thunk.name()), &a, b.as_deref()) } UnaryPlus { @@ -175,20 +181,11 @@ impl<'src, 'run> Evaluator<'src, 'run> { function(function::Context::new(self, thunk.name()), &a, &b, &c) } }; - result.map_err(|message| Error::FunctionCall { function: thunk.name(), message, }) } - Expression::StringLiteral { string_literal } => Ok(string_literal.cooked.clone()), - Expression::Backtick { contents, token } => { - if self.context.config.dry_run { - Ok(format!("`{contents}`")) - } else { - Ok(self.run_backtick(contents, token)?) - } - } Expression::Concatenation { lhs, rhs } => { Ok(self.evaluate_expression(lhs)? + &self.evaluate_expression(rhs)?) } @@ -209,12 +206,26 @@ impl<'src, 'run> Evaluator<'src, 'run> { lhs: Some(lhs), rhs, } => Ok(self.evaluate_expression(lhs)? + "/" + &self.evaluate_expression(rhs)?), - Expression::Assert { condition, error } => { - if self.evaluate_condition(condition)? { - Ok(String::new()) + Expression::Or { lhs, rhs } => { + let lhs = self.evaluate_expression(lhs)?; + if !lhs.is_empty() { + return Ok(lhs); + } + self.evaluate_expression(rhs) + } + Expression::StringLiteral { string_literal } => Ok(string_literal.cooked.clone()), + Expression::Variable { name, .. } => { + let variable = name.lexeme(); + if let Some(value) = self.scope.value(variable) { + Ok(value.to_owned()) + } else if let Some(assignment) = self + .assignments + .and_then(|assignments| assignments.get(variable)) + { + Ok(self.evaluate_assignment(assignment)?.to_owned()) } else { - Err(Error::Assert { - message: self.evaluate_expression(error)?, + Err(Error::Internal { + message: format!("attempted to evaluate undefined variable `{variable}`"), }) } } diff --git a/src/expression.rs b/src/expression.rs index c1e4b5e015..3d5c339211 100644 --- a/src/expression.rs +++ b/src/expression.rs @@ -8,6 +8,11 @@ use super::*; /// The parser parses both values and expressions into `Expression`s. #[derive(PartialEq, Debug, Clone)] pub(crate) enum Expression<'src> { + /// `lhs && rhs` + And { + lhs: Box>, + rhs: Box>, + }, /// `assert(condition, error)` Assert { condition: Condition<'src>, @@ -38,6 +43,11 @@ pub(crate) enum Expression<'src> { lhs: Option>>, rhs: Box>, }, + /// `lhs || rhs` + Or { + lhs: Box>, + rhs: Box>, + }, /// `"string_literal"` or `'string_literal'` StringLiteral { string_literal: StringLiteral<'src> }, /// `variable` @@ -53,23 +63,25 @@ impl<'src> Expression<'src> { impl<'src> Display for Expression<'src> { fn fmt(&self, f: &mut Formatter) -> fmt::Result { match self { + Self::And { lhs, rhs } => write!(f, "{lhs} && {rhs}"), Self::Assert { condition, error } => write!(f, "assert({condition}, {error})"), Self::Backtick { token, .. } => write!(f, "{}", token.lexeme()), - Self::Join { lhs: None, rhs } => write!(f, "/ {rhs}"), - Self::Join { - lhs: Some(lhs), - rhs, - } => write!(f, "{lhs} / {rhs}"), + Self::Call { thunk } => write!(f, "{thunk}"), Self::Concatenation { lhs, rhs } => write!(f, "{lhs} + {rhs}"), Self::Conditional { condition, then, otherwise, } => write!(f, "if {condition} {{ {then} }} else {{ {otherwise} }}"), + Self::Group { contents } => write!(f, "({contents})"), + Self::Join { lhs: None, rhs } => write!(f, "/ {rhs}"), + Self::Join { + lhs: Some(lhs), + rhs, + } => write!(f, "{lhs} / {rhs}"), + Self::Or { lhs, rhs } => write!(f, "{lhs} || {rhs}"), Self::StringLiteral { string_literal } => write!(f, "{string_literal}"), Self::Variable { name } => write!(f, "{}", name.lexeme()), - Self::Call { thunk } => write!(f, "{thunk}"), - Self::Group { contents } => write!(f, "({contents})"), } } } @@ -80,6 +92,13 @@ impl<'src> Serialize for Expression<'src> { S: Serializer, { match self { + Self::And { lhs, rhs } => { + let mut seq = serializer.serialize_seq(None)?; + seq.serialize_element("and")?; + seq.serialize_element(lhs)?; + seq.serialize_element(rhs)?; + seq.end() + } Self::Assert { condition, error } => { let mut seq: ::SerializeSeq = serializer.serialize_seq(None)?; seq.serialize_element("assert")?; @@ -101,13 +120,6 @@ impl<'src> Serialize for Expression<'src> { seq.serialize_element(rhs)?; seq.end() } - Self::Join { lhs, rhs } => { - let mut seq = serializer.serialize_seq(None)?; - seq.serialize_element("join")?; - seq.serialize_element(lhs)?; - seq.serialize_element(rhs)?; - seq.end() - } Self::Conditional { condition, then, @@ -121,6 +133,20 @@ impl<'src> Serialize for Expression<'src> { seq.end() } Self::Group { contents } => contents.serialize(serializer), + Self::Join { lhs, rhs } => { + let mut seq = serializer.serialize_seq(None)?; + seq.serialize_element("join")?; + seq.serialize_element(lhs)?; + seq.serialize_element(rhs)?; + seq.end() + } + Self::Or { lhs, rhs } => { + let mut seq = serializer.serialize_seq(None)?; + seq.serialize_element("or")?; + seq.serialize_element(lhs)?; + seq.serialize_element(rhs)?; + seq.end() + } Self::StringLiteral { string_literal } => string_literal.serialize(serializer), Self::Variable { name } => { let mut seq = serializer.serialize_seq(None)?; diff --git a/src/lexer.rs b/src/lexer.rs index 4d28a4460f..2c56db9dcf 100644 --- a/src/lexer.rs +++ b/src/lexer.rs @@ -496,6 +496,7 @@ impl<'src> Lexer<'src> { ']' => self.lex_delimiter(BracketR), '`' | '"' | '\'' => self.lex_string(), '{' => self.lex_delimiter(BraceL), + '|' => self.lex_digraph('|', '|', BarBar), '}' => self.lex_delimiter(BraceR), _ if Self::is_identifier_start(start) => self.lex_identifier(), _ => { @@ -948,6 +949,7 @@ mod tests { Asterisk => "*", At => "@", BangEquals => "!=", + BarBar => "||", BraceL => "{", BraceR => "}", BracketL => "[", diff --git a/src/node.rs b/src/node.rs index 3ccf862d57..f8788c4dc1 100644 --- a/src/node.rs +++ b/src/node.rs @@ -88,6 +88,7 @@ impl<'src> Node<'src> for Assignment<'src> { impl<'src> Node<'src> for Expression<'src> { fn tree(&self) -> Tree<'src> { match self { + Self::And { lhs, rhs } => Tree::atom("&&").push(lhs.tree()).push(rhs.tree()), Self::Assert { condition: Condition { lhs, rhs, operator }, error, @@ -96,25 +97,10 @@ impl<'src> Node<'src> for Expression<'src> { .push(operator.to_string()) .push(rhs.tree()) .push(error.tree()), - Self::Concatenation { lhs, rhs } => Tree::atom("+").push(lhs.tree()).push(rhs.tree()), - Self::Conditional { - condition: Condition { lhs, rhs, operator }, - then, - otherwise, - } => { - let mut tree = Tree::atom(Keyword::If.lexeme()); - tree.push_mut(lhs.tree()); - tree.push_mut(operator.to_string()); - tree.push_mut(rhs.tree()); - tree.push_mut(then.tree()); - tree.push_mut(otherwise.tree()); - tree - } + Self::Backtick { contents, .. } => Tree::atom("backtick").push(Tree::string(contents)), Self::Call { thunk } => { use Thunk::*; - let mut tree = Tree::atom("call"); - match thunk { Nullary { name, .. } => tree.push_mut(name.lexeme()), Unary { name, arg, .. } => { @@ -171,20 +157,33 @@ impl<'src> Node<'src> for Expression<'src> { tree.push_mut(c.tree()); } } - tree } - Self::Variable { name } => Tree::atom(name.lexeme()), - Self::StringLiteral { - string_literal: StringLiteral { cooked, .. }, - } => Tree::string(cooked), - Self::Backtick { contents, .. } => Tree::atom("backtick").push(Tree::string(contents)), + Self::Concatenation { lhs, rhs } => Tree::atom("+").push(lhs.tree()).push(rhs.tree()), + Self::Conditional { + condition: Condition { lhs, rhs, operator }, + then, + otherwise, + } => { + let mut tree = Tree::atom(Keyword::If.lexeme()); + tree.push_mut(lhs.tree()); + tree.push_mut(operator.to_string()); + tree.push_mut(rhs.tree()); + tree.push_mut(then.tree()); + tree.push_mut(otherwise.tree()); + tree + } Self::Group { contents } => Tree::List(vec![contents.tree()]), Self::Join { lhs: None, rhs } => Tree::atom("/").push(rhs.tree()), Self::Join { lhs: Some(lhs), rhs, } => Tree::atom("/").push(lhs.tree()).push(rhs.tree()), + Self::Or { lhs, rhs } => Tree::atom("||").push(lhs.tree()).push(rhs.tree()), + Self::StringLiteral { + string_literal: StringLiteral { cooked, .. }, + } => Tree::string(cooked), + Self::Variable { name } => Tree::atom(name.lexeme()), } } } diff --git a/src/parser.rs b/src/parser.rs index 99544acd13..e03ab7833f 100644 --- a/src/parser.rs +++ b/src/parser.rs @@ -31,6 +31,7 @@ pub(crate) struct Parser<'run, 'src> { next_token: usize, recursion_depth: usize, tokens: &'run [Token<'src>], + unstable_features: BTreeSet, working_directory: &'run Path, } @@ -51,6 +52,7 @@ impl<'run, 'src> Parser<'run, 'src> { next_token: 0, recursion_depth: 0, tokens, + unstable_features: BTreeSet::new(), working_directory, } .parse_ast() @@ -442,18 +444,19 @@ impl<'run, 'src> Parser<'run, 'src> { } } - if self.next_token == self.tokens.len() { - Ok(Ast { - items, - warnings: Vec::new(), - working_directory: self.working_directory.into(), - }) - } else { - Err(self.internal_error(format!( + if self.next_token != self.tokens.len() { + return Err(self.internal_error(format!( "Parse completed with {} unparsed tokens", self.tokens.len() - self.next_token, - ))?) + ))?); } + + Ok(Ast { + items, + unstable_features: self.unstable_features, + warnings: Vec::new(), + working_directory: self.working_directory.into(), + }) } /// Parse an alias, e.g `alias name := target` @@ -517,31 +520,63 @@ impl<'run, 'src> Parser<'run, 'src> { self.recursion_depth += 1; - let expression = if self.accepted_keyword(Keyword::If)? { - self.parse_conditional()? + let disjunct = self.parse_disjunct()?; + + let expression = if self.accepted(BarBar)? { + self + .unstable_features + .insert(UnstableFeature::LogicalOperators); + let lhs = disjunct.into(); + let rhs = self.parse_expression()?.into(); + Expression::Or { lhs, rhs } + } else { + disjunct + }; + + self.recursion_depth -= 1; + + Ok(expression) + } + + fn parse_disjunct(&mut self) -> CompileResult<'src, Expression<'src>> { + let conjunct = self.parse_conjunct()?; + + let disjunct = if self.accepted(AmpersandAmpersand)? { + self + .unstable_features + .insert(UnstableFeature::LogicalOperators); + let lhs = conjunct.into(); + let rhs = self.parse_disjunct()?.into(); + Expression::And { lhs, rhs } + } else { + conjunct + }; + + Ok(disjunct) + } + + fn parse_conjunct(&mut self) -> CompileResult<'src, Expression<'src>> { + if self.accepted_keyword(Keyword::If)? { + self.parse_conditional() } else if self.accepted(Slash)? { let lhs = None; - let rhs = self.parse_expression()?.into(); - Expression::Join { lhs, rhs } + let rhs = self.parse_conjunct()?.into(); + Ok(Expression::Join { lhs, rhs }) } else { let value = self.parse_value()?; if self.accepted(Slash)? { let lhs = Some(Box::new(value)); - let rhs = self.parse_expression()?.into(); - Expression::Join { lhs, rhs } + let rhs = self.parse_conjunct()?.into(); + Ok(Expression::Join { lhs, rhs }) } else if self.accepted(Plus)? { let lhs = value.into(); - let rhs = self.parse_expression()?.into(); - Expression::Concatenation { lhs, rhs } + let rhs = self.parse_conjunct()?.into(); + Ok(Expression::Concatenation { lhs, rhs }) } else { - value + Ok(value) } - }; - - self.recursion_depth -= 1; - - Ok(expression) + } } /// Parse a conditional, e.g. `if a == b { "foo" } else { "bar" }` diff --git a/src/summary.rs b/src/summary.rs index ee3a8d1155..76483d63ec 100644 --- a/src/summary.rs +++ b/src/summary.rs @@ -183,6 +183,10 @@ impl Assignment { #[derive(Eq, PartialEq, Hash, Ord, PartialOrd, Debug, Clone)] pub enum Expression { + And { + lhs: Box, + rhs: Box, + }, Assert { condition: Condition, error: Box, @@ -209,6 +213,10 @@ pub enum Expression { lhs: Option>, rhs: Box, }, + Or { + lhs: Box, + rhs: Box, + }, String { text: String, }, @@ -221,6 +229,10 @@ impl Expression { fn new(expression: &full::Expression) -> Self { use full::Expression::*; match expression { + And { lhs, rhs } => Self::And { + lhs: Self::new(lhs).into(), + rhs: Self::new(rhs).into(), + }, Assert { condition: full::Condition { lhs, rhs, operator }, error, @@ -250,11 +262,9 @@ impl Expression { .. } => { let mut arguments = Vec::new(); - if let Some(b) = opt_b.as_ref() { arguments.push(Self::new(b)); } - arguments.push(Self::new(a)); Self::Call { name: name.lexeme().to_owned(), @@ -308,10 +318,6 @@ impl Expression { lhs: Self::new(lhs).into(), rhs: Self::new(rhs).into(), }, - Join { lhs, rhs } => Self::Join { - lhs: lhs.as_ref().map(|lhs| Self::new(lhs).into()), - rhs: Self::new(rhs).into(), - }, Conditional { condition: full::Condition { lhs, rhs, operator }, otherwise, @@ -323,13 +329,21 @@ impl Expression { rhs: Self::new(rhs).into(), then: Self::new(then).into(), }, + Group { contents } => Self::new(contents), + Join { lhs, rhs } => Self::Join { + lhs: lhs.as_ref().map(|lhs| Self::new(lhs).into()), + rhs: Self::new(rhs).into(), + }, + Or { lhs, rhs } => Self::Or { + lhs: Self::new(lhs).into(), + rhs: Self::new(rhs).into(), + }, StringLiteral { string_literal } => Self::String { text: string_literal.cooked.clone(), }, Variable { name, .. } => Self::Variable { name: name.lexeme().to_owned(), }, - Group { contents } => Self::new(contents), } } } diff --git a/src/token_kind.rs b/src/token_kind.rs index 0db15d2dfa..850afa9629 100644 --- a/src/token_kind.rs +++ b/src/token_kind.rs @@ -7,6 +7,7 @@ pub(crate) enum TokenKind { At, Backtick, BangEquals, + BarBar, BraceL, BraceR, BracketL, @@ -50,6 +51,7 @@ impl Display for TokenKind { At => "'@'", Backtick => "backtick", BangEquals => "'!='", + BarBar => "'||'", BraceL => "'{'", BraceR => "'}'", BracketL => "'['", diff --git a/src/unstable_feature.rs b/src/unstable_feature.rs index 07d99540e5..70e26fabcf 100644 --- a/src/unstable_feature.rs +++ b/src/unstable_feature.rs @@ -3,6 +3,7 @@ use super::*; #[derive(Copy, Clone, Debug, PartialEq, Ord, Eq, PartialOrd)] pub(crate) enum UnstableFeature { FormatSubcommand, + LogicalOperators, ScriptAttribute, ScriptInterpreterSetting, } @@ -11,6 +12,10 @@ impl Display for UnstableFeature { fn fmt(&self, f: &mut Formatter) -> fmt::Result { match self { Self::FormatSubcommand => write!(f, "The `--fmt` command is currently unstable."), + Self::LogicalOperators => write!( + f, + "The logical operators `&&` and `||` are currently unstable." + ), Self::ScriptAttribute => write!(f, "The `[script]` attribute is currently unstable."), Self::ScriptInterpreterSetting => { write!(f, "The `script-interpreter` setting is currently unstable.") diff --git a/src/variables.rs b/src/variables.rs index 5797956366..9de1c98ce8 100644 --- a/src/variables.rs +++ b/src/variables.rs @@ -16,7 +16,24 @@ impl<'expression, 'src> Iterator for Variables<'expression, 'src> { fn next(&mut self) -> Option> { loop { match self.stack.pop()? { - Expression::StringLiteral { .. } | Expression::Backtick { .. } => {} + Expression::And { lhs, rhs } | Expression::Or { lhs, rhs } => { + self.stack.push(lhs); + self.stack.push(rhs); + } + Expression::Assert { + condition: + Condition { + lhs, + rhs, + operator: _, + }, + error, + } => { + self.stack.push(error); + self.stack.push(rhs); + self.stack.push(lhs); + } + Expression::Backtick { .. } | Expression::StringLiteral { .. } => {} Expression::Call { thunk } => match thunk { Thunk::Nullary { .. } => {} Thunk::Unary { arg, .. } => self.stack.push(arg), @@ -56,6 +73,10 @@ impl<'expression, 'src> Iterator for Variables<'expression, 'src> { } } }, + Expression::Concatenation { lhs, rhs } => { + self.stack.push(rhs); + self.stack.push(lhs); + } Expression::Conditional { condition: Condition { @@ -71,10 +92,8 @@ impl<'expression, 'src> Iterator for Variables<'expression, 'src> { self.stack.push(rhs); self.stack.push(lhs); } - Expression::Variable { name, .. } => return Some(name.token), - Expression::Concatenation { lhs, rhs } => { - self.stack.push(rhs); - self.stack.push(lhs); + Expression::Group { contents } => { + self.stack.push(contents); } Expression::Join { lhs, rhs } => { self.stack.push(rhs); @@ -82,22 +101,7 @@ impl<'expression, 'src> Iterator for Variables<'expression, 'src> { self.stack.push(lhs); } } - Expression::Group { contents } => { - self.stack.push(contents); - } - Expression::Assert { - condition: - Condition { - lhs, - rhs, - operator: _, - }, - error, - } => { - self.stack.push(error); - self.stack.push(rhs); - self.stack.push(lhs); - } + Expression::Variable { name, .. } => return Some(name.token), } } } diff --git a/tests/assert_success.rs b/tests/assert_success.rs index bcb364f880..f9202b7f2e 100644 --- a/tests/assert_success.rs +++ b/tests/assert_success.rs @@ -1,3 +1,4 @@ +#[track_caller] pub(crate) fn assert_success(output: &std::process::Output) { if !output.status.success() { eprintln!("stderr: {}", String::from_utf8_lossy(&output.stderr)); diff --git a/tests/conditional.rs b/tests/conditional.rs index c17392405c..4eab2f4d72 100644 --- a/tests/conditional.rs +++ b/tests/conditional.rs @@ -136,7 +136,7 @@ test! { ", stdout: "", stderr: " - error: Expected '!=', '==', '=~', '+', or '/', but found identifier + error: Expected '&&', '!=', '||', '==', '=~', '+', or '/', but found identifier ——▶ justfile:1:12 │ 1 │ a := if '' a '' { '' } else { b } diff --git a/tests/ignore_comments.rs b/tests/ignore_comments.rs index c3028a57a3..3f068227a4 100644 --- a/tests/ignore_comments.rs +++ b/tests/ignore_comments.rs @@ -125,7 +125,7 @@ fn comments_still_must_be_parsable_when_ignored() { ) .stderr( " - error: Expected '}}', '(', '+', or '/', but found identifier + error: Expected '&&', '||', '}}', '(', '+', or '/', but found identifier ——▶ justfile:4:12 │ 4 │ # {{ foo bar }} diff --git a/tests/lib.rs b/tests/lib.rs index ec71c665da..7c85460b44 100644 --- a/tests/lib.rs +++ b/tests/lib.rs @@ -73,6 +73,7 @@ mod invocation_directory; mod json; mod line_prefixes; mod list; +mod logical_operators; mod man; mod misc; mod modules; diff --git a/tests/logical_operators.rs b/tests/logical_operators.rs new file mode 100644 index 0000000000..5baa0f52f5 --- /dev/null +++ b/tests/logical_operators.rs @@ -0,0 +1,83 @@ +use super::*; + +#[track_caller] +fn evaluate(expression: &str, expected: &str) { + Test::new() + .justfile(format!("x := {expression}")) + .env("JUST_UNSTABLE", "1") + .args(["--evaluate", "x"]) + .stdout(expected) + .run(); +} + +#[test] +fn logical_operators_are_unstable() { + Test::new() + .justfile("x := 'foo' && 'bar'") + .args(["--evaluate", "x"]) + .stderr_regex(r"error: The logical operators `&&` and `\|\|` are currently unstable. .*") + .status(EXIT_FAILURE) + .run(); + + Test::new() + .justfile("x := 'foo' || 'bar'") + .args(["--evaluate", "x"]) + .stderr_regex(r"error: The logical operators `&&` and `\|\|` are currently unstable. .*") + .status(EXIT_FAILURE) + .run(); +} + +#[test] +fn and_returns_empty_string_if_lhs_is_empty() { + evaluate("'' && 'hello'", ""); +} + +#[test] +fn and_returns_rhs_if_lhs_is_non_empty() { + evaluate("'hello' && 'goodbye'", "goodbye"); +} + +#[test] +fn and_has_lower_precedence_than_plus() { + evaluate("'' && 'goodbye' + 'foo'", ""); + + evaluate("'foo' + 'hello' && 'goodbye'", "goodbye"); + + evaluate("'foo' + '' && 'goodbye'", "goodbye"); + + evaluate("'foo' + 'hello' && 'goodbye' + 'bar'", "goodbyebar"); +} + +#[test] +fn or_returns_rhs_if_lhs_is_empty() { + evaluate("'' || 'hello'", "hello"); +} + +#[test] +fn or_returns_lhs_if_lhs_is_non_empty() { + evaluate("'hello' || 'goodbye'", "hello"); +} + +#[test] +fn or_has_lower_precedence_than_plus() { + evaluate("'' || 'goodbye' + 'foo'", "goodbyefoo"); + + evaluate("'foo' + 'hello' || 'goodbye'", "foohello"); + + evaluate("'foo' + '' || 'goodbye'", "foo"); + + evaluate("'foo' + 'hello' || 'goodbye' + 'bar'", "foohello"); +} + +#[test] +fn and_has_higher_precedence_than_or() { + evaluate("('' && 'foo') || 'bar'", "bar"); + evaluate("'' && 'foo' || 'bar'", "bar"); + evaluate("'a' && 'b' || 'c'", "b"); +} + +#[test] +fn nesting() { + evaluate("'' || '' || '' || '' || 'foo'", "foo"); + evaluate("'foo' && 'foo' && 'foo' && 'foo' && 'bar'", "bar"); +} diff --git a/tests/shell_expansion.rs b/tests/shell_expansion.rs index 67fdc07d99..954e2a33db 100644 --- a/tests/shell_expansion.rs +++ b/tests/shell_expansion.rs @@ -25,7 +25,7 @@ fn shell_expanded_strings_must_not_have_whitespace() { .status(1) .stderr( " - error: Expected comment, end of file, end of line, '(', '+', or '/', but found string + error: Expected '&&', '||', comment, end of file, end of line, '(', '+', or '/', but found string ——▶ justfile:1:8 │ 1 │ x := x '$JUST_TEST_VARIABLE'