mirror of
https://github.com/martinvonz/jj.git
synced 2025-01-15 16:53:25 +00:00
ce5e95fa80
For what's currently called `Store` in the code, I have been using "backend" in plain text. That probably means that `Backend` is a good name for it.
410 lines
12 KiB
Rust
410 lines
12 KiB
Rust
// Copyright 2020 Google LLC
|
|
//
|
|
// 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::borrow::BorrowMut;
|
|
use std::io;
|
|
use std::ops::Add;
|
|
|
|
use itertools::Itertools;
|
|
use jujutsu_lib::backend::{CommitId, Signature};
|
|
use jujutsu_lib::commit::Commit;
|
|
use jujutsu_lib::repo::RepoRef;
|
|
|
|
use crate::formatter::Formatter;
|
|
|
|
pub trait Template<C> {
|
|
fn format(&self, context: &C, formatter: &mut dyn Formatter) -> io::Result<()>;
|
|
}
|
|
|
|
// TODO: Extract a trait for this type?
|
|
pub struct TemplateFormatter<'f, 't: 'f, C> {
|
|
template: Box<dyn Template<C> + 't>,
|
|
formatter: &'f mut dyn Formatter,
|
|
}
|
|
|
|
impl<'f, 't: 'f, C> TemplateFormatter<'f, 't, C> {
|
|
pub fn new(template: Box<dyn Template<C> + 't>, formatter: &'f mut dyn Formatter) -> Self {
|
|
TemplateFormatter {
|
|
template,
|
|
formatter,
|
|
}
|
|
}
|
|
|
|
pub fn format<'c, 'a: 'c>(&'a mut self, context: &'c C) -> io::Result<()> {
|
|
self.template.format(context, self.formatter.borrow_mut())
|
|
}
|
|
}
|
|
|
|
pub struct LiteralTemplate(pub String);
|
|
|
|
impl<C> Template<C> for LiteralTemplate {
|
|
fn format(&self, _context: &C, formatter: &mut dyn Formatter) -> io::Result<()> {
|
|
formatter.write_str(&self.0)
|
|
}
|
|
}
|
|
|
|
// TODO: figure out why this lifetime is needed
|
|
pub struct LabelTemplate<'a, C> {
|
|
content: Box<dyn Template<C> + 'a>,
|
|
labels: Vec<String>,
|
|
}
|
|
|
|
impl<'a, C> LabelTemplate<'a, C> {
|
|
pub fn new(content: Box<dyn Template<C> + 'a>, labels: String) -> Self {
|
|
let labels = labels
|
|
.split_whitespace()
|
|
.map(|label| label.to_string())
|
|
.collect_vec();
|
|
LabelTemplate { content, labels }
|
|
}
|
|
}
|
|
|
|
impl<'a, C> Template<C> for LabelTemplate<'a, C> {
|
|
fn format(&self, context: &C, formatter: &mut dyn Formatter) -> io::Result<()> {
|
|
for label in &self.labels {
|
|
formatter.add_label(label.clone())?;
|
|
}
|
|
self.content.format(context, formatter)?;
|
|
for _label in &self.labels {
|
|
formatter.remove_label()?;
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
// TODO: figure out why this lifetime is needed
|
|
pub struct DynamicLabelTemplate<'a, C> {
|
|
content: Box<dyn Template<C> + 'a>,
|
|
label_property: Box<dyn Fn(&C) -> String + 'a>,
|
|
}
|
|
|
|
impl<'a, C> DynamicLabelTemplate<'a, C> {
|
|
pub fn new(
|
|
content: Box<dyn Template<C> + 'a>,
|
|
label_property: Box<dyn Fn(&C) -> String + 'a>,
|
|
) -> Self {
|
|
DynamicLabelTemplate {
|
|
content,
|
|
label_property,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'a, C> Template<C> for DynamicLabelTemplate<'a, C> {
|
|
fn format(&self, context: &C, formatter: &mut dyn Formatter) -> io::Result<()> {
|
|
let labels = self.label_property.as_ref()(context);
|
|
let labels = labels
|
|
.split_whitespace()
|
|
.map(|label| label.to_string())
|
|
.collect_vec();
|
|
for label in &labels {
|
|
formatter.add_label(label.clone())?;
|
|
}
|
|
self.content.format(context, formatter)?;
|
|
for _label in &labels {
|
|
formatter.remove_label()?;
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
// TODO: figure out why this lifetime is needed
|
|
pub struct ListTemplate<'a, C>(pub Vec<Box<dyn Template<C> + 'a>>);
|
|
|
|
impl<'a, C> Template<C> for ListTemplate<'a, C> {
|
|
fn format(&self, context: &C, formatter: &mut dyn Formatter) -> io::Result<()> {
|
|
for template in &self.0 {
|
|
template.format(context, formatter)?
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
pub trait TemplateProperty<C, O> {
|
|
fn extract(&self, context: &C) -> O;
|
|
}
|
|
|
|
pub struct ConstantTemplateProperty<O> {
|
|
pub output: O,
|
|
}
|
|
|
|
impl<C, O: Clone> TemplateProperty<C, O> for ConstantTemplateProperty<O> {
|
|
fn extract(&self, _context: &C) -> O {
|
|
self.output.clone()
|
|
}
|
|
}
|
|
|
|
// TODO: figure out why this lifetime is needed
|
|
pub struct StringPropertyTemplate<'a, C> {
|
|
pub property: Box<dyn TemplateProperty<C, String> + 'a>,
|
|
}
|
|
|
|
impl<'a, C> Template<C> for StringPropertyTemplate<'a, C> {
|
|
fn format(&self, context: &C, formatter: &mut dyn Formatter) -> io::Result<()> {
|
|
let text = self.property.extract(context);
|
|
formatter.write_str(&text)
|
|
}
|
|
}
|
|
|
|
pub struct ChangeIdProperty;
|
|
|
|
impl<'r> TemplateProperty<Commit, String> for ChangeIdProperty {
|
|
fn extract(&self, context: &Commit) -> String {
|
|
context.change_id().hex()
|
|
}
|
|
}
|
|
|
|
pub struct DescriptionProperty;
|
|
|
|
impl<'r> TemplateProperty<Commit, String> for DescriptionProperty {
|
|
fn extract(&self, context: &Commit) -> String {
|
|
let description = context.description().to_owned();
|
|
if description.ends_with('\n') {
|
|
description
|
|
} else {
|
|
description.add("\n")
|
|
}
|
|
}
|
|
}
|
|
|
|
pub struct AuthorProperty;
|
|
|
|
impl<'r> TemplateProperty<Commit, Signature> for AuthorProperty {
|
|
fn extract(&self, context: &Commit) -> Signature {
|
|
context.author().clone()
|
|
}
|
|
}
|
|
|
|
pub struct CommitterProperty;
|
|
|
|
impl<'r> TemplateProperty<Commit, Signature> for CommitterProperty {
|
|
fn extract(&self, context: &Commit) -> Signature {
|
|
context.committer().clone()
|
|
}
|
|
}
|
|
|
|
pub struct OpenProperty;
|
|
|
|
impl<'r> TemplateProperty<Commit, bool> for OpenProperty {
|
|
fn extract(&self, context: &Commit) -> bool {
|
|
context.is_open()
|
|
}
|
|
}
|
|
|
|
pub struct PrunedProperty;
|
|
|
|
impl TemplateProperty<Commit, bool> for PrunedProperty {
|
|
fn extract(&self, context: &Commit) -> bool {
|
|
context.is_pruned()
|
|
}
|
|
}
|
|
|
|
pub struct CurrentCheckoutProperty<'a> {
|
|
pub repo: RepoRef<'a>,
|
|
}
|
|
|
|
impl TemplateProperty<Commit, bool> for CurrentCheckoutProperty<'_> {
|
|
fn extract(&self, context: &Commit) -> bool {
|
|
context.id() == self.repo.view().checkout()
|
|
}
|
|
}
|
|
|
|
pub struct BranchProperty<'a> {
|
|
pub repo: RepoRef<'a>,
|
|
}
|
|
|
|
impl TemplateProperty<Commit, String> for BranchProperty<'_> {
|
|
fn extract(&self, context: &Commit) -> String {
|
|
let mut names = vec![];
|
|
for (branch_name, branch_target) in self.repo.view().branches() {
|
|
let local_target = branch_target.local_target.as_ref();
|
|
if let Some(local_target) = local_target {
|
|
if local_target.has_add(context.id()) {
|
|
if local_target.is_conflict() {
|
|
names.push(format!("{}?", branch_name));
|
|
} else {
|
|
names.push(branch_name.clone());
|
|
}
|
|
}
|
|
}
|
|
for (remote_name, remote_target) in &branch_target.remote_targets {
|
|
if Some(remote_target) != local_target && remote_target.has_add(context.id()) {
|
|
if remote_target.is_conflict() {
|
|
names.push(format!("{}@{}?", branch_name, remote_name));
|
|
} else {
|
|
names.push(format!("{}@{}", branch_name, remote_name));
|
|
}
|
|
}
|
|
}
|
|
}
|
|
names.join(" ")
|
|
}
|
|
}
|
|
|
|
pub struct TagProperty<'a> {
|
|
pub repo: RepoRef<'a>,
|
|
}
|
|
|
|
impl TemplateProperty<Commit, String> for TagProperty<'_> {
|
|
fn extract(&self, context: &Commit) -> String {
|
|
let mut names = vec![];
|
|
for (tag_name, target) in self.repo.view().tags() {
|
|
if target.has_add(context.id()) {
|
|
if target.is_conflict() {
|
|
names.push(format!("{}?", tag_name));
|
|
} else {
|
|
names.push(tag_name.clone());
|
|
}
|
|
}
|
|
}
|
|
names.join(" ")
|
|
}
|
|
}
|
|
|
|
pub struct GitRefsProperty<'a> {
|
|
pub repo: RepoRef<'a>,
|
|
}
|
|
|
|
impl TemplateProperty<Commit, String> for GitRefsProperty<'_> {
|
|
fn extract(&self, context: &Commit) -> String {
|
|
// TODO: We should keep a map from commit to ref names so we don't have to walk
|
|
// all refs here.
|
|
let mut names = vec![];
|
|
for (name, target) in self.repo.view().git_refs() {
|
|
if target.has_add(context.id()) {
|
|
if target.is_conflict() {
|
|
names.push(format!("{}?", name));
|
|
} else {
|
|
names.push(name.clone());
|
|
}
|
|
}
|
|
}
|
|
names.join(" ")
|
|
}
|
|
}
|
|
|
|
pub struct ObsoleteProperty<'a> {
|
|
pub repo: RepoRef<'a>,
|
|
}
|
|
|
|
impl TemplateProperty<Commit, bool> for ObsoleteProperty<'_> {
|
|
fn extract(&self, context: &Commit) -> bool {
|
|
self.repo.evolution().is_obsolete(context.id())
|
|
}
|
|
}
|
|
|
|
pub struct OrphanProperty<'a> {
|
|
pub repo: RepoRef<'a>,
|
|
}
|
|
|
|
impl TemplateProperty<Commit, bool> for OrphanProperty<'_> {
|
|
fn extract(&self, context: &Commit) -> bool {
|
|
self.repo.evolution().is_orphan(context.id())
|
|
}
|
|
}
|
|
|
|
pub struct DivergentProperty<'a> {
|
|
pub repo: RepoRef<'a>,
|
|
}
|
|
|
|
impl TemplateProperty<Commit, bool> for DivergentProperty<'_> {
|
|
fn extract(&self, context: &Commit) -> bool {
|
|
self.repo.evolution().is_divergent(context.change_id())
|
|
}
|
|
}
|
|
|
|
pub struct ConflictProperty;
|
|
|
|
impl TemplateProperty<Commit, bool> for ConflictProperty {
|
|
fn extract(&self, context: &Commit) -> bool {
|
|
context.tree().has_conflict()
|
|
}
|
|
}
|
|
|
|
pub struct ConditionalTemplate<'a, C> {
|
|
pub condition: Box<dyn TemplateProperty<C, bool> + 'a>,
|
|
pub true_template: Box<dyn Template<C> + 'a>,
|
|
pub false_template: Option<Box<dyn Template<C> + 'a>>,
|
|
}
|
|
|
|
// TODO: figure out why this lifetime is needed
|
|
impl<'a, C> ConditionalTemplate<'a, C> {
|
|
pub fn new(
|
|
condition: Box<dyn TemplateProperty<C, bool> + 'a>,
|
|
true_template: Box<dyn Template<C> + 'a>,
|
|
false_template: Option<Box<dyn Template<C> + 'a>>,
|
|
) -> Self {
|
|
ConditionalTemplate {
|
|
condition,
|
|
true_template,
|
|
false_template,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'a, C> Template<C> for ConditionalTemplate<'a, C> {
|
|
fn format(&self, context: &C, formatter: &mut dyn Formatter) -> io::Result<()> {
|
|
if self.condition.extract(context) {
|
|
self.true_template.format(context, formatter)?;
|
|
} else if let Some(false_template) = &self.false_template {
|
|
false_template.format(context, formatter)?;
|
|
}
|
|
Ok(())
|
|
}
|
|
}
|
|
|
|
// TODO: If needed, add a ContextualTemplateFunction where the function also
|
|
// gets the context
|
|
pub struct TemplateFunction<'a, C, I, O> {
|
|
pub property: Box<dyn TemplateProperty<C, I> + 'a>,
|
|
pub function: Box<dyn Fn(I) -> O + 'a>,
|
|
}
|
|
|
|
// TODO: figure out why this lifetime is needed
|
|
impl<'a, C, I, O> TemplateFunction<'a, C, I, O> {
|
|
pub fn new(
|
|
template: Box<dyn TemplateProperty<C, I> + 'a>,
|
|
function: Box<dyn Fn(I) -> O + 'a>,
|
|
) -> Self {
|
|
TemplateFunction {
|
|
property: template,
|
|
function,
|
|
}
|
|
}
|
|
}
|
|
|
|
impl<'a, C, I, O> TemplateProperty<C, O> for TemplateFunction<'a, C, I, O> {
|
|
fn extract(&self, context: &C) -> O {
|
|
(self.function)(self.property.extract(context))
|
|
}
|
|
}
|
|
|
|
pub struct CommitIdKeyword;
|
|
|
|
impl CommitIdKeyword {
|
|
pub fn default_format(commit_id: CommitId) -> String {
|
|
commit_id.hex()
|
|
}
|
|
|
|
pub fn shortest_format(commit_id: CommitId) -> String {
|
|
// TODO: make this actually be the shortest unambiguous prefix
|
|
commit_id.hex()[..12].to_string()
|
|
}
|
|
}
|
|
|
|
impl<'r> TemplateProperty<Commit, CommitId> for CommitIdKeyword {
|
|
fn extract(&self, context: &Commit) -> CommitId {
|
|
context.id().clone()
|
|
}
|
|
}
|