// Copyright 2020-2023 The Jujutsu Authors // // Licensed under the Apache License, Version 2.0 (the "License"); // you may not use this file except in compliance with the License. // You may obtain a copy of the License at // // https://www.apache.org/licenses/LICENSE-2.0 // // Unless required by applicable law or agreed to in writing, software // distributed under the License is distributed on an "AS IS" BASIS, // WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied. // See the License for the specific language governing permissions and // limitations under the License. use std::collections::HashMap; use itertools::Itertools as _; use jj_lib::backend::{Signature, Timestamp}; use crate::template_parser::{ self, ExpressionKind, ExpressionNode, FunctionCallNode, MethodCallNode, TemplateParseError, TemplateParseResult, }; use crate::templater::{ ConcatTemplate, ConditionalTemplate, IntoTemplate, LabelTemplate, ListPropertyTemplate, ListTemplate, Literal, PlainTextFormattedProperty, PropertyPlaceholder, ReformatTemplate, SeparateTemplate, Template, TemplateFunction, TemplateProperty, TimestampRange, }; use crate::{text_util, time_util}; /// Callbacks to build language-specific evaluation objects from AST nodes. pub trait TemplateLanguage<'a> { type Context: 'a; type Property: IntoTemplateProperty<'a, Self::Context>; fn wrap_string( &self, property: impl TemplateProperty + 'a, ) -> Self::Property; fn wrap_string_list( &self, property: impl TemplateProperty> + 'a, ) -> Self::Property; fn wrap_boolean( &self, property: impl TemplateProperty + 'a, ) -> Self::Property; fn wrap_integer( &self, property: impl TemplateProperty + 'a, ) -> Self::Property; fn wrap_signature( &self, property: impl TemplateProperty + 'a, ) -> Self::Property; fn wrap_timestamp( &self, property: impl TemplateProperty + 'a, ) -> Self::Property; fn wrap_timestamp_range( &self, property: impl TemplateProperty + 'a, ) -> Self::Property; fn wrap_template(&self, template: Box + 'a>) -> Self::Property; fn wrap_list_template( &self, template: Box + 'a>, ) -> Self::Property; fn build_keyword(&self, name: &str, span: pest::Span) -> TemplateParseResult; fn build_method( &self, build_ctx: &BuildContext, property: Self::Property, function: &FunctionCallNode, ) -> TemplateParseResult; } /// Implements `TemplateLanguage::wrap_()` functions. /// /// - `impl_core_wrap_property_fns('a)` for `CoreTemplatePropertyKind`, /// - `impl_core_wrap_property_fns('a, MyKind::Core)` for `MyKind::Core(..)`. macro_rules! impl_core_wrap_property_fns { ($a:lifetime) => { $crate::template_builder::impl_core_wrap_property_fns!($a, std::convert::identity); }; ($a:lifetime, $outer:path) => { $crate::template_builder::impl_wrap_property_fns!( $a, $crate::template_builder::CoreTemplatePropertyKind, $outer, { wrap_string(String) => String, wrap_string_list(Vec) => StringList, wrap_boolean(bool) => Boolean, wrap_integer(i64) => Integer, wrap_signature(jj_lib::backend::Signature) => Signature, wrap_timestamp(jj_lib::backend::Timestamp) => Timestamp, wrap_timestamp_range($crate::templater::TimestampRange) => TimestampRange, } ); fn wrap_template( &self, template: Box + $a>, ) -> Self::Property { use $crate::template_builder::CoreTemplatePropertyKind as Kind; $outer(Kind::Template(template)) } fn wrap_list_template( &self, template: Box + $a>, ) -> Self::Property { use $crate::template_builder::CoreTemplatePropertyKind as Kind; $outer(Kind::ListTemplate(template)) } }; } macro_rules! impl_wrap_property_fns { ($a:lifetime, $kind:path, $outer:path, { $( $func:ident($ty:ty) => $var:ident, )+ }) => { $( fn $func( &self, property: impl $crate::templater::TemplateProperty< Self::Context, Output = $ty> + $a, ) -> Self::Property { use $kind as Kind; // https://github.com/rust-lang/rust/issues/48067 $outer(Kind::$var(Box::new(property))) } )+ }; } pub(crate) use {impl_core_wrap_property_fns, impl_wrap_property_fns}; /// Provides access to basic template property types. pub trait IntoTemplateProperty<'a, C> { fn try_into_boolean(self) -> Option + 'a>>; fn try_into_integer(self) -> Option + 'a>>; fn try_into_plain_text(self) -> Option + 'a>>; fn try_into_template(self) -> Option + 'a>>; } pub enum CoreTemplatePropertyKind<'a, I> { String(Box + 'a>), StringList(Box> + 'a>), Boolean(Box + 'a>), Integer(Box + 'a>), Signature(Box + 'a>), Timestamp(Box + 'a>), TimestampRange(Box + 'a>), // Similar to `TemplateProperty + 'a>`, but doesn't // capture `I` to produce `Template<()>`. The context `I` would have to be cloned // to convert `Template` to `Template<()>`. Template(Box + 'a>), ListTemplate(Box + 'a>), } impl<'a, I: 'a> IntoTemplateProperty<'a, I> for CoreTemplatePropertyKind<'a, I> { fn try_into_boolean(self) -> Option + 'a>> { match self { CoreTemplatePropertyKind::String(property) => { Some(Box::new(TemplateFunction::new(property, |s| !s.is_empty()))) } CoreTemplatePropertyKind::StringList(property) => { Some(Box::new(TemplateFunction::new(property, |l| !l.is_empty()))) } CoreTemplatePropertyKind::Boolean(property) => Some(property), CoreTemplatePropertyKind::Integer(_) => None, CoreTemplatePropertyKind::Signature(_) => None, CoreTemplatePropertyKind::Timestamp(_) => None, CoreTemplatePropertyKind::TimestampRange(_) => None, // Template types could also be evaluated to boolean, but it's less likely // to apply label() or .map() and use the result as conditional. It's also // unclear whether ListTemplate should behave as a "list" or a "template". CoreTemplatePropertyKind::Template(_) => None, CoreTemplatePropertyKind::ListTemplate(_) => None, } } fn try_into_integer(self) -> Option + 'a>> { match self { CoreTemplatePropertyKind::Integer(property) => Some(property), _ => None, } } fn try_into_plain_text(self) -> Option + 'a>> { match self { CoreTemplatePropertyKind::String(property) => Some(property), _ => { let template = self.try_into_template()?; Some(Box::new(PlainTextFormattedProperty::new(template))) } } } fn try_into_template(self) -> Option + 'a>> { match self { CoreTemplatePropertyKind::String(property) => Some(property.into_template()), CoreTemplatePropertyKind::StringList(property) => Some(property.into_template()), CoreTemplatePropertyKind::Boolean(property) => Some(property.into_template()), CoreTemplatePropertyKind::Integer(property) => Some(property.into_template()), CoreTemplatePropertyKind::Signature(property) => Some(property.into_template()), CoreTemplatePropertyKind::Timestamp(property) => Some(property.into_template()), CoreTemplatePropertyKind::TimestampRange(property) => Some(property.into_template()), CoreTemplatePropertyKind::Template(template) => Some(template), CoreTemplatePropertyKind::ListTemplate(template) => Some(template.into_template()), } } } /// Opaque struct that represents a template value. pub struct Expression

{ property: P, labels: Vec, } impl

Expression

{ fn unlabeled(property: P) -> Self { let labels = vec![]; Expression { property, labels } } fn with_label(property: P, label: impl Into) -> Self { let labels = vec![label.into()]; Expression { property, labels } } pub fn try_into_boolean<'a, C: 'a>( self, ) -> Option + 'a>> where P: IntoTemplateProperty<'a, C>, { self.property.try_into_boolean() } pub fn try_into_integer<'a, C: 'a>( self, ) -> Option + 'a>> where P: IntoTemplateProperty<'a, C>, { self.property.try_into_integer() } pub fn try_into_plain_text<'a, C: 'a>( self, ) -> Option + 'a>> where P: IntoTemplateProperty<'a, C>, { self.property.try_into_plain_text() } pub fn try_into_template<'a, C: 'a>(self) -> Option + 'a>> where P: IntoTemplateProperty<'a, C>, { let template = self.property.try_into_template()?; if self.labels.is_empty() { Some(template) } else { Some(Box::new(LabelTemplate::new(template, Literal(self.labels)))) } } } pub struct BuildContext<'i, P> { /// Map of functions to create `L::Property`. local_variables: HashMap<&'i str, &'i (dyn Fn() -> P)>, } fn build_method_call<'a, L: TemplateLanguage<'a>>( language: &L, build_ctx: &BuildContext, method: &MethodCallNode, ) -> TemplateParseResult> { let mut expression = build_expression(language, build_ctx, &method.object)?; expression.property = language.build_method(build_ctx, expression.property, &method.function)?; expression.labels.push(method.function.name.to_owned()); Ok(expression) } pub fn build_core_method<'a, L: TemplateLanguage<'a>>( language: &L, build_ctx: &BuildContext, property: CoreTemplatePropertyKind<'a, L::Context>, function: &FunctionCallNode, ) -> TemplateParseResult { match property { CoreTemplatePropertyKind::String(property) => { build_string_method(language, build_ctx, property, function) } CoreTemplatePropertyKind::StringList(property) => { build_formattable_list_method(language, build_ctx, property, function, |item| { language.wrap_string(item) }) } CoreTemplatePropertyKind::Boolean(property) => { build_boolean_method(language, build_ctx, property, function) } CoreTemplatePropertyKind::Integer(property) => { build_integer_method(language, build_ctx, property, function) } CoreTemplatePropertyKind::Signature(property) => { build_signature_method(language, build_ctx, property, function) } CoreTemplatePropertyKind::Timestamp(property) => { build_timestamp_method(language, build_ctx, property, function) } CoreTemplatePropertyKind::TimestampRange(property) => { build_timestamp_range_method(language, build_ctx, property, function) } CoreTemplatePropertyKind::Template(_) => { Err(TemplateParseError::no_such_method("Template", function)) } CoreTemplatePropertyKind::ListTemplate(template) => { build_list_template_method(language, build_ctx, template, function) } } } fn build_string_method<'a, L: TemplateLanguage<'a>>( language: &L, build_ctx: &BuildContext, self_property: impl TemplateProperty + 'a, function: &FunctionCallNode, ) -> TemplateParseResult { let property = match function.name { "contains" => { let [needle_node] = template_parser::expect_exact_arguments(function)?; // TODO: or .try_into_string() to disable implicit type cast? let needle_property = expect_plain_text_expression(language, build_ctx, needle_node)?; language.wrap_boolean(TemplateFunction::new( (self_property, needle_property), |(haystack, needle)| haystack.contains(&needle), )) } "starts_with" => { let [needle_node] = template_parser::expect_exact_arguments(function)?; let needle_property = expect_plain_text_expression(language, build_ctx, needle_node)?; language.wrap_boolean(TemplateFunction::new( (self_property, needle_property), move |(haystack, needle)| haystack.starts_with(&needle), )) } "ends_with" => { let [needle_node] = template_parser::expect_exact_arguments(function)?; let needle_property = expect_plain_text_expression(language, build_ctx, needle_node)?; language.wrap_boolean(TemplateFunction::new( (self_property, needle_property), move |(haystack, needle)| haystack.ends_with(&needle), )) } "remove_prefix" => { let [needle_node] = template_parser::expect_exact_arguments(function)?; let needle_property = expect_plain_text_expression(language, build_ctx, needle_node)?; language.wrap_string(TemplateFunction::new( (self_property, needle_property), move |(haystack, needle)| { haystack .strip_prefix(&needle) .map(ToOwned::to_owned) .unwrap_or(haystack) }, )) } "remove_suffix" => { let [needle_node] = template_parser::expect_exact_arguments(function)?; let needle_property = expect_plain_text_expression(language, build_ctx, needle_node)?; language.wrap_string(TemplateFunction::new( (self_property, needle_property), move |(haystack, needle)| { haystack .strip_suffix(&needle) .map(ToOwned::to_owned) .unwrap_or(haystack) }, )) } "substr" => { let [start_idx, end_idx] = template_parser::expect_exact_arguments(function)?; let start_idx_property = expect_integer_expression(language, build_ctx, start_idx)?; let end_idx_property = expect_integer_expression(language, build_ctx, end_idx)?; language.wrap_string(TemplateFunction::new( (self_property, start_idx_property, end_idx_property), |(s, start_idx, end_idx)| { // TODO: If we add .len() method, we'll expose bytes-based and char-based APIs. // Having different index units would be confusing, so we might want to change // .substr() to bytes-based and round up/down towards char or grapheme-cluster // boundary. let to_idx = |i: i64| -> usize { let magnitude = usize::try_from(i.unsigned_abs()).unwrap_or(usize::MAX); if i < 0 { s.chars().count().saturating_sub(magnitude) } else { magnitude } }; let start_idx = to_idx(start_idx); let end_idx = to_idx(end_idx); if start_idx >= end_idx { String::new() } else { s.chars() .skip(start_idx) .take(end_idx - start_idx) .collect() } }, )) } "first_line" => { template_parser::expect_no_arguments(function)?; language.wrap_string(TemplateFunction::new(self_property, |s| { s.lines().next().unwrap_or_default().to_string() })) } "lines" => { template_parser::expect_no_arguments(function)?; language.wrap_string_list(TemplateFunction::new(self_property, |s| { s.lines().map(|l| l.to_owned()).collect() })) } "upper" => { template_parser::expect_no_arguments(function)?; language.wrap_string(TemplateFunction::new(self_property, |s| s.to_uppercase())) } "lower" => { template_parser::expect_no_arguments(function)?; language.wrap_string(TemplateFunction::new(self_property, |s| s.to_lowercase())) } _ => return Err(TemplateParseError::no_such_method("String", function)), }; Ok(property) } fn build_boolean_method<'a, L: TemplateLanguage<'a>>( _language: &L, _build_ctx: &BuildContext, _self_property: impl TemplateProperty + 'a, function: &FunctionCallNode, ) -> TemplateParseResult { Err(TemplateParseError::no_such_method("Boolean", function)) } fn build_integer_method<'a, L: TemplateLanguage<'a>>( _language: &L, _build_ctx: &BuildContext, _self_property: impl TemplateProperty + 'a, function: &FunctionCallNode, ) -> TemplateParseResult { Err(TemplateParseError::no_such_method("Integer", function)) } fn build_signature_method<'a, L: TemplateLanguage<'a>>( language: &L, _build_ctx: &BuildContext, self_property: impl TemplateProperty + 'a, function: &FunctionCallNode, ) -> TemplateParseResult { let property = match function.name { "name" => { template_parser::expect_no_arguments(function)?; language.wrap_string(TemplateFunction::new(self_property, |signature| { signature.name })) } "email" => { template_parser::expect_no_arguments(function)?; language.wrap_string(TemplateFunction::new(self_property, |signature| { signature.email })) } "username" => { template_parser::expect_no_arguments(function)?; language.wrap_string(TemplateFunction::new(self_property, |signature| { let (username, _) = text_util::split_email(&signature.email); username.to_owned() })) } "timestamp" => { template_parser::expect_no_arguments(function)?; language.wrap_timestamp(TemplateFunction::new(self_property, |signature| { signature.timestamp })) } _ => return Err(TemplateParseError::no_such_method("Signature", function)), }; Ok(property) } fn build_timestamp_method<'a, L: TemplateLanguage<'a>>( language: &L, _build_ctx: &BuildContext, self_property: impl TemplateProperty + 'a, function: &FunctionCallNode, ) -> TemplateParseResult { let property = match function.name { "ago" => { template_parser::expect_no_arguments(function)?; language.wrap_string(TemplateFunction::new(self_property, |timestamp| { time_util::format_timestamp_relative_to_now(×tamp) })) } "format" => { // No dynamic string is allowed as the templater has no runtime error type. let [format_node] = template_parser::expect_exact_arguments(function)?; let format = template_parser::expect_string_literal_with(format_node, |format, span| { time_util::FormattingItems::parse(format).ok_or_else(|| { TemplateParseError::unexpected_expression("Invalid time format", span) }) })? .into_owned(); language.wrap_string(TemplateFunction::new(self_property, move |timestamp| { time_util::format_absolute_timestamp_with(×tamp, &format) })) } "utc" => { template_parser::expect_no_arguments(function)?; language.wrap_timestamp(TemplateFunction::new(self_property, |mut timestamp| { timestamp.tz_offset = 0; timestamp })) } _ => return Err(TemplateParseError::no_such_method("Timestamp", function)), }; Ok(property) } fn build_timestamp_range_method<'a, L: TemplateLanguage<'a>>( language: &L, _build_ctx: &BuildContext, self_property: impl TemplateProperty + 'a, function: &FunctionCallNode, ) -> TemplateParseResult { let property = match function.name { "start" => { template_parser::expect_no_arguments(function)?; language.wrap_timestamp(TemplateFunction::new(self_property, |time_range| { time_range.start })) } "end" => { template_parser::expect_no_arguments(function)?; language.wrap_timestamp(TemplateFunction::new(self_property, |time_range| { time_range.end })) } "duration" => { template_parser::expect_no_arguments(function)?; language.wrap_string(TemplateFunction::new(self_property, |time_range| { time_range.duration() })) } _ => { return Err(TemplateParseError::no_such_method( "TimestampRange", function, )) } }; Ok(property) } fn build_list_template_method<'a, L: TemplateLanguage<'a>>( language: &L, build_ctx: &BuildContext, self_template: Box + 'a>, function: &FunctionCallNode, ) -> TemplateParseResult { let property = match function.name { "join" => { let [separator_node] = template_parser::expect_exact_arguments(function)?; let separator = expect_template_expression(language, build_ctx, separator_node)?; language.wrap_template(self_template.join(separator)) } _ => return Err(TemplateParseError::no_such_method("ListTemplate", function)), }; Ok(property) } /// Builds method call expression for printable list property. pub fn build_formattable_list_method<'a, L, O>( language: &L, build_ctx: &BuildContext, self_property: impl TemplateProperty> + 'a, function: &FunctionCallNode, // TODO: Generic L: WrapProperty trait might be needed to support more // list operations such as first()/slice(). For .map(), a simple callback works. wrap_item: impl Fn(PropertyPlaceholder) -> L::Property, ) -> TemplateParseResult where L: TemplateLanguage<'a>, O: Template<()> + Clone + 'a, { let property = match function.name { "join" => { let [separator_node] = template_parser::expect_exact_arguments(function)?; let separator = expect_template_expression(language, build_ctx, separator_node)?; let template = ListPropertyTemplate::new(self_property, separator, |_, formatter, item| { item.format(&(), formatter) }); language.wrap_template(Box::new(template)) } "map" => build_map_operation(language, build_ctx, self_property, function, wrap_item)?, _ => return Err(TemplateParseError::no_such_method("List", function)), }; Ok(property) } pub fn build_unformattable_list_method<'a, L, O>( language: &L, build_ctx: &BuildContext, self_property: impl TemplateProperty> + 'a, function: &FunctionCallNode, wrap_item: impl Fn(PropertyPlaceholder) -> L::Property, ) -> TemplateParseResult where L: TemplateLanguage<'a>, O: Clone + 'a, { let property = match function.name { // No "join" "map" => build_map_operation(language, build_ctx, self_property, function, wrap_item)?, _ => return Err(TemplateParseError::no_such_method("List", function)), }; Ok(property) } /// Builds expression that extracts iterable property and applies template to /// each item. /// /// `wrap_item()` is the function to wrap a list item of type `O` as a property. fn build_map_operation<'a, L, O, P>( language: &L, build_ctx: &BuildContext, self_property: P, function: &FunctionCallNode, wrap_item: impl Fn(PropertyPlaceholder) -> L::Property, ) -> TemplateParseResult where L: TemplateLanguage<'a>, P: TemplateProperty + 'a, P::Output: IntoIterator, O: Clone + 'a, { // Build an item template with placeholder property, then evaluate it // for each item. // // It would be nice if we could build a template of (L::Context, O) // input, but doing that for a generic item type wouldn't be easy. It's // also invalid to convert &C to &(C, _). let [lambda_node] = template_parser::expect_exact_arguments(function)?; let item_placeholder = PropertyPlaceholder::new(); let item_template = template_parser::expect_lambda_with(lambda_node, |lambda, _span| { let item_fn = || wrap_item(item_placeholder.clone()); let mut local_variables = build_ctx.local_variables.clone(); if let [name] = lambda.params.as_slice() { local_variables.insert(name, &item_fn); } else { return Err(TemplateParseError::unexpected_expression( "Expected 1 lambda parameters", lambda.params_span, )); } let build_ctx = BuildContext { local_variables }; expect_template_expression(language, &build_ctx, &lambda.body) })?; let list_template = ListPropertyTemplate::new( self_property, Literal(" "), // separator move |context, formatter, item| { item_placeholder.with_value(item, || item_template.format(context, formatter)) }, ); Ok(language.wrap_list_template(Box::new(list_template))) } fn build_global_function<'a, L: TemplateLanguage<'a>>( language: &L, build_ctx: &BuildContext, function: &FunctionCallNode, ) -> TemplateParseResult> { let property = match function.name { "fill" => { let [width_node, content_node] = template_parser::expect_exact_arguments(function)?; let width = expect_integer_expression(language, build_ctx, width_node)?; let content = expect_template_expression(language, build_ctx, content_node)?; let template = ReformatTemplate::new(content, move |context, formatter, recorded| { let width = width.extract(context).try_into().unwrap_or(0); text_util::write_wrapped(formatter, recorded, width) }); language.wrap_template(Box::new(template)) } "indent" => { let [prefix_node, content_node] = template_parser::expect_exact_arguments(function)?; let prefix = expect_template_expression(language, build_ctx, prefix_node)?; let content = expect_template_expression(language, build_ctx, content_node)?; let template = ReformatTemplate::new(content, move |context, formatter, recorded| { text_util::write_indented(formatter, recorded, |formatter| { // If Template::format() returned a custom error type, we would need to // handle template evaluation error out of this closure: // prefix.format(context, &mut prefix_recorder)?; // write_indented(formatter, recorded, |formatter| { // prefix_recorder.replay(formatter) // }) prefix.format(context, formatter) }) }); language.wrap_template(Box::new(template)) } "label" => { let [label_node, content_node] = template_parser::expect_exact_arguments(function)?; let label_property = expect_plain_text_expression(language, build_ctx, label_node)?; let content = expect_template_expression(language, build_ctx, content_node)?; let labels = TemplateFunction::new(label_property, |s| { s.split_whitespace().map(ToString::to_string).collect() }); language.wrap_template(Box::new(LabelTemplate::new(content, labels))) } "if" => { let ([condition_node, true_node], [false_node]) = template_parser::expect_arguments(function)?; let condition = expect_boolean_expression(language, build_ctx, condition_node)?; let true_template = expect_template_expression(language, build_ctx, true_node)?; let false_template = false_node .map(|node| expect_template_expression(language, build_ctx, node)) .transpose()?; let template = ConditionalTemplate::new(condition, true_template, false_template); language.wrap_template(Box::new(template)) } "concat" => { let contents = function .args .iter() .map(|node| expect_template_expression(language, build_ctx, node)) .try_collect()?; language.wrap_template(Box::new(ConcatTemplate(contents))) } "separate" => { let ([separator_node], content_nodes) = template_parser::expect_some_arguments(function)?; let separator = expect_template_expression(language, build_ctx, separator_node)?; let contents = content_nodes .iter() .map(|node| expect_template_expression(language, build_ctx, node)) .try_collect()?; language.wrap_template(Box::new(SeparateTemplate::new(separator, contents))) } _ => return Err(TemplateParseError::no_such_function(function)), }; Ok(Expression::unlabeled(property)) } /// Builds intermediate expression tree from AST nodes. pub fn build_expression<'a, L: TemplateLanguage<'a>>( language: &L, build_ctx: &BuildContext, node: &ExpressionNode, ) -> TemplateParseResult> { match &node.kind { ExpressionKind::Identifier(name) => { if let Some(make) = build_ctx.local_variables.get(name) { // Don't label a local variable with its name Ok(Expression::unlabeled(make())) } else { let property = language.build_keyword(name, node.span)?; Ok(Expression::with_label(property, *name)) } } ExpressionKind::Boolean(value) => { let property = language.wrap_boolean(Literal(*value)); Ok(Expression::unlabeled(property)) } ExpressionKind::Integer(value) => { let property = language.wrap_integer(Literal(*value)); Ok(Expression::unlabeled(property)) } ExpressionKind::String(value) => { let property = language.wrap_string(Literal(value.clone())); Ok(Expression::unlabeled(property)) } ExpressionKind::Concat(nodes) => { let templates = nodes .iter() .map(|node| expect_template_expression(language, build_ctx, node)) .try_collect()?; let property = language.wrap_template(Box::new(ConcatTemplate(templates))); Ok(Expression::unlabeled(property)) } ExpressionKind::FunctionCall(function) => { build_global_function(language, build_ctx, function) } ExpressionKind::MethodCall(method) => build_method_call(language, build_ctx, method), ExpressionKind::Lambda(_) => Err(TemplateParseError::unexpected_expression( "Lambda cannot be defined here", node.span, )), ExpressionKind::AliasExpanded(id, subst) => build_expression(language, build_ctx, subst) .map_err(|e| e.within_alias_expansion(*id, node.span)), } } /// Builds template evaluation tree from AST nodes, with fresh build context. pub fn build<'a, L: TemplateLanguage<'a>>( language: &L, node: &ExpressionNode, ) -> TemplateParseResult + 'a>> { let build_ctx = BuildContext { local_variables: HashMap::new(), }; expect_template_expression(language, &build_ctx, node) } pub fn expect_boolean_expression<'a, L: TemplateLanguage<'a>>( language: &L, build_ctx: &BuildContext, node: &ExpressionNode, ) -> TemplateParseResult + 'a>> { build_expression(language, build_ctx, node)? .try_into_boolean() .ok_or_else(|| TemplateParseError::expected_type("Boolean", node.span)) } pub fn expect_integer_expression<'a, L: TemplateLanguage<'a>>( language: &L, build_ctx: &BuildContext, node: &ExpressionNode, ) -> TemplateParseResult + 'a>> { build_expression(language, build_ctx, node)? .try_into_integer() .ok_or_else(|| TemplateParseError::expected_type("Integer", node.span)) } pub fn expect_plain_text_expression<'a, L: TemplateLanguage<'a>>( language: &L, build_ctx: &BuildContext, node: &ExpressionNode, ) -> TemplateParseResult + 'a>> { // Since any formattable type can be converted to a string property, // the expected type is not a String, but a Template. build_expression(language, build_ctx, node)? .try_into_plain_text() .ok_or_else(|| TemplateParseError::expected_type("Template", node.span)) } pub fn expect_template_expression<'a, L: TemplateLanguage<'a>>( language: &L, build_ctx: &BuildContext, node: &ExpressionNode, ) -> TemplateParseResult + 'a>> { build_expression(language, build_ctx, node)? .try_into_template() .ok_or_else(|| TemplateParseError::expected_type("Template", node.span)) } #[cfg(test)] mod tests { use std::iter; use jj_lib::backend::MillisSinceEpoch; use super::*; use crate::formatter::{self, ColorFormatter}; use crate::template_parser::TemplateAliasesMap; /// Minimal template language for testing. #[derive(Clone, Default)] struct TestTemplateLanguage { keywords: HashMap<&'static str, TestTemplateKeywordFn>, } impl TemplateLanguage<'static> for TestTemplateLanguage { type Context = (); type Property = CoreTemplatePropertyKind<'static, ()>; impl_core_wrap_property_fns!('static); fn build_keyword( &self, name: &str, span: pest::Span, ) -> TemplateParseResult { self.keywords .get(name) .map(|f| f(self)) .ok_or_else(|| TemplateParseError::no_such_keyword(name, span)) } fn build_method( &self, build_ctx: &BuildContext, property: Self::Property, function: &FunctionCallNode, ) -> TemplateParseResult { build_core_method(self, build_ctx, property, function) } } type TestTemplateKeywordFn = fn(&TestTemplateLanguage) -> CoreTemplatePropertyKind<'static, ()>; /// Helper to set up template evaluation environment. #[derive(Clone, Default)] struct TestTemplateEnv { language: TestTemplateLanguage, aliases_map: TemplateAliasesMap, color_rules: Vec<(Vec, formatter::Style)>, } impl TestTemplateEnv { fn add_keyword(&mut self, name: &'static str, f: TestTemplateKeywordFn) { self.language.keywords.insert(name, f); } fn add_alias(&mut self, decl: impl AsRef, defn: impl Into) { self.aliases_map.insert(decl, defn).unwrap(); } fn add_color(&mut self, label: &str, fg_color: crossterm::style::Color) { let labels = label.split_whitespace().map(|s| s.to_owned()).collect(); let style = formatter::Style { fg_color: Some(fg_color), ..Default::default() }; self.color_rules.push((labels, style)); } fn parse(&self, template: &str) -> TemplateParseResult>> { let node = template_parser::parse(template, &self.aliases_map)?; build(&self.language, &node) } fn parse_err(&self, template: &str) -> String { let err = self.parse(template).err().unwrap(); iter::successors(Some(&err), |e| e.origin()).join("\n") } fn render_ok(&self, template: &str) -> String { let template = self.parse(template).unwrap(); let mut output = Vec::new(); let mut formatter = ColorFormatter::new(&mut output, self.color_rules.clone().into()); template.format(&(), &mut formatter).unwrap(); drop(formatter); String::from_utf8(output).unwrap() } } fn new_signature(name: &str, email: &str) -> Signature { Signature { name: name.to_owned(), email: email.to_owned(), timestamp: new_timestamp(0, 0), } } fn new_timestamp(msec: i64, tz_offset: i32) -> Timestamp { Timestamp { timestamp: MillisSinceEpoch(msec), tz_offset, } } #[test] fn test_parsed_tree() { let mut env = TestTemplateEnv::default(); env.add_keyword("divergent", |language| { language.wrap_boolean(Literal(false)) }); env.add_keyword("empty", |language| language.wrap_boolean(Literal(true))); env.add_keyword("hello", |language| { language.wrap_string(Literal("Hello".to_owned())) }); // Empty insta::assert_snapshot!(env.render_ok(r#" "#), @""); // Single term with whitespace insta::assert_snapshot!(env.render_ok(r#" hello.upper() "#), @"HELLO"); // Multiple terms insta::assert_snapshot!(env.render_ok(r#" hello.upper() ++ true "#), @"HELLOtrue"); // Parenthesized single term insta::assert_snapshot!(env.render_ok(r#"(hello.upper())"#), @"HELLO"); // Parenthesized multiple terms and concatenation insta::assert_snapshot!(env.render_ok(r#"(hello.upper() ++ " ") ++ empty"#), @"HELLO true"); // Parenthesized "if" condition insta::assert_snapshot!(env.render_ok(r#"if((divergent), "t", "f")"#), @"f"); // Parenthesized method chaining insta::assert_snapshot!(env.render_ok(r#"(hello).upper()"#), @"HELLO"); } #[test] fn test_parse_error() { let mut env = TestTemplateEnv::default(); env.add_keyword("description", |language| { language.wrap_string(Literal("".to_owned())) }); env.add_keyword("empty", |language| language.wrap_boolean(Literal(true))); insta::assert_snapshot!(env.parse_err(r#"description ()"#), @r###" --> 1:13 | 1 | description () | ^--- | = expected EOI "###); insta::assert_snapshot!(env.parse_err(r#"foo"#), @r###" --> 1:1 | 1 | foo | ^-^ | = Keyword "foo" doesn't exist "###); insta::assert_snapshot!(env.parse_err(r#"foo()"#), @r###" --> 1:1 | 1 | foo() | ^-^ | = Function "foo" doesn't exist "###); insta::assert_snapshot!(env.parse_err(r#"false()"#), @r###" --> 1:1 | 1 | false() | ^---^ | = Expected identifier "###); insta::assert_snapshot!(env.parse_err(r#"description.first_line().foo()"#), @r###" --> 1:26 | 1 | description.first_line().foo() | ^-^ | = Method "foo" doesn't exist for type "String" "###); insta::assert_snapshot!(env.parse_err(r#"10000000000000000000"#), @r###" --> 1:1 | 1 | 10000000000000000000 | ^------------------^ | = Invalid integer literal: number too large to fit in target type "###); insta::assert_snapshot!(env.parse_err(r#"42.foo()"#), @r###" --> 1:4 | 1 | 42.foo() | ^-^ | = Method "foo" doesn't exist for type "Integer" "###); insta::assert_snapshot!(env.parse_err(r#"(-empty)"#), @r###" --> 1:2 | 1 | (-empty) | ^--- | = expected template "###); insta::assert_snapshot!(env.parse_err(r#"("foo" ++ "bar").baz()"#), @r###" --> 1:18 | 1 | ("foo" ++ "bar").baz() | ^-^ | = Method "baz" doesn't exist for type "Template" "###); insta::assert_snapshot!(env.parse_err(r#"description.contains()"#), @r###" --> 1:22 | 1 | description.contains() | ^ | = Function "contains": Expected 1 arguments "###); insta::assert_snapshot!(env.parse_err(r#"description.first_line("foo")"#), @r###" --> 1:24 | 1 | description.first_line("foo") | ^---^ | = Function "first_line": Expected 0 arguments "###); insta::assert_snapshot!(env.parse_err(r#"label()"#), @r###" --> 1:7 | 1 | label() | ^ | = Function "label": Expected 2 arguments "###); insta::assert_snapshot!(env.parse_err(r#"label("foo", "bar", "baz")"#), @r###" --> 1:7 | 1 | label("foo", "bar", "baz") | ^-----------------^ | = Function "label": Expected 2 arguments "###); insta::assert_snapshot!(env.parse_err(r#"if()"#), @r###" --> 1:4 | 1 | if() | ^ | = Function "if": Expected 2 to 3 arguments "###); insta::assert_snapshot!(env.parse_err(r#"if("foo", "bar", "baz", "quux")"#), @r###" --> 1:4 | 1 | if("foo", "bar", "baz", "quux") | ^-------------------------^ | = Function "if": Expected 2 to 3 arguments "###); insta::assert_snapshot!(env.parse_err(r#"if(label("foo", "bar"), "baz")"#), @r###" --> 1:4 | 1 | if(label("foo", "bar"), "baz") | ^-----------------^ | = Expected expression of type "Boolean" "###); insta::assert_snapshot!(env.parse_err(r#"|x| description"#), @r###" --> 1:1 | 1 | |x| description | ^-------------^ | = Lambda cannot be defined here "###); } #[test] fn test_boolean_cast() { let mut env = TestTemplateEnv::default(); insta::assert_snapshot!(env.render_ok(r#"if("", true, false)"#), @"false"); insta::assert_snapshot!(env.render_ok(r#"if("a", true, false)"#), @"true"); env.add_keyword("sl0", |language| { language.wrap_string_list(Literal::>(vec![])) }); env.add_keyword("sl1", |language| { language.wrap_string_list(Literal(vec!["".to_owned()])) }); insta::assert_snapshot!(env.render_ok(r#"if(sl0, true, false)"#), @"false"); insta::assert_snapshot!(env.render_ok(r#"if(sl1, true, false)"#), @"true"); // No implicit cast of integer insta::assert_snapshot!(env.parse_err(r#"if(0, true, false)"#), @r###" --> 1:4 | 1 | if(0, true, false) | ^ | = Expected expression of type "Boolean" "###); insta::assert_snapshot!(env.parse_err(r#"if(label("", ""), true, false)"#), @r###" --> 1:4 | 1 | if(label("", ""), true, false) | ^-----------^ | = Expected expression of type "Boolean" "###); insta::assert_snapshot!(env.parse_err(r#"if(sl0.map(|x| x), true, false)"#), @r###" --> 1:4 | 1 | if(sl0.map(|x| x), true, false) | ^------------^ | = Expected expression of type "Boolean" "###); } #[test] fn test_list_method() { let mut env = TestTemplateEnv::default(); env.add_keyword("empty", |language| language.wrap_boolean(Literal(true))); env.add_keyword("sep", |language| { language.wrap_string(Literal("sep".to_owned())) }); insta::assert_snapshot!(env.render_ok(r#""".lines().join("|")"#), @""); insta::assert_snapshot!(env.render_ok(r#""a\nb\nc".lines().join("|")"#), @"a|b|c"); // Null separator insta::assert_snapshot!(env.render_ok(r#""a\nb\nc".lines().join("\0")"#), @"a\0b\0c"); // Keyword as separator insta::assert_snapshot!( env.render_ok(r#""a\nb\nc".lines().join(sep.upper())"#), @"aSEPbSEPc"); insta::assert_snapshot!( env.render_ok(r#""a\nb\nc".lines().map(|s| s ++ s)"#), @"aa bb cc"); // Global keyword in item template insta::assert_snapshot!( env.render_ok(r#""a\nb\nc".lines().map(|s| s ++ empty)"#), @"atrue btrue ctrue"); // Override global keyword 'empty' insta::assert_snapshot!( env.render_ok(r#""a\nb\nc".lines().map(|empty| empty)"#), @"a b c"); // Nested map operations insta::assert_snapshot!( env.render_ok(r#""a\nb\nc".lines().map(|s| "x\ny".lines().map(|t| s ++ t))"#), @"ax ay bx by cx cy"); // Nested map/join operations insta::assert_snapshot!( env.render_ok(r#""a\nb\nc".lines().map(|s| "x\ny".lines().map(|t| s ++ t).join(",")).join(";")"#), @"ax,ay;bx,by;cx,cy"); // Nested string operations insta::assert_snapshot!( env.render_ok(r#""!a\n!b\nc\nend".remove_suffix("end").lines().map(|s| s.remove_prefix("!"))"#), @"a b c"); // Lambda expression in alias env.add_alias("identity", "|x| x"); insta::assert_snapshot!(env.render_ok(r#""a\nb\nc".lines().map(identity)"#), @"a b c"); // Not a lambda expression insta::assert_snapshot!(env.parse_err(r#""a".lines().map(empty)"#), @r###" --> 1:17 | 1 | "a".lines().map(empty) | ^---^ | = Expected lambda expression "###); // Bad lambda parameter count insta::assert_snapshot!(env.parse_err(r#""a".lines().map(|| "")"#), @r###" --> 1:18 | 1 | "a".lines().map(|| "") | ^ | = Expected 1 lambda parameters "###); insta::assert_snapshot!(env.parse_err(r#""a".lines().map(|a, b| "")"#), @r###" --> 1:18 | 1 | "a".lines().map(|a, b| "") | ^--^ | = Expected 1 lambda parameters "###); // Error in lambda expression insta::assert_snapshot!(env.parse_err(r#""a".lines().map(|s| s.unknown())"#), @r###" --> 1:23 | 1 | "a".lines().map(|s| s.unknown()) | ^-----^ | = Method "unknown" doesn't exist for type "String" "###); // Error in lambda alias env.add_alias("too_many_params", "|x, y| x"); insta::assert_snapshot!(env.parse_err(r#""a".lines().map(too_many_params)"#), @r###" --> 1:17 | 1 | "a".lines().map(too_many_params) | ^-------------^ | = Alias "too_many_params" cannot be expanded --> 1:2 | 1 | |x, y| x | ^--^ | = Expected 1 lambda parameters "###); } #[test] fn test_string_method() { let mut env = TestTemplateEnv::default(); env.add_keyword("description", |language| { language.wrap_string(Literal("description 1".to_owned())) }); insta::assert_snapshot!(env.render_ok(r#""fooo".contains("foo")"#), @"true"); insta::assert_snapshot!(env.render_ok(r#""foo".contains("fooo")"#), @"false"); insta::assert_snapshot!(env.render_ok(r#"description.contains("description")"#), @"true"); insta::assert_snapshot!( env.render_ok(r#""description 123".contains(description.first_line())"#), @"true"); insta::assert_snapshot!(env.render_ok(r#""".first_line()"#), @""); insta::assert_snapshot!(env.render_ok(r#""foo\nbar".first_line()"#), @"foo"); insta::assert_snapshot!(env.render_ok(r#""".lines()"#), @""); insta::assert_snapshot!(env.render_ok(r#""a\nb\nc\n".lines()"#), @"a b c"); insta::assert_snapshot!(env.render_ok(r#""".starts_with("")"#), @"true"); insta::assert_snapshot!(env.render_ok(r#""everything".starts_with("")"#), @"true"); insta::assert_snapshot!(env.render_ok(r#""".starts_with("foo")"#), @"false"); insta::assert_snapshot!(env.render_ok(r#""foo".starts_with("foo")"#), @"true"); insta::assert_snapshot!(env.render_ok(r#""foobar".starts_with("foo")"#), @"true"); insta::assert_snapshot!(env.render_ok(r#""foobar".starts_with("bar")"#), @"false"); insta::assert_snapshot!(env.render_ok(r#""".ends_with("")"#), @"true"); insta::assert_snapshot!(env.render_ok(r#""everything".ends_with("")"#), @"true"); insta::assert_snapshot!(env.render_ok(r#""".ends_with("foo")"#), @"false"); insta::assert_snapshot!(env.render_ok(r#""foo".ends_with("foo")"#), @"true"); insta::assert_snapshot!(env.render_ok(r#""foobar".ends_with("foo")"#), @"false"); insta::assert_snapshot!(env.render_ok(r#""foobar".ends_with("bar")"#), @"true"); insta::assert_snapshot!(env.render_ok(r#""".remove_prefix("wip: ")"#), @""); insta::assert_snapshot!( env.render_ok(r#""wip: testing".remove_prefix("wip: ")"#), @"testing"); insta::assert_snapshot!( env.render_ok(r#""bar@my.example.com".remove_suffix("@other.example.com")"#), @"bar@my.example.com"); insta::assert_snapshot!( env.render_ok(r#""bar@other.example.com".remove_suffix("@other.example.com")"#), @"bar"); insta::assert_snapshot!(env.render_ok(r#""foo".substr(0, 0)"#), @""); insta::assert_snapshot!(env.render_ok(r#""foo".substr(0, 1)"#), @"f"); insta::assert_snapshot!(env.render_ok(r#""foo".substr(0, 99)"#), @"foo"); insta::assert_snapshot!(env.render_ok(r#""abcdef".substr(2, -1)"#), @"cde"); insta::assert_snapshot!(env.render_ok(r#""abcdef".substr(-3, 99)"#), @"def"); insta::assert_snapshot!(env.render_ok(r#""abc💩".substr(2, -1)"#), @"c"); insta::assert_snapshot!(env.render_ok(r#""abc💩".substr(-1, 99)"#), @"💩"); // ranges with end > start are empty insta::assert_snapshot!(env.render_ok(r#""abcdef".substr(4, 2)"#), @""); insta::assert_snapshot!(env.render_ok(r#""abcdef".substr(-2, -4)"#), @""); } #[test] fn test_signature() { let mut env = TestTemplateEnv::default(); env.add_keyword("author", |language| { language.wrap_signature(Literal(new_signature("Test User", "test.user@example.com"))) }); insta::assert_snapshot!(env.render_ok(r#"author"#), @"Test User "); insta::assert_snapshot!(env.render_ok(r#"author.name()"#), @"Test User"); insta::assert_snapshot!(env.render_ok(r#"author.email()"#), @"test.user@example.com"); insta::assert_snapshot!(env.render_ok(r#"author.username()"#), @"test.user"); env.add_keyword("author", |language| { language.wrap_signature(Literal(new_signature( "Another Test User", "test.user@example.com", ))) }); insta::assert_snapshot!(env.render_ok(r#"author"#), @"Another Test User "); insta::assert_snapshot!(env.render_ok(r#"author.name()"#), @"Another Test User"); insta::assert_snapshot!(env.render_ok(r#"author.email()"#), @"test.user@example.com"); insta::assert_snapshot!(env.render_ok(r#"author.username()"#), @"test.user"); env.add_keyword("author", |language| { language.wrap_signature(Literal(new_signature( "Test User", "test.user@invalid@example.com", ))) }); insta::assert_snapshot!(env.render_ok(r#"author"#), @"Test User "); insta::assert_snapshot!(env.render_ok(r#"author.name()"#), @"Test User"); insta::assert_snapshot!(env.render_ok(r#"author.email()"#), @"test.user@invalid@example.com"); insta::assert_snapshot!(env.render_ok(r#"author.username()"#), @"test.user"); env.add_keyword("author", |language| { language.wrap_signature(Literal(new_signature("Test User", "test.user"))) }); insta::assert_snapshot!(env.render_ok(r#"author"#), @"Test User "); insta::assert_snapshot!(env.render_ok(r#"author.email()"#), @"test.user"); insta::assert_snapshot!(env.render_ok(r#"author.username()"#), @"test.user"); env.add_keyword("author", |language| { language.wrap_signature(Literal(new_signature( "Test User", "test.user+tag@example.com", ))) }); insta::assert_snapshot!(env.render_ok(r#"author"#), @"Test User "); insta::assert_snapshot!(env.render_ok(r#"author.email()"#), @"test.user+tag@example.com"); insta::assert_snapshot!(env.render_ok(r#"author.username()"#), @"test.user+tag"); env.add_keyword("author", |language| { language.wrap_signature(Literal(new_signature("Test User", "x@y"))) }); insta::assert_snapshot!(env.render_ok(r#"author"#), @"Test User "); insta::assert_snapshot!(env.render_ok(r#"author.email()"#), @"x@y"); insta::assert_snapshot!(env.render_ok(r#"author.username()"#), @"x"); env.add_keyword("author", |language| { language.wrap_signature(Literal(new_signature("", "test.user@example.com"))) }); insta::assert_snapshot!(env.render_ok(r#"author"#), @""); insta::assert_snapshot!(env.render_ok(r#"author.name()"#), @""); insta::assert_snapshot!(env.render_ok(r#"author.email()"#), @"test.user@example.com"); insta::assert_snapshot!(env.render_ok(r#"author.username()"#), @"test.user"); env.add_keyword("author", |language| { language.wrap_signature(Literal(new_signature("Test User", ""))) }); insta::assert_snapshot!(env.render_ok(r#"author"#), @"Test User"); insta::assert_snapshot!(env.render_ok(r#"author.name()"#), @"Test User"); insta::assert_snapshot!(env.render_ok(r#"author.email()"#), @""); insta::assert_snapshot!(env.render_ok(r#"author.username()"#), @""); env.add_keyword("author", |language| { language.wrap_signature(Literal(new_signature("", ""))) }); insta::assert_snapshot!(env.render_ok(r#"author"#), @""); insta::assert_snapshot!(env.render_ok(r#"author.name()"#), @""); insta::assert_snapshot!(env.render_ok(r#"author.email()"#), @""); insta::assert_snapshot!(env.render_ok(r#"author.username()"#), @""); } #[test] fn test_timestamp_method() { let mut env = TestTemplateEnv::default(); env.add_keyword("t0", |language| { language.wrap_timestamp(Literal(new_timestamp(0, 0))) }); insta::assert_snapshot!( env.render_ok(r#"t0.format("%Y%m%d %H:%M:%S")"#), @"19700101 00:00:00"); // Invalid format string insta::assert_snapshot!(env.parse_err(r#"t0.format("%_")"#), @r###" --> 1:11 | 1 | t0.format("%_") | ^--^ | = Invalid time format "###); // Invalid type insta::assert_snapshot!(env.parse_err(r#"t0.format(0)"#), @r###" --> 1:11 | 1 | t0.format(0) | ^ | = Expected string literal "###); // Dynamic string isn't supported yet insta::assert_snapshot!(env.parse_err(r#"t0.format("%Y" ++ "%m")"#), @r###" --> 1:11 | 1 | t0.format("%Y" ++ "%m") | ^----------^ | = Expected string literal "###); // Literal alias expansion env.add_alias("time_format", r#""%Y-%m-%d""#); env.add_alias("bad_time_format", r#""%_""#); insta::assert_snapshot!(env.render_ok(r#"t0.format(time_format)"#), @"1970-01-01"); insta::assert_snapshot!(env.parse_err(r#"t0.format(bad_time_format)"#), @r###" --> 1:11 | 1 | t0.format(bad_time_format) | ^-------------^ | = Alias "bad_time_format" cannot be expanded --> 1:1 | 1 | "%_" | ^--^ | = Invalid time format "###); } #[test] fn test_fill_function() { let mut env = TestTemplateEnv::default(); env.add_color("error", crossterm::style::Color::DarkRed); insta::assert_snapshot!( env.render_ok(r#"fill(20, "The quick fox jumps over the " ++ label("error", "lazy") ++ " dog\n")"#), @r###" The quick fox jumps over the lazy dog "###); // A low value will not chop words, but can chop a label by words insta::assert_snapshot!( env.render_ok(r#"fill(9, "Longlonglongword an some short words " ++ label("error", "longlonglongword and short words") ++ " back out\n")"#), @r###" Longlonglongword an some short words longlonglongword and short words back out "###); // Filling to 0 means breaking at every word insta::assert_snapshot!( env.render_ok(r#"fill(0, "The quick fox jumps over the " ++ label("error", "lazy") ++ " dog\n")"#), @r###" The quick fox jumps over the lazy dog "###); // Filling to -0 is the same as 0 insta::assert_snapshot!( env.render_ok(r#"fill(-0, "The quick fox jumps over the " ++ label("error", "lazy") ++ " dog\n")"#), @r###" The quick fox jumps over the lazy dog "###); // Filling to negatives are clamped to the same as zero insta::assert_snapshot!( env.render_ok(r#"fill(-10, "The quick fox jumps over the " ++ label("error", "lazy") ++ " dog\n")"#), @r###" The quick fox jumps over the lazy dog "###); // Word-wrap, then indent insta::assert_snapshot!( env.render_ok(r#""START marker to help insta\n" ++ indent(" ", fill(20, "The quick fox jumps over the " ++ label("error", "lazy") ++ " dog\n"))"#), @r###" START marker to help insta The quick fox jumps over the lazy dog "###); // Word-wrap indented (no special handling for leading spaces) insta::assert_snapshot!( env.render_ok(r#""START marker to help insta\n" ++ fill(20, indent(" ", "The quick fox jumps over the " ++ label("error", "lazy") ++ " dog\n"))"#), @r###" START marker to help insta The quick fox jumps over the lazy dog "###); } #[test] fn test_indent_function() { let mut env = TestTemplateEnv::default(); env.add_color("error", crossterm::style::Color::DarkRed); env.add_color("warning", crossterm::style::Color::DarkYellow); env.add_color("hint", crossterm::style::Color::DarkCyan); // Empty line shouldn't be indented. Not using insta here because we test // whitespace existence. assert_eq!(env.render_ok(r#"indent("__", "")"#), ""); assert_eq!(env.render_ok(r#"indent("__", "\n")"#), "\n"); assert_eq!(env.render_ok(r#"indent("__", "a\n\nb")"#), "__a\n\n__b"); // "\n" at end of labeled text insta::assert_snapshot!( env.render_ok(r#"indent("__", label("error", "a\n") ++ label("warning", "b\n"))"#), @r###" __a __b "###); // "\n" in labeled text insta::assert_snapshot!( env.render_ok(r#"indent("__", label("error", "a") ++ label("warning", "b\nc"))"#), @r###" __ab __c "###); // Labeled prefix + unlabeled content insta::assert_snapshot!( env.render_ok(r#"indent(label("error", "XX"), "a\nb\n")"#), @r###" XXa XXb "###); // Nested indent, silly but works insta::assert_snapshot!( env.render_ok(r#"indent(label("hint", "A"), label("warning", indent(label("hint", "B"), label("error", "x\n") ++ "y")))"#), @r###" ABx ABy "###); } #[test] fn test_label_function() { let mut env = TestTemplateEnv::default(); env.add_keyword("empty", |language| language.wrap_boolean(Literal(true))); env.add_color("error", crossterm::style::Color::DarkRed); env.add_color("warning", crossterm::style::Color::DarkYellow); // Literal insta::assert_snapshot!( env.render_ok(r#"label("error", "text")"#), @"text"); // Evaluated property insta::assert_snapshot!( env.render_ok(r#"label("error".first_line(), "text")"#), @"text"); // Template insta::assert_snapshot!( env.render_ok(r#"label(if(empty, "error", "warning"), "text")"#), @"text"); } #[test] fn test_concat_function() { let mut env = TestTemplateEnv::default(); env.add_keyword("empty", |language| language.wrap_boolean(Literal(true))); env.add_keyword("hidden", |language| language.wrap_boolean(Literal(false))); env.add_color("empty", crossterm::style::Color::DarkGreen); env.add_color("error", crossterm::style::Color::DarkRed); env.add_color("warning", crossterm::style::Color::DarkYellow); insta::assert_snapshot!(env.render_ok(r#"concat()"#), @""); insta::assert_snapshot!( env.render_ok(r#"concat(hidden, empty)"#), @"falsetrue"); insta::assert_snapshot!( env.render_ok(r#"concat(label("error", ""), label("warning", "a"), "b")"#), @"ab"); } #[test] fn test_separate_function() { let mut env = TestTemplateEnv::default(); env.add_keyword("description", |language| { language.wrap_string(Literal("".to_owned())) }); env.add_keyword("empty", |language| language.wrap_boolean(Literal(true))); env.add_keyword("hidden", |language| language.wrap_boolean(Literal(false))); env.add_color("empty", crossterm::style::Color::DarkGreen); env.add_color("error", crossterm::style::Color::DarkRed); env.add_color("warning", crossterm::style::Color::DarkYellow); insta::assert_snapshot!(env.render_ok(r#"separate(" ")"#), @""); insta::assert_snapshot!(env.render_ok(r#"separate(" ", "")"#), @""); insta::assert_snapshot!(env.render_ok(r#"separate(" ", "a")"#), @"a"); insta::assert_snapshot!(env.render_ok(r#"separate(" ", "a", "b")"#), @"a b"); insta::assert_snapshot!(env.render_ok(r#"separate(" ", "a", "", "b")"#), @"a b"); insta::assert_snapshot!(env.render_ok(r#"separate(" ", "a", "b", "")"#), @"a b"); insta::assert_snapshot!(env.render_ok(r#"separate(" ", "", "a", "b")"#), @"a b"); // Labeled insta::assert_snapshot!( env.render_ok(r#"separate(" ", label("error", ""), label("warning", "a"), "b")"#), @"a b"); // List template insta::assert_snapshot!(env.render_ok(r#"separate(" ", "a", ("" ++ ""))"#), @"a"); insta::assert_snapshot!(env.render_ok(r#"separate(" ", "a", ("" ++ "b"))"#), @"a b"); // Nested separate insta::assert_snapshot!( env.render_ok(r#"separate(" ", "a", separate("|", "", ""))"#), @"a"); insta::assert_snapshot!( env.render_ok(r#"separate(" ", "a", separate("|", "b", ""))"#), @"a b"); insta::assert_snapshot!( env.render_ok(r#"separate(" ", "a", separate("|", "b", "c"))"#), @"a b|c"); // Conditional template insta::assert_snapshot!( env.render_ok(r#"separate(" ", "a", if(true, ""))"#), @"a"); insta::assert_snapshot!( env.render_ok(r#"separate(" ", "a", if(true, "", "f"))"#), @"a"); insta::assert_snapshot!( env.render_ok(r#"separate(" ", "a", if(false, "t", ""))"#), @"a"); insta::assert_snapshot!( env.render_ok(r#"separate(" ", "a", if(true, "t", "f"))"#), @"a t"); // Separate keywords insta::assert_snapshot!( env.render_ok(r#"separate(" ", hidden, description, empty)"#), @"false true"); // Keyword as separator insta::assert_snapshot!( env.render_ok(r#"separate(hidden, "X", "Y", "Z")"#), @"XfalseYfalseZ"); } }