blob: e75211b248acaf6b5b99ef7ba954b7bd6956fd78 [file] [log] [blame]
// Copyright ©2015 The Gonum 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 network
import (
"fmt"
"math"
"sort"
"testing"
"gonum.org/v1/gonum/floats/scalar"
"gonum.org/v1/gonum/graph/simple"
)
var pageRankTests = []struct {
g []set
damp float64
tol float64
wantTol float64
want map[int64]float64
}{
{
// Example graph from http://en.wikipedia.org/wiki/File:PageRanks-Example.svg 16:17, 8 July 2009
g: []set{
A: nil,
B: linksTo(C),
C: linksTo(B),
D: linksTo(A, B),
E: linksTo(D, B, F),
F: linksTo(B, E),
G: linksTo(B, E),
H: linksTo(B, E),
I: linksTo(B, E),
J: linksTo(E),
K: linksTo(E),
},
damp: 0.85,
tol: 1e-8,
wantTol: 1e-8,
want: map[int64]float64{
A: 0.03278149,
B: 0.38440095,
C: 0.34291029,
D: 0.03908709,
E: 0.08088569,
F: 0.03908709,
G: 0.01616948,
H: 0.01616948,
I: 0.01616948,
J: 0.01616948,
K: 0.01616948,
},
},
{
// Example graph from http://en.wikipedia.org/w/index.php?title=PageRank&oldid=659286279#Power_Method
// Expected result calculated with the given MATLAB code.
g: []set{
A: linksTo(B, C),
B: linksTo(D),
C: linksTo(D, E),
D: linksTo(E),
E: linksTo(A),
},
damp: 0.80,
tol: 1e-3,
wantTol: 1e-3,
want: map[int64]float64{
A: 0.250,
B: 0.140,
C: 0.140,
D: 0.208,
E: 0.262,
},
},
}
func TestPageRank(t *testing.T) {
for i, test := range pageRankTests {
g := simple.NewDirectedGraph()
for u, e := range test.g {
// Add nodes that are not defined by an edge.
if g.Node(int64(u)) == nil {
g.AddNode(simple.Node(u))
}
for v := range e {
g.SetEdge(simple.Edge{F: simple.Node(u), T: simple.Node(v)})
}
}
got := pageRank(g, test.damp, test.tol)
prec := 1 - int(math.Log10(test.wantTol))
for n := range test.g {
if !scalar.EqualWithinAbsOrRel(got[int64(n)], test.want[int64(n)], test.wantTol, test.wantTol) {
t.Errorf("unexpected PageRank result for test %d:\ngot: %v\nwant:%v",
i, orderedFloats(got, prec), orderedFloats(test.want, prec))
break
}
}
}
}
func TestPageRankSparse(t *testing.T) {
for i, test := range pageRankTests {
g := simple.NewDirectedGraph()
for u, e := range test.g {
// Add nodes that are not defined by an edge.
if g.Node(int64(u)) == nil {
g.AddNode(simple.Node(u))
}
for v := range e {
g.SetEdge(simple.Edge{F: simple.Node(u), T: simple.Node(v)})
}
}
got := pageRankSparse(g, test.damp, test.tol)
prec := 1 - int(math.Log10(test.wantTol))
for n := range test.g {
if !scalar.EqualWithinAbsOrRel(got[int64(n)], test.want[int64(n)], test.wantTol, test.wantTol) {
t.Errorf("unexpected PageRank result for test %d:\ngot: %v\nwant:%v",
i, orderedFloats(got, prec), orderedFloats(test.want, prec))
break
}
}
}
}
var edgeWeightedPageRankTests = []struct {
g []set
self, absent float64
edges map[int]map[int64]float64
damp float64
tol float64
wantTol float64
want map[int64]float64
}{
{
// This test case is created according to the result with the following python code
// on python 3.6.4 (using "networkx" of version 2.1)
//
// >>> import networkx as nx
// >>> D = nx.DiGraph()
// >>> D.add_weighted_edges_from([('A', 'B', 0.3), ('A','C', 1.2), ('B', 'A', 0.4), ('C', 'B', 0.3), ('D', 'A', 0.3), ('D', 'B', 2.1)])
// >>> nx.pagerank(D, alpha=0.85, tol=1e-10)
// {'A': 0.3409109390701202, 'B': 0.3522682754411842, 'C': 0.2693207854886954, 'D': 0.037500000000000006}
g: []set{
A: linksTo(B, C),
B: linksTo(A),
C: linksTo(B),
D: linksTo(A, B),
},
edges: map[int]map[int64]float64{
A: {
B: 0.3,
C: 1.2,
},
B: {
A: 0.4,
},
C: {
B: 0.3,
},
D: {
A: 0.3,
B: 2.1,
},
},
damp: 0.85,
tol: 1e-10,
wantTol: 1e-8,
want: map[int64]float64{
A: 0.3409120160955594,
B: 0.3522678129306601,
C: 0.2693201709737804,
D: 0.037500000000000006,
},
},
}
func TestEdgeWeightedPageRank(t *testing.T) {
for i, test := range edgeWeightedPageRankTests {
g := simple.NewWeightedDirectedGraph(test.self, test.absent)
for u, e := range test.g {
// Add nodes that are not defined by an edge.
if g.Node(int64(u)) == nil {
g.AddNode(simple.Node(u))
}
ws, ok := test.edges[u]
if !ok {
t.Errorf("edges not found for %v", u)
}
for v := range e {
if w, ok := ws[v]; ok {
g.SetWeightedEdge(g.NewWeightedEdge(simple.Node(u), simple.Node(v), w))
}
}
}
got := edgeWeightedPageRank(g, test.damp, test.tol)
prec := 1 - int(math.Log10(test.wantTol))
for n := range test.g {
if !scalar.EqualWithinAbsOrRel(got[int64(n)], test.want[int64(n)], test.wantTol, test.wantTol) {
t.Errorf("unexpected PageRank result for test %d:\ngot: %v\nwant:%v",
i, orderedFloats(got, prec), orderedFloats(test.want, prec))
break
}
}
}
}
func TestEdgeWeightedPageRankSparse(t *testing.T) {
for i, test := range edgeWeightedPageRankTests {
g := simple.NewWeightedDirectedGraph(test.self, test.absent)
for u, e := range test.g {
// Add nodes that are not defined by an edge.
if g.Node(int64(u)) == nil {
g.AddNode(simple.Node(u))
}
ws, ok := test.edges[u]
if !ok {
t.Errorf("edges not found for %v", u)
}
for v := range e {
if w, ok := ws[v]; ok {
g.SetWeightedEdge(g.NewWeightedEdge(simple.Node(u), simple.Node(v), w))
}
}
}
got := edgeWeightedPageRankSparse(g, test.damp, test.tol)
prec := 1 - int(math.Log10(test.wantTol))
for n := range test.g {
if !scalar.EqualWithinAbsOrRel(got[int64(n)], test.want[int64(n)], test.wantTol, test.wantTol) {
t.Errorf("unexpected PageRank result for test %d:\ngot: %v\nwant:%v",
i, orderedFloats(got, prec), orderedFloats(test.want, prec))
break
}
}
}
}
func orderedFloats(w map[int64]float64, prec int) []keyFloatVal {
o := make(orderedFloatsMap, 0, len(w))
for k, v := range w {
o = append(o, keyFloatVal{prec: prec, key: k, val: v})
}
sort.Sort(o)
return o
}
type keyFloatVal struct {
prec int
key int64
val float64
}
func (kv keyFloatVal) String() string { return fmt.Sprintf("%c:%.*f", kv.key+'A', kv.prec, kv.val) }
type orderedFloatsMap []keyFloatVal
func (o orderedFloatsMap) Len() int { return len(o) }
func (o orderedFloatsMap) Less(i, j int) bool { return o[i].key < o[j].key }
func (o orderedFloatsMap) Swap(i, j int) { o[i], o[j] = o[j], o[i] }