// Copyright 2020 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::cmp::Ordering; use std::fmt::{Debug, Error, Formatter}; use std::hash::{Hash, Hasher}; use std::sync::Arc; use crate::backend; use crate::backend::{ChangeId, CommitId, Signature, TreeId}; use crate::repo_path::RepoPath; use crate::store::Store; use crate::tree::Tree; #[derive(Clone)] pub struct Commit { store: Arc, id: CommitId, data: Arc, } impl Debug for Commit { fn fmt(&self, f: &mut Formatter<'_>) -> Result<(), Error> { f.debug_struct("Commit").field("id", &self.id).finish() } } impl PartialEq for Commit { fn eq(&self, other: &Self) -> bool { self.id == other.id } } impl Eq for Commit {} impl Ord for Commit { fn cmp(&self, other: &Self) -> Ordering { self.id.cmp(&other.id) } } impl PartialOrd for Commit { fn partial_cmp(&self, other: &Self) -> Option { Some(self.id.cmp(&other.id)) } } impl Hash for Commit { fn hash(&self, state: &mut H) { self.id.hash(state) } } impl Commit { pub fn new(store: Arc, id: CommitId, data: Arc) -> Self { Commit { store, id, data } } pub fn store(&self) -> &Arc { &self.store } pub fn id(&self) -> &CommitId { &self.id } pub fn parent_ids(&self) -> &[CommitId] { &self.data.parents } pub fn parents(&self) -> Vec { self.data .parents .iter() .map(|id| self.store.get_commit(id).unwrap()) .collect() } pub fn predecessor_ids(&self) -> &[CommitId] { &self.data.predecessors } pub fn predecessors(&self) -> Vec { self.data .predecessors .iter() .map(|id| self.store.get_commit(id).unwrap()) .collect() } pub fn tree(&self) -> Tree { self.store .get_tree(&RepoPath::root(), &self.data.root_tree) .unwrap() } pub fn tree_id(&self) -> &TreeId { &self.data.root_tree } pub fn change_id(&self) -> &ChangeId { &self.data.change_id } pub fn store_commit(&self) -> &backend::Commit { &self.data } pub fn description(&self) -> &str { &self.data.description } pub fn author(&self) -> &Signature { &self.data.author } pub fn committer(&self) -> &Signature { &self.data.committer } /// A commit is discardable if it has one parent, no change from its /// parent, and an empty description. pub fn is_discardable(&self) -> bool { if self.description().is_empty() { if let [parent_commit] = &*self.parents() { return self.tree_id() == parent_commit.tree_id(); } } false } }