1
2
3
4
5
6
7
8
9
10
11
12
13
14
15
16
17
18
19
20
21
22
23
24
25
26
27
28
29
30
31
32
33
34
35
36
37
38
39
40
41
42
43
44
45
46
47
48
49
50
51
52
53
54
55
56
57
58
59
60
61
62
63
64
65
66
67
68
69
70
71
72
73
74
75
76
77
78
79
80
81
82
83
84
85
86
87
88
89
90
91
92
93
94
95
96
97
98
99
100
101
102
103
104
105
106
107
108
109
110
111
112
113
114
115
116
117
118
119
120
121
122
123
124
125
126
127
128
129
130
131
132
133
134
135
136
137
138
139
140
141
142
143
144
145
146
147
148
149
150
151
152
153
154
use fixedbitset::FixedBitSet;
use nested::Nested;
use petgraph::{
algo::kosaraju_scc,
graph::IndexType,
prelude::*,
visit::{IntoNeighborsDirected, IntoNodeIdentifiers, VisitMap, Visitable},
};
use std::{collections::HashMap, slice};
#[derive(Clone, Debug)]
pub(crate) struct Sccs<Ix: IndexType> {
sccs: Nested<Vec<NodeIndex<Ix>>>,
multi_map: HashMap<NodeIndex<Ix>, usize>,
}
impl<Ix: IndexType> Sccs<Ix> {
pub fn new<G>(graph: G, mut scc_sorter: impl FnMut(&mut Vec<NodeIndex<Ix>>)) -> Self
where
G: IntoNeighborsDirected<NodeId = NodeIndex<Ix>> + Visitable + IntoNodeIdentifiers,
<G as Visitable>::Map: VisitMap<NodeIndex<Ix>>,
{
let sccs = kosaraju_scc(graph);
let sccs: Nested<Vec<_>> = sccs
.into_iter()
.map(|mut scc| {
if scc.len() > 1 {
scc_sorter(&mut scc);
}
scc
})
.rev()
.collect();
let mut multi_map = HashMap::new();
for (idx, scc) in sccs.iter().enumerate() {
if scc.len() > 1 {
multi_map.extend(scc.iter().map(|ix| (*ix, idx)));
}
}
Self { sccs, multi_map }
}
pub fn is_same_scc(&self, a: NodeIndex<Ix>, b: NodeIndex<Ix>) -> bool {
if a == b {
return true;
}
match (self.multi_map.get(&a), self.multi_map.get(&b)) {
(Some(a_scc), Some(b_scc)) => a_scc == b_scc,
_ => false,
}
}
pub fn multi_sccs(&self) -> impl Iterator<Item = &[NodeIndex<Ix>]> + DoubleEndedIterator {
self.sccs.iter().filter(|scc| scc.len() > 1)
}
pub fn externals<'a, G>(&'a self, graph: G) -> impl Iterator<Item = NodeIndex<Ix>> + 'a
where
G: 'a + IntoNodeIdentifiers + IntoNeighborsDirected<NodeId = NodeIndex<Ix>>,
Ix: IndexType,
{
let mut external_sccs = FixedBitSet::with_capacity(self.sccs.len());
let mut internal_sccs = FixedBitSet::with_capacity(self.sccs.len());
graph
.node_identifiers()
.filter(move |ix| match self.multi_map.get(ix) {
Some(&scc_idx) => {
if external_sccs.contains(scc_idx) {
return true;
}
if internal_sccs.contains(scc_idx) {
return false;
}
let scc = &self.sccs[scc_idx];
let is_external = scc
.iter()
.flat_map(|ix| {
graph.neighbors_directed(*ix, Incoming)
})
.all(|neighbor_ix| {
match self.multi_map.get(&neighbor_ix) {
Some(neighbor_scc_idx) => neighbor_scc_idx == &scc_idx,
None => false,
}
});
if is_external {
external_sccs.insert(scc_idx);
} else {
internal_sccs.insert(scc_idx);
}
is_external
}
None => {
graph.neighbors_directed(*ix, Incoming).next().is_none()
}
})
}
pub fn node_iter(&self, direction: Direction) -> NodeIter<Ix> {
NodeIter {
node_ixs: self.sccs.data().iter(),
direction,
}
}
}
#[derive(Clone, Debug)]
pub(crate) struct NodeIter<'a, Ix> {
node_ixs: slice::Iter<'a, NodeIndex<Ix>>,
direction: Direction,
}
impl<'a, Ix> NodeIter<'a, Ix> {
#[allow(dead_code)]
pub fn direction(&self) -> Direction {
self.direction
}
}
impl<'a, Ix: IndexType> Iterator for NodeIter<'a, Ix> {
type Item = NodeIndex<Ix>;
fn next(&mut self) -> Option<NodeIndex<Ix>> {
match self.direction {
Direction::Outgoing => self.node_ixs.next().copied(),
Direction::Incoming => self.node_ixs.next_back().copied(),
}
}
}