template: add support for logical equality operator
Some checks are pending
binaries / Build binary artifacts (linux-aarch64-gnu, ubuntu-24.04, aarch64-unknown-linux-gnu) (push) Waiting to run
binaries / Build binary artifacts (linux-aarch64-musl, ubuntu-24.04, aarch64-unknown-linux-musl) (push) Waiting to run
binaries / Build binary artifacts (linux-x86_64-gnu, ubuntu-24.04, x86_64-unknown-linux-gnu) (push) Waiting to run
binaries / Build binary artifacts (linux-x86_64-musl, ubuntu-24.04, x86_64-unknown-linux-musl) (push) Waiting to run
binaries / Build binary artifacts (macos-aarch64, macos-14, aarch64-apple-darwin) (push) Waiting to run
binaries / Build binary artifacts (macos-x86_64, macos-13, x86_64-apple-darwin) (push) Waiting to run
binaries / Build binary artifacts (win-x86_64, windows-2022, x86_64-pc-windows-msvc) (push) Waiting to run
nix / flake check (macos-14) (push) Waiting to run
nix / flake check (ubuntu-latest) (push) Waiting to run
build / build (, macos-13) (push) Waiting to run
build / build (, macos-14) (push) Waiting to run
build / build (, ubuntu-latest) (push) Waiting to run
build / build (, windows-latest) (push) Waiting to run
build / build (--all-features, ubuntu-latest) (push) Waiting to run
build / Build jj-lib without Git support (push) Waiting to run
build / Check protos (push) Waiting to run
build / Check formatting (push) Waiting to run
build / Check that MkDocs can build the docs (push) Waiting to run
build / Check that MkDocs can build the docs with Poetry 1.8 (push) Waiting to run
build / cargo-deny (advisories) (push) Waiting to run
build / cargo-deny (bans licenses sources) (push) Waiting to run
build / Clippy check (push) Waiting to run
Codespell / Codespell (push) Waiting to run
website / prerelease-docs-build-deploy (ubuntu-latest) (push) Waiting to run
Scorecards supply-chain security / Scorecards analysis (push) Waiting to run

This commit is contained in:
Benjamin Tan 2024-11-06 21:18:34 +08:00
parent aa040021da
commit 1372b39341
6 changed files with 127 additions and 11 deletions

View file

@ -14,6 +14,9 @@ to [Semantic Versioning](https://semver.org/spec/v2.0.0.html).
### New features
* Templates now support the `==` logical operator for `Boolean`, `Integer`, and
`String` types.
### Fixed bugs
## [0.23.0] - 2024-11-06

View file

@ -40,10 +40,11 @@ identifier = @{ (ASCII_ALPHA | "_") ~ (ASCII_ALPHANUMERIC | "_")* }
concat_op = { "++" }
logical_or_op = { "||" }
logical_and_op = { "&&" }
logical_eq_op = { "==" }
logical_not_op = { "!" }
negate_op = { "-" }
prefix_ops = _{ logical_not_op | negate_op }
infix_ops = _{ logical_or_op | logical_and_op }
infix_ops = _{ logical_or_op | logical_and_op | logical_eq_op }
function = { identifier ~ "(" ~ whitespace* ~ function_arguments ~ whitespace* ~ ")" }
keyword_argument = { identifier ~ whitespace* ~ "=" ~ whitespace* ~ template }

View file

@ -605,6 +605,41 @@ fn build_binary_operation<'a, L: TemplateLanguage<'a> + ?Sized>(
let out = lhs.and_then(move |l| Ok(l && rhs.extract()?));
Ok(L::wrap_boolean(out))
}
BinaryOp::LogicalEq => {
let lhs = build_expression(language, diagnostics, build_ctx, lhs_node)?;
let rhs = build_expression(language, diagnostics, build_ctx, rhs_node)?;
// TODO: Implement as a method on Property type
// e.g., lhs.try_eq(rhs) -> Option<Box<dyn TemplateProperty<Output = bool>>>
match (lhs.type_name(), rhs.type_name()) {
("Boolean", "Boolean") => {
let lhs = lhs.try_into_boolean().unwrap();
let rhs = rhs.try_into_boolean().unwrap();
let out = lhs.and_then(move |l| Ok(l == rhs.extract()?));
Ok(L::wrap_boolean(out))
}
("Integer", "Integer") => {
let lhs = lhs.try_into_integer().unwrap();
let rhs = rhs.try_into_integer().unwrap();
let out = lhs.and_then(move |l| Ok(l == rhs.extract()?));
Ok(L::wrap_boolean(out))
}
("String", "String") => {
let lhs = lhs.try_into_plain_text().unwrap();
let rhs = rhs.try_into_plain_text().unwrap();
let out = lhs.and_then(move |l| Ok(l == rhs.extract()?));
Ok(L::wrap_boolean(out))
}
(lhs_type @ ("Boolean" | "Integer" | "String"), rhs_type) => Err(
TemplateParseError::expected_type(lhs_type, rhs_type, rhs_node.span),
),
(lhs_type, _) => {
let message = format!(
r#"Expected expression of type "Boolean", "Integer", or "String", but actual type is "{lhs_type}""#
);
Err(TemplateParseError::expression(message, lhs_node.span))
}
}
}
}
}
@ -1677,14 +1712,14 @@ mod tests {
env.add_keyword("description", || L::wrap_string(Literal("".to_owned())));
env.add_keyword("empty", || L::wrap_boolean(Literal(true)));
insta::assert_snapshot!(env.parse_err(r#"description ()"#), @r###"
insta::assert_snapshot!(env.parse_err(r#"description ()"#), @r#"
--> 1:13
|
1 | description ()
| ^---
|
= expected <EOI>, `++`, `||`, or `&&`
"###);
= expected <EOI>, `++`, `||`, `&&`, or `==`
"#);
insta::assert_snapshot!(env.parse_err(r#"foo"#), @r###"
--> 1:1
@ -1728,6 +1763,62 @@ mod tests {
|
= Expected expression of type "Boolean", but actual type is "Integer"
"###);
insta::assert_snapshot!(env.parse_err(r#"true == 1"#), @r#"
--> 1:9
|
1 | true == 1
| ^
|
= Expected expression of type "Boolean", but actual type is "Integer"
"#);
insta::assert_snapshot!(env.parse_err(r#"true == 'a'"#), @r#"
--> 1:9
|
1 | true == 'a'
| ^-^
|
= Expected expression of type "Boolean", but actual type is "String"
"#);
insta::assert_snapshot!(env.parse_err(r#"1 == true"#), @r#"
--> 1:6
|
1 | 1 == true
| ^--^
|
= Expected expression of type "Integer", but actual type is "Boolean"
"#);
insta::assert_snapshot!(env.parse_err(r#"1 == 'a'"#), @r#"
--> 1:6
|
1 | 1 == 'a'
| ^-^
|
= Expected expression of type "Integer", but actual type is "String"
"#);
insta::assert_snapshot!(env.parse_err(r#"'a' == true"#), @r#"
--> 1:8
|
1 | 'a' == true
| ^--^
|
= Expected expression of type "String", but actual type is "Boolean"
"#);
insta::assert_snapshot!(env.parse_err(r#"'a' == 1"#), @r#"
--> 1:8
|
1 | 'a' == 1
| ^
|
= Expected expression of type "String", but actual type is "Integer"
"#);
insta::assert_snapshot!(env.parse_err(r#"'a' == label("", "")"#), @r#"
--> 1:8
|
1 | 'a' == label("", "")
| ^-----------^
|
= Expected expression of type "String", but actual type is "Template"
"#);
insta::assert_snapshot!(env.parse_err(r#"description.first_line().foo()"#), @r###"
--> 1:26
@ -1945,6 +2036,12 @@ mod tests {
insta::assert_snapshot!(env.render_ok(r#"!false"#), @"true");
insta::assert_snapshot!(env.render_ok(r#"false || !false"#), @"true");
insta::assert_snapshot!(env.render_ok(r#"false && true"#), @"false");
insta::assert_snapshot!(env.render_ok(r#"true == true"#), @"true");
insta::assert_snapshot!(env.render_ok(r#"true == false"#), @"false");
insta::assert_snapshot!(env.render_ok(r#"1 == 1"#), @"true");
insta::assert_snapshot!(env.render_ok(r#"1 == 2"#), @"false");
insta::assert_snapshot!(env.render_ok(r#"'a' == 'a'"#), @"true");
insta::assert_snapshot!(env.render_ok(r#"'a' == 'b'"#), @"false");
insta::assert_snapshot!(env.render_ok(r#" !"" "#), @"true");
insta::assert_snapshot!(env.render_ok(r#" "" || "a".lines() "#), @"true");

View file

@ -74,6 +74,7 @@ impl Rule {
Rule::concat_op => Some("++"),
Rule::logical_or_op => Some("||"),
Rule::logical_and_op => Some("&&"),
Rule::logical_eq_op => Some("=="),
Rule::logical_not_op => Some("!"),
Rule::negate_op => Some("-"),
Rule::prefix_ops => None,
@ -374,6 +375,8 @@ pub enum BinaryOp {
LogicalOr,
/// `&&`
LogicalAnd,
/// `==`
LogicalEq,
}
pub type ExpressionNode<'i> = dsl_util::ExpressionNode<'i, ExpressionKind<'i>>;
@ -504,6 +507,7 @@ fn parse_expression_node(pair: Pair<Rule>) -> TemplateParseResult<ExpressionNode
PrattParser::new()
.op(Op::infix(Rule::logical_or_op, Assoc::Left))
.op(Op::infix(Rule::logical_and_op, Assoc::Left))
.op(Op::infix(Rule::logical_eq_op, Assoc::Left))
.op(Op::prefix(Rule::logical_not_op) | Op::prefix(Rule::negate_op))
});
PRATT
@ -523,6 +527,7 @@ fn parse_expression_node(pair: Pair<Rule>) -> TemplateParseResult<ExpressionNode
let op_kind = match op.as_rule() {
Rule::logical_or_op => BinaryOp::LogicalOr,
Rule::logical_and_op => BinaryOp::LogicalAnd,
Rule::logical_eq_op => BinaryOp::LogicalEq,
r => panic!("unexpected infix operator rule {r:?}"),
};
let lhs = Box::new(lhs?);
@ -851,12 +856,16 @@ mod tests {
parse_normalized("(!(x.f())) || (!(g()))"),
);
assert_eq!(
parse_normalized("x.f() || y || z"),
parse_normalized("((x.f()) || y) || z"),
parse_normalized("!x.f() == !x.f() || !g() == !g()"),
parse_normalized("((!(x.f())) == (!(x.f()))) || ((!(g())) == (!(g())))"),
);
assert_eq!(
parse_normalized("x || y && z.h()"),
parse_normalized("x || (y && (z.h()))"),
parse_normalized("x.f() || y == y || z"),
parse_normalized("((x.f()) || (y == y)) || z"),
);
assert_eq!(
parse_normalized("x || y == y && z.h() == z"),
parse_normalized("x || ((y == y) && ((z.h()) == z))"),
);
// Logical operator bounds more tightly than concatenation. This might
@ -869,6 +878,10 @@ mod tests {
parse_normalized(r"x ++ y || z"),
parse_normalized(r"x ++ (y || z)"),
);
assert_eq!(
parse_normalized(r"x == y ++ z"),
parse_normalized(r"(x == y) ++ z"),
);
// Expression span
assert_eq!(parse_template(" ! x ").unwrap().span.as_str(), "! x");

View file

@ -25,15 +25,15 @@ fn test_templater_parse_error() {
let repo_path = test_env.env_root().join("repo");
let render_err = |template| test_env.jj_cmd_failure(&repo_path, &["log", "-T", template]);
insta::assert_snapshot!(render_err(r#"description ()"#), @r###"
insta::assert_snapshot!(render_err(r#"description ()"#), @r#"
Error: Failed to parse template: Syntax error
Caused by: --> 1:13
|
1 | description ()
| ^---
|
= expected <EOI>, `++`, `||`, or `&&`
"###);
= expected <EOI>, `++`, `||`, `&&`, or `==`
"#);
// Typo
test_env.add_config(

View file

@ -31,6 +31,8 @@ The following operators are supported.
* `x.f()`: Method call.
* `-x`: Negate integer value.
* `!x`: Logical not.
* `x == y`: Logical equal. Operands must be either `Boolean`, `Integer`, or
`String`.
* `x && y`: Logical and, short-circuiting.
* `x || y`: Logical or, short-circuiting.
* `x ++ y`: Concatenate `x` and `y` templates.