#![allow(missing_docs)]
use std::cmp::min;
use std::collections::{BTreeMap, HashSet};
use crate::backend::CommitId;
use crate::default_index_store::{IndexEntry, IndexPosition};
use crate::nightly_shims::BTreeMapExt;
use crate::revset::{RevsetGraphEdge, RevsetGraphEdgeType};
pub struct RevsetGraphIterator<'revset, 'index> {
input_set_iter: Box<dyn Iterator<Item = IndexEntry<'index>> + 'revset>,
look_ahead: BTreeMap<IndexPosition, IndexEntry<'index>>,
min_position: IndexPosition,
edges: BTreeMap<IndexPosition, HashSet<(IndexPosition, RevsetGraphEdge)>>,
skip_transitive_edges: bool,
}
impl<'revset, 'index> RevsetGraphIterator<'revset, 'index> {
pub fn new(
input_set_iter: Box<dyn Iterator<Item = IndexEntry<'index>> + 'revset>,
) -> RevsetGraphIterator<'revset, 'index> {
RevsetGraphIterator {
input_set_iter,
look_ahead: Default::default(),
min_position: IndexPosition::MAX,
edges: Default::default(),
skip_transitive_edges: true,
}
}
pub fn set_skip_transitive_edges(mut self, skip_transitive_edges: bool) -> Self {
self.skip_transitive_edges = skip_transitive_edges;
self
}
fn next_index_entry(&mut self) -> Option<IndexEntry<'index>> {
if let Some(index_entry) = self.look_ahead.pop_last_value() {
return Some(index_entry);
}
self.input_set_iter.next()
}
fn edges_from_internal_commit(
&mut self,
index_entry: &IndexEntry<'index>,
) -> HashSet<(IndexPosition, RevsetGraphEdge)> {
if let Some(edges) = self.edges.get(&index_entry.position()) {
return edges.clone();
}
let mut edges = HashSet::new();
for parent in index_entry.parents() {
let parent_position = parent.position();
let parent_commit_id = parent.commit_id();
self.consume_to(parent_position);
if self.look_ahead.contains_key(&parent_position) {
edges.insert((parent_position, RevsetGraphEdge::direct(parent_commit_id)));
} else {
let parent_edges = self.edges_from_external_commit(parent);
if parent_edges
.iter()
.all(|(_, edge)| edge.edge_type == RevsetGraphEdgeType::Missing)
{
edges.insert((parent_position, RevsetGraphEdge::missing(parent_commit_id)));
} else {
edges.extend(parent_edges);
}
}
}
self.edges.insert(index_entry.position(), edges.clone());
edges
}
fn edges_from_external_commit(
&mut self,
index_entry: IndexEntry<'index>,
) -> HashSet<(IndexPosition, RevsetGraphEdge)> {
let position = index_entry.position();
let mut stack = vec![index_entry];
while let Some(entry) = stack.last() {
let position = entry.position();
let mut edges = HashSet::new();
let mut parents_complete = true;
for parent in entry.parents() {
let parent_position = parent.position();
self.consume_to(parent_position);
if self.look_ahead.contains_key(&parent_position) {
edges.insert((
parent_position,
RevsetGraphEdge::indirect(parent.commit_id()),
));
} else if let Some(parent_edges) = self.edges.get(&parent_position) {
if parent_edges
.iter()
.all(|(_, edge)| edge.edge_type == RevsetGraphEdgeType::Missing)
{
edges.insert((
parent_position,
RevsetGraphEdge::missing(parent.commit_id()),
));
} else {
edges.extend(parent_edges.iter().cloned());
}
} else if parent_position < self.min_position {
edges.insert((
parent_position,
RevsetGraphEdge::missing(parent.commit_id()),
));
} else {
stack.push(parent);
parents_complete = false;
}
}
if parents_complete {
stack.pop().unwrap();
self.edges.insert(position, edges);
}
}
self.edges.get(&position).unwrap().clone()
}
fn remove_transitive_edges(
&mut self,
edges: HashSet<(IndexPosition, RevsetGraphEdge)>,
) -> HashSet<(IndexPosition, RevsetGraphEdge)> {
if !edges
.iter()
.any(|(_, edge)| edge.edge_type == RevsetGraphEdgeType::Indirect)
{
return edges;
}
let mut min_generation = u32::MAX;
let mut initial_targets = HashSet::new();
let mut work = vec![];
for (target, edge) in &edges {
initial_targets.insert(target);
if edge.edge_type != RevsetGraphEdgeType::Missing {
let entry = self.look_ahead.get(target).unwrap().clone();
min_generation = min(min_generation, entry.generation_number());
work.extend(self.edges_from_internal_commit(&entry));
}
}
let mut unwanted = HashSet::new();
while let Some((target, edge)) = work.pop() {
if edge.edge_type == RevsetGraphEdgeType::Missing || target < self.min_position {
continue;
}
if !unwanted.insert(target) {
continue;
}
if initial_targets.contains(&target) {
continue;
}
let entry = self.look_ahead.get(&target).unwrap().clone();
if entry.generation_number() < min_generation {
continue;
}
work.extend(self.edges_from_internal_commit(&entry));
}
edges
.into_iter()
.filter(|(target, _)| !unwanted.contains(target))
.collect()
}
fn consume_to(&mut self, pos: IndexPosition) {
while pos < self.min_position {
if let Some(next_entry) = self.input_set_iter.next() {
let next_position = next_entry.position();
self.look_ahead.insert(next_position, next_entry);
self.min_position = next_position;
} else {
break;
}
}
}
}
impl<'revset, 'index> Iterator for RevsetGraphIterator<'revset, 'index> {
type Item = (CommitId, Vec<RevsetGraphEdge>);
fn next(&mut self) -> Option<Self::Item> {
let index_entry = self.next_index_entry()?;
let mut edges = self.edges_from_internal_commit(&index_entry);
if self.skip_transitive_edges {
edges = self.remove_transitive_edges(edges);
}
let mut edges: Vec<_> = edges.into_iter().collect();
edges.sort_by(|(target_pos1, _), (target_pos2, _)| target_pos2.cmp(target_pos1));
let edges = edges.into_iter().map(|(_, edge)| edge).collect();
Some((index_entry.commit_id(), edges))
}
}