commit
da854a9a18
5 changed files with 572 additions and 2 deletions
|
|
@ -165,6 +165,9 @@ type Description struct {
|
||||||
Iterators []Description `json:",omitempty"`
|
Iterators []Description `json:",omitempty"`
|
||||||
}
|
}
|
||||||
|
|
||||||
|
// ApplyMorphism is a curried function that can generates a new iterator based on some prior iterator.
|
||||||
|
type ApplyMorphism func(QuadStore, Iterator) Iterator
|
||||||
|
|
||||||
type Nexter interface {
|
type Nexter interface {
|
||||||
// Next advances the iterator to the next value, which will then be available through
|
// Next advances the iterator to the next value, which will then be available through
|
||||||
// the Result method. It returns false if no further advancement is possible, or if an
|
// the Result method. It returns false if no further advancement is possible, or if an
|
||||||
|
|
|
||||||
340
graph/path/path.go
Normal file
340
graph/path/path.go
Normal file
|
|
@ -0,0 +1,340 @@
|
||||||
|
// Copyright 2014 The Cayley Authors. All rights reserved.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package path
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/google/cayley/graph"
|
||||||
|
"github.com/google/cayley/graph/iterator"
|
||||||
|
"github.com/google/cayley/quad"
|
||||||
|
)
|
||||||
|
|
||||||
|
type morphism struct {
|
||||||
|
Name string
|
||||||
|
Reversal func() morphism
|
||||||
|
Apply graph.ApplyMorphism
|
||||||
|
}
|
||||||
|
|
||||||
|
// Path represents either a morphism (a pre-defined path stored for later use),
|
||||||
|
// or a concrete path, consisting of a morphism and an underlying QuadStore.
|
||||||
|
type Path struct {
|
||||||
|
stack []morphism
|
||||||
|
qs graph.QuadStore // Optionally. A nil qs is equivalent to a morphism.
|
||||||
|
}
|
||||||
|
|
||||||
|
// IsMorphism returns whether this Path is a morphism.
|
||||||
|
func (p *Path) IsMorphism() bool { return p.qs == nil }
|
||||||
|
|
||||||
|
// StartMorphism creates a new Path with no underlying QuadStore.
|
||||||
|
func StartMorphism(nodes ...string) *Path {
|
||||||
|
return StartPath(nil, nodes...)
|
||||||
|
}
|
||||||
|
|
||||||
|
// StartPath creates a new Path from a set of nodes and an underlying QuadStore.
|
||||||
|
func StartPath(qs graph.QuadStore, nodes ...string) *Path {
|
||||||
|
return &Path{
|
||||||
|
stack: []morphism{
|
||||||
|
isMorphism(nodes...),
|
||||||
|
},
|
||||||
|
qs: qs,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func PathFromIterator(qs graph.QuadStore, it graph.Iterator) *Path {
|
||||||
|
return &Path{
|
||||||
|
stack: []morphism{
|
||||||
|
iteratorMorphism(it),
|
||||||
|
},
|
||||||
|
qs: qs,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// NewPath creates a new, empty Path.
|
||||||
|
func NewPath(qs graph.QuadStore) *Path {
|
||||||
|
return &Path{
|
||||||
|
qs: qs,
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
// Reverse returns a new Path that is the reverse of the current one.
|
||||||
|
func (p *Path) Reverse() *Path {
|
||||||
|
newPath := NewPath(p.qs)
|
||||||
|
for i := len(p.stack) - 1; i >= 0; i-- {
|
||||||
|
newPath.stack = append(newPath.stack, p.stack[i].Reversal())
|
||||||
|
}
|
||||||
|
return newPath
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *Path) Is(nodes ...string) *Path {
|
||||||
|
p.stack = append(p.stack, isMorphism(nodes...))
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *Path) Tag(tags ...string) *Path {
|
||||||
|
p.stack = append(p.stack, tagMorphism(tags...))
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
|
||||||
|
// Out updates this Path to represent the nodes that are adjacent to the
|
||||||
|
// current nodes, via the given outbound predicate.
|
||||||
|
//
|
||||||
|
// For example:
|
||||||
|
// // Returns the list of nodes that "A" follows.
|
||||||
|
// //
|
||||||
|
// // Will return []string{"B"} if there is a predicate (edge) from "A"
|
||||||
|
// // to "B" labelled "follows".
|
||||||
|
// StartPath(qs, "A").Out("follows")
|
||||||
|
func (p *Path) Out(via ...interface{}) *Path {
|
||||||
|
p.stack = append(p.stack, outMorphism(via...))
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
|
||||||
|
// In updates this Path to represent the nodes that are adjacent to the
|
||||||
|
// current nodes, via the given inbound predicate.
|
||||||
|
//
|
||||||
|
// For example:
|
||||||
|
// // Return the list of nodes that follow "B".
|
||||||
|
// //
|
||||||
|
// // Will return []string{"A", "C", "D"} if there are the appropriate
|
||||||
|
// // edges from those nodes to "B" labelled "follows".
|
||||||
|
// StartPath(qs, "B").In("follows")
|
||||||
|
func (p *Path) In(via ...interface{}) *Path {
|
||||||
|
p.stack = append(p.stack, inMorphism(via...))
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
|
||||||
|
// And updates the current Path to represent the nodes that match both the
|
||||||
|
// current Path so far, and the given Path.
|
||||||
|
func (p *Path) And(path *Path) *Path {
|
||||||
|
p.stack = append(p.stack, andMorphism(path))
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
|
||||||
|
// And updates the current Path to represent the nodes that match either the
|
||||||
|
// current Path so far, or the given Path.
|
||||||
|
func (p *Path) Or(path *Path) *Path {
|
||||||
|
p.stack = append(p.stack, orMorphism(path))
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
|
||||||
|
// Except updates the current Path to represent the all of the current nodes
|
||||||
|
// except those in the supplied Path.
|
||||||
|
//
|
||||||
|
// For example:
|
||||||
|
// // Will return []string{"B"}
|
||||||
|
// StartPath(qs, "A", "B").Except(StartPath(qs, "A"))
|
||||||
|
func (p *Path) Except(path *Path) *Path {
|
||||||
|
p.stack = append(p.stack, exceptMorphism(path))
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *Path) Follow(path *Path) *Path {
|
||||||
|
p.stack = append(p.stack, followMorphism(path))
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
|
||||||
|
func (p *Path) FollowReverse(path *Path) *Path {
|
||||||
|
p.stack = append(p.stack, followMorphism(path.Reverse()))
|
||||||
|
return p
|
||||||
|
}
|
||||||
|
|
||||||
|
// BuildIterator returns an iterator from this given Path. Note that you must
|
||||||
|
// call this with a full path (not a morphism), since a morphism does not have
|
||||||
|
// the ability to fetch the underlying quads. This function will panic if
|
||||||
|
// called with a morphism (i.e. if p.IsMorphism() is true).
|
||||||
|
func (p *Path) BuildIterator() graph.Iterator {
|
||||||
|
if p.IsMorphism() {
|
||||||
|
panic("Building an iterator from a morphism. Bind a QuadStore with BuildIteratorOn(qs)")
|
||||||
|
}
|
||||||
|
return p.BuildIteratorOn(p.qs)
|
||||||
|
}
|
||||||
|
|
||||||
|
// BuildIteratorOn will return an iterator for this path on the given QuadStore.
|
||||||
|
func (p *Path) BuildIteratorOn(qs graph.QuadStore) graph.Iterator {
|
||||||
|
return p.Morphism()(qs, qs.NodesAllIterator())
|
||||||
|
}
|
||||||
|
|
||||||
|
// Morphism returns the morphism of this path. The returned value is a
|
||||||
|
// function that, when given a QuadStore and an existing Iterator, will
|
||||||
|
// return a new Iterator that yields the subset of values from the existing
|
||||||
|
// iterator matched by the current Path.
|
||||||
|
func (p *Path) Morphism() graph.ApplyMorphism {
|
||||||
|
return func(qs graph.QuadStore, it graph.Iterator) graph.Iterator {
|
||||||
|
i := it.Clone()
|
||||||
|
for _, m := range p.stack {
|
||||||
|
i = m.Apply(qs, i)
|
||||||
|
}
|
||||||
|
return i
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func isMorphism(nodes ...string) morphism {
|
||||||
|
return morphism{
|
||||||
|
"is",
|
||||||
|
func() morphism { return isMorphism(nodes...) },
|
||||||
|
func(qs graph.QuadStore, it graph.Iterator) graph.Iterator {
|
||||||
|
var sub graph.Iterator
|
||||||
|
if len(nodes) == 0 {
|
||||||
|
sub = qs.NodesAllIterator()
|
||||||
|
} else {
|
||||||
|
fixed := qs.FixedIterator()
|
||||||
|
for _, n := range nodes {
|
||||||
|
fixed.Add(qs.ValueOf(n))
|
||||||
|
}
|
||||||
|
sub = fixed
|
||||||
|
}
|
||||||
|
and := iterator.NewAnd(qs)
|
||||||
|
and.AddSubIterator(sub)
|
||||||
|
and.AddSubIterator(it)
|
||||||
|
return and
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func tagMorphism(tags ...string) morphism {
|
||||||
|
return morphism{
|
||||||
|
"tag",
|
||||||
|
func() morphism { return tagMorphism(tags...) },
|
||||||
|
func(qs graph.QuadStore, it graph.Iterator) graph.Iterator {
|
||||||
|
for _, t := range tags {
|
||||||
|
it.Tagger().Add(t)
|
||||||
|
}
|
||||||
|
return it
|
||||||
|
}}
|
||||||
|
}
|
||||||
|
|
||||||
|
func outMorphism(via ...interface{}) morphism {
|
||||||
|
return morphism{
|
||||||
|
"out",
|
||||||
|
func() morphism { return inMorphism(via...) },
|
||||||
|
func(qs graph.QuadStore, it graph.Iterator) graph.Iterator {
|
||||||
|
path := buildViaPath(qs, via...)
|
||||||
|
return inOutIterator(path, it, false)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func inMorphism(via ...interface{}) morphism {
|
||||||
|
return morphism{
|
||||||
|
"in",
|
||||||
|
func() morphism { return outMorphism(via...) },
|
||||||
|
func(qs graph.QuadStore, it graph.Iterator) graph.Iterator {
|
||||||
|
path := buildViaPath(qs, via...)
|
||||||
|
return inOutIterator(path, it, true)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func iteratorMorphism(it graph.Iterator) morphism {
|
||||||
|
return morphism{
|
||||||
|
"iterator",
|
||||||
|
func() morphism { return iteratorMorphism(it) },
|
||||||
|
func(qs graph.QuadStore, subIt graph.Iterator) graph.Iterator {
|
||||||
|
and := iterator.NewAnd(qs)
|
||||||
|
and.AddSubIterator(it)
|
||||||
|
and.AddSubIterator(subIt)
|
||||||
|
return and
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func andMorphism(p *Path) morphism {
|
||||||
|
return morphism{
|
||||||
|
"and",
|
||||||
|
func() morphism { return andMorphism(p) },
|
||||||
|
func(qs graph.QuadStore, it graph.Iterator) graph.Iterator {
|
||||||
|
subIt := p.BuildIteratorOn(qs)
|
||||||
|
and := iterator.NewAnd(qs)
|
||||||
|
and.AddSubIterator(it)
|
||||||
|
and.AddSubIterator(subIt)
|
||||||
|
return and
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func orMorphism(p *Path) morphism {
|
||||||
|
return morphism{
|
||||||
|
"or",
|
||||||
|
func() morphism { return orMorphism(p) },
|
||||||
|
func(qs graph.QuadStore, it graph.Iterator) graph.Iterator {
|
||||||
|
subIt := p.BuildIteratorOn(qs)
|
||||||
|
and := iterator.NewOr()
|
||||||
|
and.AddSubIterator(it)
|
||||||
|
and.AddSubIterator(subIt)
|
||||||
|
return and
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func followMorphism(p *Path) morphism {
|
||||||
|
return morphism{
|
||||||
|
"follow",
|
||||||
|
func() morphism { return followMorphism(p.Reverse()) },
|
||||||
|
func(qs graph.QuadStore, base graph.Iterator) graph.Iterator {
|
||||||
|
return p.Morphism()(qs, base)
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func exceptMorphism(p *Path) morphism {
|
||||||
|
return morphism{
|
||||||
|
"except",
|
||||||
|
func() morphism { return exceptMorphism(p) },
|
||||||
|
func(qs graph.QuadStore, base graph.Iterator) graph.Iterator {
|
||||||
|
subIt := p.BuildIteratorOn(qs)
|
||||||
|
notIt := iterator.NewNot(subIt, qs.NodesAllIterator())
|
||||||
|
and := iterator.NewAnd(qs)
|
||||||
|
and.AddSubIterator(base)
|
||||||
|
and.AddSubIterator(notIt)
|
||||||
|
return and
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func inOutIterator(viaPath *Path, it graph.Iterator, reverse bool) graph.Iterator {
|
||||||
|
in, out := quad.Subject, quad.Object
|
||||||
|
if reverse {
|
||||||
|
in, out = out, in
|
||||||
|
}
|
||||||
|
lto := iterator.NewLinksTo(viaPath.qs, it, in)
|
||||||
|
and := iterator.NewAnd(viaPath.qs)
|
||||||
|
and.AddSubIterator(iterator.NewLinksTo(viaPath.qs, viaPath.BuildIterator(), quad.Predicate))
|
||||||
|
and.AddSubIterator(lto)
|
||||||
|
return iterator.NewHasA(viaPath.qs, and, out)
|
||||||
|
}
|
||||||
|
|
||||||
|
func buildViaPath(qs graph.QuadStore, via ...interface{}) *Path {
|
||||||
|
if len(via) == 0 {
|
||||||
|
return PathFromIterator(qs, qs.NodesAllIterator())
|
||||||
|
} else if len(via) == 1 {
|
||||||
|
v := via[0]
|
||||||
|
switch v := v.(type) {
|
||||||
|
case *Path:
|
||||||
|
return v
|
||||||
|
case string:
|
||||||
|
return StartPath(qs, v)
|
||||||
|
default:
|
||||||
|
panic("Invalid type passed to buildViaPath.")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
var strings []string
|
||||||
|
for _, s := range via {
|
||||||
|
if str, ok := s.(string); ok {
|
||||||
|
strings = append(strings, str)
|
||||||
|
} else {
|
||||||
|
panic("Non-string type passed to long Via path")
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return StartPath(qs, strings...)
|
||||||
|
}
|
||||||
181
graph/path/path_test.go
Normal file
181
graph/path/path_test.go
Normal file
|
|
@ -0,0 +1,181 @@
|
||||||
|
// Copyright 2014 The Cayley Authors. All rights reserved.
|
||||||
|
//
|
||||||
|
// Licensed under the Apache License, Version 2.0 (the "License");
|
||||||
|
// you may not use this file except in compliance with the License.
|
||||||
|
// You may obtain a copy of the License at
|
||||||
|
//
|
||||||
|
// http://www.apache.org/licenses/LICENSE-2.0
|
||||||
|
//
|
||||||
|
// Unless required by applicable law or agreed to in writing, software
|
||||||
|
// distributed under the License is distributed on an "AS IS" BASIS,
|
||||||
|
// WITHOUT WARRANTIES OR CONDITIONS OF ANY KIND, either express or implied.
|
||||||
|
// See the License for the specific language governing permissions and
|
||||||
|
// limitations under the License.
|
||||||
|
|
||||||
|
package path
|
||||||
|
|
||||||
|
import (
|
||||||
|
"reflect"
|
||||||
|
"sort"
|
||||||
|
"testing"
|
||||||
|
|
||||||
|
"github.com/google/cayley/graph"
|
||||||
|
"github.com/google/cayley/quad"
|
||||||
|
|
||||||
|
_ "github.com/google/cayley/graph/memstore"
|
||||||
|
_ "github.com/google/cayley/writer"
|
||||||
|
)
|
||||||
|
|
||||||
|
// This is a simple test graph.
|
||||||
|
//
|
||||||
|
// +---+ +---+
|
||||||
|
// | A |------- ->| F |<--
|
||||||
|
// +---+ \------>+---+-/ +---+ \--+---+
|
||||||
|
// ------>|#B#| | | E |
|
||||||
|
// +---+-------/ >+---+ | +---+
|
||||||
|
// | C | / v
|
||||||
|
// +---+ -/ +---+
|
||||||
|
// ---- +---+/ |#G#|
|
||||||
|
// \-->|#D#|------------->+---+
|
||||||
|
// +---+
|
||||||
|
//
|
||||||
|
|
||||||
|
var simpleGraph = []quad.Quad{
|
||||||
|
{"A", "follows", "B", ""},
|
||||||
|
{"C", "follows", "B", ""},
|
||||||
|
{"C", "follows", "D", ""},
|
||||||
|
{"D", "follows", "B", ""},
|
||||||
|
{"B", "follows", "F", ""},
|
||||||
|
{"F", "follows", "G", ""},
|
||||||
|
{"D", "follows", "G", ""},
|
||||||
|
{"E", "follows", "F", ""},
|
||||||
|
{"B", "status", "cool", "status_graph"},
|
||||||
|
{"D", "status", "cool", "status_graph"},
|
||||||
|
{"G", "status", "cool", "status_graph"},
|
||||||
|
{"predicates", "are", "follows", ""},
|
||||||
|
{"predicates", "are", "status", ""},
|
||||||
|
}
|
||||||
|
|
||||||
|
func makeTestStore(data []quad.Quad) graph.QuadStore {
|
||||||
|
qs, _ := graph.NewQuadStore("memstore", "", nil)
|
||||||
|
w, _ := graph.NewQuadWriter("single", qs, nil)
|
||||||
|
for _, t := range data {
|
||||||
|
w.AddQuad(t)
|
||||||
|
}
|
||||||
|
return qs
|
||||||
|
}
|
||||||
|
|
||||||
|
func runTopLevel(path *Path) []string {
|
||||||
|
var out []string
|
||||||
|
it := path.BuildIterator()
|
||||||
|
it, _ = it.Optimize()
|
||||||
|
for graph.Next(it) {
|
||||||
|
v := path.qs.NameOf(it.Result())
|
||||||
|
out = append(out, v)
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
func runTag(path *Path, tag string) []string {
|
||||||
|
var out []string
|
||||||
|
it := path.BuildIterator()
|
||||||
|
it, _ = it.Optimize()
|
||||||
|
for graph.Next(it) {
|
||||||
|
tags := make(map[string]graph.Value)
|
||||||
|
it.TagResults(tags)
|
||||||
|
out = append(out, path.qs.NameOf(tags[tag]))
|
||||||
|
for it.NextPath() {
|
||||||
|
tags := make(map[string]graph.Value)
|
||||||
|
it.TagResults(tags)
|
||||||
|
out = append(out, path.qs.NameOf(tags[tag]))
|
||||||
|
}
|
||||||
|
}
|
||||||
|
return out
|
||||||
|
}
|
||||||
|
|
||||||
|
type test struct {
|
||||||
|
message string
|
||||||
|
path *Path
|
||||||
|
expect []string
|
||||||
|
tag string
|
||||||
|
}
|
||||||
|
|
||||||
|
func testSet(qs graph.QuadStore) []test {
|
||||||
|
return []test{
|
||||||
|
{
|
||||||
|
message: "use out",
|
||||||
|
path: StartPath(qs, "A").Out("follows"),
|
||||||
|
expect: []string{"B"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "use in",
|
||||||
|
path: StartPath(qs, "B").In("follows"),
|
||||||
|
expect: []string{"A", "C", "D"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "use path Out",
|
||||||
|
path: StartPath(qs, "B").Out(StartPath(qs, "predicates").Out("are")),
|
||||||
|
expect: []string{"F", "cool"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "use And",
|
||||||
|
path: StartPath(qs, "D").Out("follows").And(
|
||||||
|
StartPath(qs, "C").Out("follows")),
|
||||||
|
expect: []string{"B"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "use Or",
|
||||||
|
path: StartPath(qs, "F").Out("follows").Or(
|
||||||
|
StartPath(qs, "A").Out("follows")),
|
||||||
|
expect: []string{"B", "G"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "implicit All",
|
||||||
|
path: StartPath(qs),
|
||||||
|
expect: []string{"A", "B", "C", "D", "E", "F", "G", "follows", "status", "cool", "status_graph", "predicates", "are"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "follow",
|
||||||
|
path: StartPath(qs, "C").Follow(StartMorphism().Out("follows").Out("follows")),
|
||||||
|
expect: []string{"B", "F", "G"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "followR",
|
||||||
|
path: StartPath(qs, "F").FollowReverse(StartMorphism().Out("follows").Out("follows")),
|
||||||
|
expect: []string{"A", "C", "D"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "is, tag, instead of FollowR",
|
||||||
|
path: StartPath(qs).Tag("first").Follow(StartMorphism().Out("follows").Out("follows")).Is("F"),
|
||||||
|
expect: []string{"A", "C", "D"},
|
||||||
|
tag: "first",
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "use Except to filter out a single vertex",
|
||||||
|
path: StartPath(qs, "A", "B").Except(StartPath(qs, "A")),
|
||||||
|
expect: []string{"B"},
|
||||||
|
},
|
||||||
|
{
|
||||||
|
message: "use chained Except",
|
||||||
|
path: StartPath(qs, "A", "B", "C").Except(StartPath(qs, "B")).Except(StartPath(qs, "A")),
|
||||||
|
expect: []string{"C"},
|
||||||
|
},
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
||||||
|
func TestMorphisms(t *testing.T) {
|
||||||
|
qs := makeTestStore(simpleGraph)
|
||||||
|
for _, test := range testSet(qs) {
|
||||||
|
var got []string
|
||||||
|
if test.tag == "" {
|
||||||
|
got = runTopLevel(test.path)
|
||||||
|
} else {
|
||||||
|
got = runTag(test.path, test.tag)
|
||||||
|
}
|
||||||
|
sort.Strings(got)
|
||||||
|
sort.Strings(test.expect)
|
||||||
|
if !reflect.DeepEqual(got, test.expect) {
|
||||||
|
t.Errorf("Failed to %s, got: %v expected: %v", test.message, got, test.expect)
|
||||||
|
}
|
||||||
|
}
|
||||||
|
}
|
||||||
|
|
@ -45,8 +45,8 @@ type Delta struct {
|
||||||
}
|
}
|
||||||
|
|
||||||
type Handle struct {
|
type Handle struct {
|
||||||
QuadStore QuadStore
|
QuadStore
|
||||||
QuadWriter QuadWriter
|
QuadWriter
|
||||||
}
|
}
|
||||||
|
|
||||||
type IgnoreOpts struct {
|
type IgnoreOpts struct {
|
||||||
|
|
|
||||||
46
imports.go
Normal file
46
imports.go
Normal file
|
|
@ -0,0 +1,46 @@
|
||||||
|
package cayley
|
||||||
|
|
||||||
|
import (
|
||||||
|
"github.com/google/cayley/graph"
|
||||||
|
_ "github.com/google/cayley/graph/memstore"
|
||||||
|
"github.com/google/cayley/graph/path"
|
||||||
|
"github.com/google/cayley/quad"
|
||||||
|
_ "github.com/google/cayley/writer"
|
||||||
|
)
|
||||||
|
|
||||||
|
type Iterator graph.Iterator
|
||||||
|
type QuadStore graph.QuadStore
|
||||||
|
type QuadWriter graph.QuadWriter
|
||||||
|
|
||||||
|
type Path path.Path
|
||||||
|
|
||||||
|
var StartMorphism = path.StartMorphism
|
||||||
|
var StartPath = path.StartPath
|
||||||
|
|
||||||
|
var RawNext = graph.Next
|
||||||
|
|
||||||
|
type Handle struct {
|
||||||
|
graph.QuadStore
|
||||||
|
graph.QuadWriter
|
||||||
|
}
|
||||||
|
|
||||||
|
func Quad(subject, predicate, object, label string) quad.Quad {
|
||||||
|
return quad.Quad{subject, predicate, object, label}
|
||||||
|
}
|
||||||
|
|
||||||
|
func NewMemoryGraph() (*Handle, error) {
|
||||||
|
qs, err := graph.NewQuadStore("memstore", "", nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
qw, err := graph.NewQuadWriter("single", qs, nil)
|
||||||
|
if err != nil {
|
||||||
|
return nil, err
|
||||||
|
}
|
||||||
|
return &Handle{qs, qw}, nil
|
||||||
|
}
|
||||||
|
|
||||||
|
func (h *Handle) Close() {
|
||||||
|
h.QuadStore.Close()
|
||||||
|
h.QuadWriter.Close()
|
||||||
|
}
|
||||||
Loading…
Add table
Add a link
Reference in a new issue