gonum.org/v1/gonum@v0.14.0/graph/multi/directed_test.go (about)

     1  // Copyright ©2014 The Gonum Authors. All rights reserved.
     2  // Use of this source code is governed by a BSD-style
     3  // license that can be found in the LICENSE file.
     4  
     5  package multi_test
     6  
     7  import (
     8  	"math"
     9  	"testing"
    10  
    11  	"golang.org/x/exp/rand"
    12  
    13  	"gonum.org/v1/gonum/graph"
    14  	"gonum.org/v1/gonum/graph/internal/set"
    15  	"gonum.org/v1/gonum/graph/iterator"
    16  	"gonum.org/v1/gonum/graph/multi"
    17  	"gonum.org/v1/gonum/graph/testgraph"
    18  )
    19  
    20  func directedBuilder(nodes []graph.Node, edges []testgraph.WeightedLine, _, _ float64) (g graph.Graph, n []graph.Node, e []testgraph.Edge, s, a float64, ok bool) {
    21  	seen := set.NewNodes()
    22  	dg := multi.NewDirectedGraph()
    23  	for _, n := range nodes {
    24  		seen.Add(n)
    25  		dg.AddNode(n)
    26  	}
    27  	for _, edge := range edges {
    28  		f := dg.Node(edge.From().ID())
    29  		if f == nil {
    30  			f = edge.From()
    31  		}
    32  		t := dg.Node(edge.To().ID())
    33  		if t == nil {
    34  			t = edge.To()
    35  		}
    36  		cl := multi.Line{F: f, T: t, UID: edge.ID()}
    37  		seen.Add(cl.F)
    38  		seen.Add(cl.T)
    39  		e = append(e, cl)
    40  		dg.SetLine(cl)
    41  	}
    42  	if len(seen) != 0 {
    43  		n = make([]graph.Node, 0, len(seen))
    44  	}
    45  	for _, sn := range seen {
    46  		n = append(n, sn)
    47  	}
    48  	return dg, n, e, math.NaN(), math.NaN(), true
    49  }
    50  
    51  func TestDirected(t *testing.T) {
    52  	t.Run("EdgeExistence", func(t *testing.T) {
    53  		testgraph.EdgeExistence(t, directedBuilder, reversesEdges)
    54  	})
    55  	t.Run("LineExistence", func(t *testing.T) {
    56  		testgraph.LineExistence(t, directedBuilder, usesEmpty, reversesEdges)
    57  	})
    58  	t.Run("NodeExistence", func(t *testing.T) {
    59  		testgraph.NodeExistence(t, directedBuilder)
    60  	})
    61  	t.Run("ReturnAdjacentNodes", func(t *testing.T) {
    62  		testgraph.ReturnAdjacentNodes(t, directedBuilder, usesEmpty, reversesEdges)
    63  	})
    64  	t.Run("ReturnAllLines", func(t *testing.T) {
    65  		testgraph.ReturnAllLines(t, directedBuilder, usesEmpty)
    66  	})
    67  	t.Run("ReturnAllNodes", func(t *testing.T) {
    68  		testgraph.ReturnAllNodes(t, directedBuilder, usesEmpty)
    69  	})
    70  	t.Run("ReturnNodeSlice", func(t *testing.T) {
    71  		testgraph.ReturnNodeSlice(t, directedBuilder, usesEmpty)
    72  	})
    73  
    74  	t.Run("AddNodes", func(t *testing.T) {
    75  		testgraph.AddNodes(t, multi.NewDirectedGraph(), 100)
    76  	})
    77  	t.Run("AddArbitraryNodes", func(t *testing.T) {
    78  		testgraph.AddArbitraryNodes(t,
    79  			multi.NewDirectedGraph(),
    80  			testgraph.NewRandomNodes(100, 1, func(id int64) graph.Node { return multi.Node(id) }),
    81  		)
    82  	})
    83  	t.Run("RemoveNodes", func(t *testing.T) {
    84  		g := multi.NewDirectedGraph()
    85  		it := testgraph.NewRandomNodes(100, 1, func(id int64) graph.Node { return multi.Node(id) })
    86  		for it.Next() {
    87  			g.AddNode(it.Node())
    88  		}
    89  		it.Reset()
    90  		rnd := rand.New(rand.NewSource(1))
    91  		for it.Next() {
    92  			u := it.Node()
    93  			d := rnd.Intn(5)
    94  			vit := g.Nodes()
    95  			for d >= 0 && vit.Next() {
    96  				v := vit.Node()
    97  				d--
    98  				g.SetLine(g.NewLine(u, v))
    99  			}
   100  		}
   101  		testgraph.RemoveNodes(t, g)
   102  	})
   103  	t.Run("AddLines", func(t *testing.T) {
   104  		testgraph.AddLines(t, 100,
   105  			multi.NewDirectedGraph(),
   106  			func(id int64) graph.Node { return multi.Node(id) },
   107  			true, // Can update nodes.
   108  		)
   109  	})
   110  	t.Run("RemoveLines", func(t *testing.T) {
   111  		g := multi.NewDirectedGraph()
   112  		it := testgraph.NewRandomNodes(100, 1, func(id int64) graph.Node { return multi.Node(id) })
   113  		for it.Next() {
   114  			g.AddNode(it.Node())
   115  		}
   116  		it.Reset()
   117  		var lines []graph.Line
   118  		rnd := rand.New(rand.NewSource(1))
   119  		for it.Next() {
   120  			u := it.Node()
   121  			d := rnd.Intn(5)
   122  			vit := g.Nodes()
   123  			for d >= 0 && vit.Next() {
   124  				v := vit.Node()
   125  				d--
   126  				l := g.NewLine(u, v)
   127  				g.SetLine(l)
   128  				lines = append(lines, l)
   129  			}
   130  		}
   131  		rnd.Shuffle(len(lines), func(i, j int) {
   132  			lines[i], lines[j] = lines[j], lines[i]
   133  		})
   134  		testgraph.RemoveLines(t, g, iterator.NewOrderedLines(lines))
   135  	})
   136  }
   137  
   138  // Tests Issue #27
   139  func TestEdgeOvercounting(t *testing.T) {
   140  	g := generateDummyGraph()
   141  
   142  	if neigh := graph.NodesOf(g.From(int64(2))); len(neigh) != 2 {
   143  		t.Errorf("Node 2 has incorrect number of neighbors got neighbors %v (count %d), expected 2 neighbors {0,1}", neigh, len(neigh))
   144  	}
   145  }
   146  
   147  func generateDummyGraph() *multi.DirectedGraph {
   148  	nodes := [4]struct{ srcID, targetID int }{
   149  		{2, 1},
   150  		{1, 0},
   151  		{2, 0},
   152  		{0, 2},
   153  	}
   154  
   155  	g := multi.NewDirectedGraph()
   156  
   157  	for i, n := range nodes {
   158  		g.SetLine(multi.Line{F: multi.Node(n.srcID), T: multi.Node(n.targetID), UID: int64(i)})
   159  	}
   160  
   161  	return g
   162  }
   163  
   164  // Test for issue #123 https://github.com/gonum/graph/issues/123
   165  func TestIssue123DirectedGraph(t *testing.T) {
   166  	defer func() {
   167  		if r := recover(); r != nil {
   168  			t.Errorf("unexpected panic: %v", r)
   169  		}
   170  	}()
   171  	g := multi.NewDirectedGraph()
   172  
   173  	n0 := g.NewNode()
   174  	g.AddNode(n0)
   175  
   176  	n1 := g.NewNode()
   177  	g.AddNode(n1)
   178  
   179  	g.RemoveNode(n0.ID())
   180  
   181  	n2 := g.NewNode()
   182  	g.AddNode(n2)
   183  }