blob: 421188adcda735b828472e6f5ba83f63df8153e9 [file]
// Copyright 2026 The Go Authors. All rights reserved.
// Use of this source code is governed by a BSD-style
// license that can be found in the LICENSE file.
package graph
import (
"iter"
"slices"
)
type transpose[NodeID comparable] struct {
Graph Graph[NodeID]
preds map[NodeID][]NodeID
}
// transpose returns a graph like g but with all edges reversed. Node IDs are
// identical to the underlying graph.
//
// Transpose preserves compactness.
func Transpose[NodeID comparable](g Graph[NodeID]) Graph[NodeID] {
if g, ok := g.(transpose[NodeID]); ok {
// Transpose(Transpose(g)) == g
return g.Graph
}
preds := make(map[NodeID][]NodeID)
for nid := range g.Nodes() {
for succ := range g.Out(nid) {
preds[succ] = append(preds[succ], nid)
}
}
return transpose[NodeID]{g, preds}
}
func (t transpose[NodeID]) NumNodes() int {
return len(t.preds)
}
func (t transpose[NodeID]) Nodes() iter.Seq[NodeID] {
return t.Graph.Nodes()
}
func (t transpose[NodeID]) Out(n NodeID) iter.Seq[NodeID] {
return slices.Values(t.preds[n])
}
func (t transpose[NodeID]) unwrapPreservingNodes() Graph[NodeID] {
return t.Graph
}