// Copyright 2024 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 jj_cli::cli_util::CliRunner; use jj_cli::commit_templater::{ CommitTemplateBuildFnTable, CommitTemplateLanguage, CommitTemplateLanguageExtension, }; use jj_cli::template_builder::TemplateLanguage; use jj_cli::template_parser::{self, TemplateParseError}; use jj_cli::templater::{TemplateFunction, TemplatePropertyError}; use jj_lib::backend::CommitId; use jj_lib::commit::Commit; use jj_lib::extensions_map::ExtensionsMap; use jj_lib::object_id::ObjectId; use jj_lib::repo::Repo; use jj_lib::revset::RevsetExpression; use once_cell::sync::OnceCell; struct HexCounter; fn num_digits_in_id(id: &CommitId) -> i64 { let mut count = 0; for ch in id.hex().chars() { if ch.is_ascii_digit() { count += 1; } } count } fn num_char_in_id(commit: Commit, ch_match: char) -> Result { let mut count = 0; for ch in commit.id().hex().chars() { if ch == ch_match { count += 1; } } Ok(count) } struct MostDigitsInId { count: OnceCell, } impl MostDigitsInId { fn new() -> Self { Self { count: OnceCell::new(), } } fn count(&self, repo: &dyn Repo) -> i64 { *self.count.get_or_init(|| { RevsetExpression::all() .evaluate_programmatic(repo) .unwrap() .iter() .map(|id| num_digits_in_id(&id)) .max() .unwrap_or(0) }) } } impl CommitTemplateLanguageExtension for HexCounter { fn build_fn_table<'repo>(&self) -> CommitTemplateBuildFnTable<'repo> { type L<'repo> = CommitTemplateLanguage<'repo>; let mut table = CommitTemplateBuildFnTable::empty(); table.commit_methods.insert( "has_most_digits", |language, _build_context, property, call| { template_parser::expect_no_arguments(call)?; let most_digits = language .cache_extension::() .unwrap() .count(language.repo()); Ok(L::wrap_boolean(TemplateFunction::new( property, move |commit| Ok(num_digits_in_id(commit.id()) == most_digits), ))) }, ); table.commit_methods.insert( "num_digits_in_id", |_language, _build_context, property, call| { template_parser::expect_no_arguments(call)?; Ok(L::wrap_integer(TemplateFunction::new(property, |commit| { Ok(num_digits_in_id(commit.id())) }))) }, ); table.commit_methods.insert( "num_char_in_id", |_language, _build_context, property, call| { let [string_arg] = template_parser::expect_exact_arguments(call)?; let char_arg = template_parser::expect_string_literal_with(string_arg, |string, span| { let chars: Vec<_> = string.chars().collect(); match chars[..] { [ch] => Ok(ch), _ => Err(TemplateParseError::unexpected_expression( "Expected singular character argument", span, )), } })?; Ok(L::wrap_integer(TemplateFunction::new( property, move |commit| num_char_in_id(commit, char_arg), ))) }, ); table } fn build_cache_extensions(&self, extensions: &mut ExtensionsMap) { extensions.insert(MostDigitsInId::new()); } } fn main() -> std::process::ExitCode { CliRunner::init() .set_commit_template_extension(Box::new(HexCounter)) .run() }