Merge pull request #145 from kortschak/gremlin

Fix gremlin timeout handling
This commit is contained in:
Barak Michener 2014-08-26 11:56:57 -04:00
commit 1e62aaf374
7 changed files with 272 additions and 283 deletions

View file

@ -25,21 +25,15 @@ import (
"github.com/google/cayley/quad" "github.com/google/cayley/quad"
) )
func getStrings(obj *otto.Object, field string) []string { func propertiesOf(obj *otto.Object, name string) []string {
strings := make([]string, 0) val, _ := obj.Get(name)
val, _ := obj.Get(field) if val.IsUndefined() {
if !val.IsUndefined() { return nil
}
export, _ := val.Export() export, _ := val.Export()
array := export.([]interface{}) return export.([]string)
for _, arg := range array {
strings = append(strings, arg.(string))
}
}
return strings
} }
func getStringArgs(obj *otto.Object) []string { return getStrings(obj, "string_args") }
func buildIteratorTree(obj *otto.Object, ts graph.TripleStore) graph.Iterator { func buildIteratorTree(obj *otto.Object, ts graph.TripleStore) graph.Iterator {
if !isVertexChain(obj) { if !isVertexChain(obj) {
return iterator.NewNull() return iterator.NewNull()
@ -47,22 +41,18 @@ func buildIteratorTree(obj *otto.Object, ts graph.TripleStore) graph.Iterator {
return buildIteratorTreeHelper(obj, ts, iterator.NewNull()) return buildIteratorTreeHelper(obj, ts, iterator.NewNull())
} }
func makeListOfStringsFromArrayValue(obj *otto.Object) []string { func stringsFrom(obj *otto.Object) []string {
var output []string var output []string
lengthValue, _ := obj.Get("length") lengthValue, _ := obj.Get("length")
length, _ := lengthValue.ToInteger() length, _ := lengthValue.ToInteger()
ulength := uint32(length) ulength := uint32(length)
for index := uint32(0); index < ulength; index += 1 { for i := uint32(0); i < ulength; i++ {
name := strconv.FormatInt(int64(index), 10) name := strconv.FormatInt(int64(i), 10)
value, err := obj.Get(name) value, err := obj.Get(name)
if err != nil { if err != nil || !value.IsString() {
continue continue
} }
if !value.IsString() { output = append(output, value.String())
continue
}
s, _ := value.ToString()
output = append(output, s)
} }
return output return output
} }
@ -87,7 +77,7 @@ func buildIteratorFromValue(val otto.Value, ts graph.TripleStore) graph.Iterator
return buildIteratorTree(val.Object(), ts) return buildIteratorTree(val.Object(), ts)
case "Array": case "Array":
// Had better be an array of strings // Had better be an array of strings
strings := makeListOfStringsFromArrayValue(val.Object()) strings := stringsFrom(val.Object())
it := ts.FixedIterator() it := ts.FixedIterator()
for _, x := range strings { for _, x := range strings {
it.Add(ts.ValueOf(x)) it.Add(ts.ValueOf(x))
@ -101,8 +91,7 @@ func buildIteratorFromValue(val otto.Value, ts graph.TripleStore) graph.Iterator
fallthrough fallthrough
case "String": case "String":
it := ts.FixedIterator() it := ts.FixedIterator()
str, _ := val.ToString() it.Add(ts.ValueOf(val.String()))
it.Add(ts.ValueOf(str))
return it return it
default: default:
glog.Errorln("Trying to handle unsupported Javascript value.") glog.Errorln("Trying to handle unsupported Javascript value.")
@ -130,10 +119,9 @@ func buildInOutIterator(obj *otto.Object, ts graph.TripleStore, base graph.Itera
var tags []string var tags []string
one, _ := argArray.Get("1") one, _ := argArray.Get("1")
if one.IsString() { if one.IsString() {
s, _ := one.ToString() tags = append(tags, one.String())
tags = append(tags, s)
} else if one.Class() == "Array" { } else if one.Class() == "Array" {
tags = makeListOfStringsFromArrayValue(one.Object()) tags = stringsFrom(one.Object())
} }
for _, tag := range tags { for _, tag := range tags {
predicateNodeIterator.Tagger().Add(tag) predicateNodeIterator.Tagger().Add(tag)
@ -152,21 +140,19 @@ func buildInOutIterator(obj *otto.Object, ts graph.TripleStore, base graph.Itera
} }
func buildIteratorTreeHelper(obj *otto.Object, ts graph.TripleStore, base graph.Iterator) graph.Iterator { func buildIteratorTreeHelper(obj *otto.Object, ts graph.TripleStore, base graph.Iterator) graph.Iterator {
var it graph.Iterator var it graph.Iterator = base
it = base
// TODO: Better error handling // TODO: Better error handling
kindVal, _ := obj.Get("_gremlin_type")
stringArgs := getStringArgs(obj)
var subIt graph.Iterator var subIt graph.Iterator
prevVal, _ := obj.Get("_gremlin_prev") if prev, _ := obj.Get("_gremlin_prev"); !prev.IsObject() {
if !prevVal.IsObject() {
subIt = base subIt = base
} else { } else {
subIt = buildIteratorTreeHelper(prevVal.Object(), ts, base) subIt = buildIteratorTreeHelper(prev.Object(), ts, base)
} }
kind, _ := kindVal.ToString() stringArgs := propertiesOf(obj, "string_args")
switch kind { val, _ := obj.Get("_gremlin_type")
switch val.String() {
case "vertex": case "vertex":
if len(stringArgs) == 0 { if len(stringArgs) == 0 {
it = ts.NodesAllIterator() it = ts.NodesAllIterator()

View file

@ -17,46 +17,38 @@ package gremlin
// Builds a new Gremlin environment pointing at a session. // Builds a new Gremlin environment pointing at a session.
import ( import (
"sync"
"github.com/barakmich/glog" "github.com/barakmich/glog"
"github.com/robertkrimen/otto" "github.com/robertkrimen/otto"
"github.com/google/cayley/graph"
) )
func BuildEnviron(ses *Session) *otto.Otto { type worker struct {
ts graph.TripleStore
env *otto.Otto
sync.Mutex
results chan interface{}
shape map[string]interface{}
count int
limit int
kill <-chan struct{}
}
func newWorker(ts graph.TripleStore) *worker {
env := otto.New() env := otto.New()
setupGremlin(env, ses) wk := &worker{
return env ts: ts,
} env: env,
limit: -1,
func concatStringArgs(call otto.FunctionCall) *[]interface{} {
outStrings := make([]interface{}, 0)
for _, arg := range call.ArgumentList {
if arg.IsString() {
outStrings = append(outStrings, arg.String())
} }
if arg.IsObject() && arg.Class() == "Array" {
obj, _ := arg.Export()
for _, x := range obj.([]interface{}) {
outStrings = append(outStrings, x.(string))
}
}
}
return &outStrings
}
func isVertexChain(obj *otto.Object) bool {
val, _ := obj.Get("_gremlin_type")
if x, _ := val.ToString(); x == "vertex" {
return true
}
val, _ = obj.Get("_gremlin_prev")
if val.IsObject() {
return isVertexChain(val.Object())
}
return false
}
func setupGremlin(env *otto.Otto, ses *Session) {
graph, _ := env.Object("graph = {}") graph, _ := env.Object("graph = {}")
env.Run("g = graph")
graph.Set("Vertex", func(call otto.FunctionCall) otto.Value { graph.Set("Vertex", func(call otto.FunctionCall) otto.Value {
call.Otto.Run("var out = {}") call.Otto.Run("var out = {}")
out, err := call.Otto.Object("out") out, err := call.Otto.Object("out")
@ -65,31 +57,64 @@ func setupGremlin(env *otto.Otto, ses *Session) {
return otto.TrueValue() return otto.TrueValue()
} }
out.Set("_gremlin_type", "vertex") out.Set("_gremlin_type", "vertex")
outStrings := concatStringArgs(call) args := argsOf(call)
if len(*outStrings) > 0 { if len(args) > 0 {
out.Set("string_args", *outStrings) out.Set("string_args", args)
} }
embedTraversals(env, ses, out) wk.embedTraversals(env, out)
embedFinals(env, ses, out) wk.embedFinals(env, out)
return out.Value() return out.Value()
}) })
env.Run("graph.V = graph.Vertex")
graph.Set("Morphism", func(call otto.FunctionCall) otto.Value { graph.Set("Morphism", func(call otto.FunctionCall) otto.Value {
call.Otto.Run("var out = {}") call.Otto.Run("var out = {}")
out, _ := call.Otto.Object("out") out, _ := call.Otto.Object("out")
out.Set("_gremlin_type", "morphism") out.Set("_gremlin_type", "morphism")
embedTraversals(env, ses, out) wk.embedTraversals(env, out)
return out.Value() return out.Value()
}) })
env.Run("graph.M = graph.Morphism")
graph.Set("Emit", func(call otto.FunctionCall) otto.Value { graph.Set("Emit", func(call otto.FunctionCall) otto.Value {
value := call.Argument(0) value := call.Argument(0)
if value.IsDefined() { if value.IsDefined() {
ses.SendResult(&Result{val: &value}) wk.send(&Result{val: &value})
} }
return otto.NullValue() return otto.NullValue()
}) })
env.Run("graph.V = graph.Vertex")
env.Run("graph.M = graph.Morphism")
env.Run("g = graph")
return wk
}
func (wk *worker) wantShape() bool {
return wk.shape != nil
}
func argsOf(call otto.FunctionCall) []string {
var out []string
for _, arg := range call.ArgumentList {
if arg.IsString() {
out = append(out, arg.String())
}
if arg.IsObject() && arg.Class() == "Array" {
obj, _ := arg.Export()
for _, x := range obj.([]interface{}) {
out = append(out, x.(string))
}
}
}
return out
}
func isVertexChain(obj *otto.Object) bool {
val, _ := obj.Get("_gremlin_type")
if val.String() == "vertex" {
return true
}
val, _ = obj.Get("_gremlin_prev")
if val.IsObject() {
return isVertexChain(val.Object())
}
return false
} }

View file

@ -26,45 +26,45 @@ import (
const TopResultTag = "id" const TopResultTag = "id"
func embedFinals(env *otto.Otto, ses *Session, obj *otto.Object) { func (wk *worker) embedFinals(env *otto.Otto, obj *otto.Object) {
obj.Set("All", allFunc(env, ses, obj)) obj.Set("All", wk.allFunc(env, obj))
obj.Set("GetLimit", limitFunc(env, ses, obj)) obj.Set("GetLimit", wk.limitFunc(env, obj))
obj.Set("ToArray", toArrayFunc(env, ses, obj, false)) obj.Set("ToArray", wk.toArrayFunc(env, obj, false))
obj.Set("ToValue", toValueFunc(env, ses, obj, false)) obj.Set("ToValue", wk.toValueFunc(env, obj, false))
obj.Set("TagArray", toArrayFunc(env, ses, obj, true)) obj.Set("TagArray", wk.toArrayFunc(env, obj, true))
obj.Set("TagValue", toValueFunc(env, ses, obj, true)) obj.Set("TagValue", wk.toValueFunc(env, obj, true))
obj.Set("Map", mapFunc(env, ses, obj)) obj.Set("Map", wk.mapFunc(env, obj))
obj.Set("ForEach", mapFunc(env, ses, obj)) obj.Set("ForEach", wk.mapFunc(env, obj))
} }
func allFunc(env *otto.Otto, ses *Session, obj *otto.Object) func(otto.FunctionCall) otto.Value { func (wk *worker) allFunc(env *otto.Otto, obj *otto.Object) func(otto.FunctionCall) otto.Value {
return func(call otto.FunctionCall) otto.Value { return func(call otto.FunctionCall) otto.Value {
it := buildIteratorTree(obj, ses.ts) it := buildIteratorTree(obj, wk.ts)
it.Tagger().Add(TopResultTag) it.Tagger().Add(TopResultTag)
ses.limit = -1 wk.limit = -1
ses.count = 0 wk.count = 0
runIteratorOnSession(it, ses) wk.runIterator(it)
return otto.NullValue() return otto.NullValue()
} }
} }
func limitFunc(env *otto.Otto, ses *Session, obj *otto.Object) func(otto.FunctionCall) otto.Value { func (wk *worker) limitFunc(env *otto.Otto, obj *otto.Object) func(otto.FunctionCall) otto.Value {
return func(call otto.FunctionCall) otto.Value { return func(call otto.FunctionCall) otto.Value {
if len(call.ArgumentList) > 0 { if len(call.ArgumentList) > 0 {
limitVal, _ := call.Argument(0).ToInteger() limitVal, _ := call.Argument(0).ToInteger()
it := buildIteratorTree(obj, ses.ts) it := buildIteratorTree(obj, wk.ts)
it.Tagger().Add(TopResultTag) it.Tagger().Add(TopResultTag)
ses.limit = int(limitVal) wk.limit = int(limitVal)
ses.count = 0 wk.count = 0
runIteratorOnSession(it, ses) wk.runIterator(it)
} }
return otto.NullValue() return otto.NullValue()
} }
} }
func toArrayFunc(env *otto.Otto, ses *Session, obj *otto.Object, withTags bool) func(otto.FunctionCall) otto.Value { func (wk *worker) toArrayFunc(env *otto.Otto, obj *otto.Object, withTags bool) func(otto.FunctionCall) otto.Value {
return func(call otto.FunctionCall) otto.Value { return func(call otto.FunctionCall) otto.Value {
it := buildIteratorTree(obj, ses.ts) it := buildIteratorTree(obj, wk.ts)
it.Tagger().Add(TopResultTag) it.Tagger().Add(TopResultTag)
limit := -1 limit := -1
if len(call.ArgumentList) > 0 { if len(call.ArgumentList) > 0 {
@ -74,10 +74,10 @@ func toArrayFunc(env *otto.Otto, ses *Session, obj *otto.Object, withTags bool)
var val otto.Value var val otto.Value
var err error var err error
if !withTags { if !withTags {
array := runIteratorToArrayNoTags(it, ses, limit) array := wk.runIteratorToArrayNoTags(it, limit)
val, err = call.Otto.ToValue(array) val, err = call.Otto.ToValue(array)
} else { } else {
array := runIteratorToArray(it, ses, limit) array := wk.runIteratorToArray(it, limit)
val, err = call.Otto.ToValue(array) val, err = call.Otto.ToValue(array)
} }
@ -89,21 +89,21 @@ func toArrayFunc(env *otto.Otto, ses *Session, obj *otto.Object, withTags bool)
} }
} }
func toValueFunc(env *otto.Otto, ses *Session, obj *otto.Object, withTags bool) func(otto.FunctionCall) otto.Value { func (wk *worker) toValueFunc(env *otto.Otto, obj *otto.Object, withTags bool) func(otto.FunctionCall) otto.Value {
return func(call otto.FunctionCall) otto.Value { return func(call otto.FunctionCall) otto.Value {
it := buildIteratorTree(obj, ses.ts) it := buildIteratorTree(obj, wk.ts)
it.Tagger().Add(TopResultTag) it.Tagger().Add(TopResultTag)
limit := 1 limit := 1
var val otto.Value var val otto.Value
var err error var err error
if !withTags { if !withTags {
array := runIteratorToArrayNoTags(it, ses, limit) array := wk.runIteratorToArrayNoTags(it, limit)
if len(array) < 1 { if len(array) < 1 {
return otto.NullValue() return otto.NullValue()
} }
val, err = call.Otto.ToValue(array[0]) val, err = call.Otto.ToValue(array[0])
} else { } else {
array := runIteratorToArray(it, ses, limit) array := wk.runIteratorToArray(it, limit)
if len(array) < 1 { if len(array) < 1 {
return otto.NullValue() return otto.NullValue()
} }
@ -115,13 +115,12 @@ func toValueFunc(env *otto.Otto, ses *Session, obj *otto.Object, withTags bool)
} else { } else {
return val return val
} }
} }
} }
func mapFunc(env *otto.Otto, ses *Session, obj *otto.Object) func(otto.FunctionCall) otto.Value { func (wk *worker) mapFunc(env *otto.Otto, obj *otto.Object) func(otto.FunctionCall) otto.Value {
return func(call otto.FunctionCall) otto.Value { return func(call otto.FunctionCall) otto.Value {
it := buildIteratorTree(obj, ses.ts) it := buildIteratorTree(obj, wk.ts)
it.Tagger().Add(TopResultTag) it.Tagger().Add(TopResultTag)
limit := -1 limit := -1
if len(call.ArgumentList) == 0 { if len(call.ArgumentList) == 0 {
@ -132,26 +131,26 @@ func mapFunc(env *otto.Otto, ses *Session, obj *otto.Object) func(otto.FunctionC
limitParsed, _ := call.Argument(0).ToInteger() limitParsed, _ := call.Argument(0).ToInteger()
limit = int(limitParsed) limit = int(limitParsed)
} }
runIteratorWithCallback(it, ses, callback, call, limit) wk.runIteratorWithCallback(it, callback, call, limit)
return otto.NullValue() return otto.NullValue()
} }
} }
func tagsToValueMap(m map[string]graph.Value, ses *Session) map[string]string { func (wk *worker) tagsToValueMap(m map[string]graph.Value) map[string]string {
outputMap := make(map[string]string) outputMap := make(map[string]string)
for k, v := range m { for k, v := range m {
outputMap[k] = ses.ts.NameOf(v) outputMap[k] = wk.ts.NameOf(v)
} }
return outputMap return outputMap
} }
func runIteratorToArray(it graph.Iterator, ses *Session, limit int) []map[string]string { func (wk *worker) runIteratorToArray(it graph.Iterator, limit int) []map[string]string {
output := make([]map[string]string, 0) output := make([]map[string]string, 0)
count := 0 n := 0
it, _ = it.Optimize() it, _ = it.Optimize()
for { for {
select { select {
case <-ses.kill: case <-wk.kill:
return nil return nil
default: default:
} }
@ -160,22 +159,22 @@ func runIteratorToArray(it graph.Iterator, ses *Session, limit int) []map[string
} }
tags := make(map[string]graph.Value) tags := make(map[string]graph.Value)
it.TagResults(tags) it.TagResults(tags)
output = append(output, tagsToValueMap(tags, ses)) output = append(output, wk.tagsToValueMap(tags))
count++ n++
if limit >= 0 && count >= limit { if limit >= 0 && n >= limit {
break break
} }
for it.NextPath() { for it.NextPath() {
select { select {
case <-ses.kill: case <-wk.kill:
return nil return nil
default: default:
} }
tags := make(map[string]graph.Value) tags := make(map[string]graph.Value)
it.TagResults(tags) it.TagResults(tags)
output = append(output, tagsToValueMap(tags, ses)) output = append(output, wk.tagsToValueMap(tags))
count++ n++
if limit >= 0 && count >= limit { if limit >= 0 && n >= limit {
break break
} }
} }
@ -184,22 +183,22 @@ func runIteratorToArray(it graph.Iterator, ses *Session, limit int) []map[string
return output return output
} }
func runIteratorToArrayNoTags(it graph.Iterator, ses *Session, limit int) []string { func (wk *worker) runIteratorToArrayNoTags(it graph.Iterator, limit int) []string {
output := make([]string, 0) output := make([]string, 0)
count := 0 n := 0
it, _ = it.Optimize() it, _ = it.Optimize()
for { for {
select { select {
case <-ses.kill: case <-wk.kill:
return nil return nil
default: default:
} }
if !graph.Next(it) { if !graph.Next(it) {
break break
} }
output = append(output, ses.ts.NameOf(it.Result())) output = append(output, wk.ts.NameOf(it.Result()))
count++ n++
if limit >= 0 && count >= limit { if limit >= 0 && n >= limit {
break break
} }
} }
@ -207,13 +206,13 @@ func runIteratorToArrayNoTags(it graph.Iterator, ses *Session, limit int) []stri
return output return output
} }
func runIteratorWithCallback(it graph.Iterator, ses *Session, callback otto.Value, this otto.FunctionCall, limit int) { func (wk *worker) runIteratorWithCallback(it graph.Iterator, callback otto.Value, this otto.FunctionCall, limit int) {
count := 0 n := 0
it, _ = it.Optimize() it, _ = it.Optimize()
glog.V(2).Infoln(it.DebugString(0)) glog.V(2).Infoln(it.DebugString(0))
for { for {
select { select {
case <-ses.kill: case <-wk.kill:
return return
default: default:
} }
@ -222,24 +221,24 @@ func runIteratorWithCallback(it graph.Iterator, ses *Session, callback otto.Valu
} }
tags := make(map[string]graph.Value) tags := make(map[string]graph.Value)
it.TagResults(tags) it.TagResults(tags)
val, _ := this.Otto.ToValue(tagsToValueMap(tags, ses)) val, _ := this.Otto.ToValue(wk.tagsToValueMap(tags))
val, _ = callback.Call(this.This, val) val, _ = callback.Call(this.This, val)
count++ n++
if limit >= 0 && count >= limit { if limit >= 0 && n >= limit {
break break
} }
for it.NextPath() { for it.NextPath() {
select { select {
case <-ses.kill: case <-wk.kill:
return return
default: default:
} }
tags := make(map[string]graph.Value) tags := make(map[string]graph.Value)
it.TagResults(tags) it.TagResults(tags)
val, _ := this.Otto.ToValue(tagsToValueMap(tags, ses)) val, _ := this.Otto.ToValue(wk.tagsToValueMap(tags))
val, _ = callback.Call(this.This, val) val, _ = callback.Call(this.This, val)
count++ n++
if limit >= 0 && count >= limit { if limit >= 0 && n >= limit {
break break
} }
} }
@ -247,16 +246,37 @@ func runIteratorWithCallback(it graph.Iterator, ses *Session, callback otto.Valu
it.Close() it.Close()
} }
func runIteratorOnSession(it graph.Iterator, ses *Session) { func (wk *worker) send(r *Result) bool {
if ses.wantShape { if wk.limit >= 0 && wk.limit == wk.count {
iterator.OutputQueryShapeForIterator(it, ses.ts, ses.shape) return false
}
select {
case <-wk.kill:
return false
default:
}
if wk.results != nil {
wk.results <- r
wk.count++
if wk.limit >= 0 && wk.limit == wk.count {
return false
} else {
return true
}
}
return false
}
func (wk *worker) runIterator(it graph.Iterator) {
if wk.wantShape() {
iterator.OutputQueryShapeForIterator(it, wk.ts, wk.shape)
return return
} }
it, _ = it.Optimize() it, _ = it.Optimize()
glog.V(2).Infoln(it.DebugString(0)) glog.V(2).Infoln(it.DebugString(0))
for { for {
select { select {
case <-ses.kill: case <-wk.kill:
return return
default: default:
} }
@ -265,18 +285,18 @@ func runIteratorOnSession(it graph.Iterator, ses *Session) {
} }
tags := make(map[string]graph.Value) tags := make(map[string]graph.Value)
it.TagResults(tags) it.TagResults(tags)
if !ses.SendResult(&Result{actualResults: &tags}) { if !wk.send(&Result{actualResults: tags}) {
break break
} }
for it.NextPath() { for it.NextPath() {
select { select {
case <-ses.kill: case <-wk.kill:
return return
default: default:
} }
tags := make(map[string]graph.Value) tags := make(map[string]graph.Value)
it.TagResults(tags) it.TagResults(tags)
if !ses.SendResult(&Result{actualResults: &tags}) { if !wk.send(&Result{actualResults: tags}) {
break break
} }
} }

View file

@ -257,7 +257,7 @@ func runQueryGetTag(g []quad.Quad, query string, tag string) []string {
for res := range c { for res := range c {
data := res.(*Result) data := res.(*Result)
if data.val == nil { if data.val == nil {
val := (*data.actualResults)[tag] val := data.actualResults[tag]
if val != nil { if val != nil {
results = append(results, js.ts.NameOf(val)) results = append(results, js.ts.NameOf(val))
} }

View file

@ -18,7 +18,6 @@ import (
"errors" "errors"
"fmt" "fmt"
"sort" "sort"
"sync"
"time" "time"
"github.com/robertkrimen/otto" "github.com/robertkrimen/otto"
@ -32,31 +31,28 @@ var ErrKillTimeout = errors.New("query timed out")
type Session struct { type Session struct {
ts graph.TripleStore ts graph.TripleStore
results chan interface{}
env *otto.Otto wk *worker
envLock sync.Mutex
debug bool
limit int
count int
dataOutput []interface{}
wantShape bool
shape map[string]interface{}
err error
script *otto.Script script *otto.Script
kill chan struct{} persist *otto.Otto
timeout time.Duration timeout time.Duration
emptyEnv *otto.Otto kill chan struct{}
debug bool
dataOutput []interface{}
err error
} }
func NewSession(ts graph.TripleStore, timeout time.Duration, persist bool) *Session { func NewSession(ts graph.TripleStore, timeout time.Duration, persist bool) *Session {
g := Session{ g := Session{
ts: ts, ts: ts,
limit: -1, wk: newWorker(ts),
timeout: timeout, timeout: timeout,
} }
g.env = BuildEnviron(&g)
if persist { if persist {
g.emptyEnv = g.env g.persist = g.wk.env
} }
return &g return &g
} }
@ -65,7 +61,7 @@ type Result struct {
metaresult bool metaresult bool
err error err error
val *otto.Value val *otto.Value
actualResults *map[string]graph.Value actualResults map[string]graph.Value
} }
func (s *Session) ToggleDebug() { func (s *Session) ToggleDebug() {
@ -74,15 +70,14 @@ func (s *Session) ToggleDebug() {
func (s *Session) GetQuery(input string, out chan map[string]interface{}) { func (s *Session) GetQuery(input string, out chan map[string]interface{}) {
defer close(out) defer close(out)
s.shape = make(map[string]interface{}) s.wk.shape = make(map[string]interface{})
s.wantShape = true s.wk.env.Run(input)
s.env.Run(input) out <- s.wk.shape
out <- s.shape s.wk.shape = nil
s.shape = nil
} }
func (s *Session) InputParses(input string) (query.ParseResult, error) { func (s *Session) InputParses(input string) (query.ParseResult, error) {
script, err := s.env.Compile("", input) script, err := s.wk.env.Compile("", input)
if err != nil { if err != nil {
return query.ParseFail, err return query.ParseFail, err
} }
@ -90,35 +85,13 @@ func (s *Session) InputParses(input string) (query.ParseResult, error) {
return query.Parsed, nil return query.Parsed, nil
} }
func (s *Session) SendResult(r *Result) bool {
if s.limit >= 0 && s.limit == s.count {
return false
}
s.envLock.Lock()
kill := s.kill
s.envLock.Unlock()
select {
case <-kill:
return false
default:
}
if s.results != nil {
s.results <- r
s.count++
if s.limit >= 0 && s.limit == s.count {
return false
} else {
return true
}
}
return false
}
func (s *Session) runUnsafe(input interface{}) (otto.Value, error) { func (s *Session) runUnsafe(input interface{}) (otto.Value, error) {
wk := s.wk
defer func() { defer func() {
if r := recover(); r != nil { if r := recover(); r != nil {
if r == ErrKillTimeout { if r == ErrKillTimeout {
s.err = ErrKillTimeout s.err = ErrKillTimeout
wk.env = s.persist
return return
} }
panic(r) panic(r)
@ -126,49 +99,40 @@ func (s *Session) runUnsafe(input interface{}) (otto.Value, error) {
}() }()
// Use buffered chan to prevent blocking. // Use buffered chan to prevent blocking.
s.env.Interrupt = make(chan func(), 1) wk.env.Interrupt = make(chan func(), 1)
s.kill = make(chan struct{})
wk.kill = s.kill
ready := make(chan struct{})
done := make(chan struct{}) done := make(chan struct{})
defer close(done)
if s.timeout >= 0 { if s.timeout >= 0 {
go func() { go func() {
time.Sleep(s.timeout) time.Sleep(s.timeout)
<-ready
select { select {
case <-done: case <-done:
return
default: default:
close(s.kill) close(s.kill)
s.envLock.Lock() wk.Lock()
defer s.envLock.Unlock() if wk.env != nil {
s.kill = nil wk.env.Interrupt <- func() {
if s.env != nil {
s.env.Interrupt <- func() {
panic(ErrKillTimeout) panic(ErrKillTimeout)
} }
s.env = s.emptyEnv
} }
return wk.Unlock()
} }
}() }()
} }
s.envLock.Lock() wk.Lock()
env := s.env env := wk.env
if s.kill == nil { wk.Unlock()
s.kill = make(chan struct{}) return env.Run(input)
}
s.envLock.Unlock()
close(ready)
out, err := env.Run(input)
close(done)
return out, err
} }
func (s *Session) ExecInput(input string, out chan interface{}, limit int) { func (s *Session) ExecInput(input string, out chan interface{}, _ int) {
defer close(out) defer close(out)
s.err = nil s.err = nil
s.results = out s.wk.results = out
var err error var err error
var value otto.Value var value otto.Value
if s.script == nil { if s.script == nil {
@ -181,11 +145,11 @@ func (s *Session) ExecInput(input string, out chan interface{}, limit int) {
err: err, err: err,
val: &value, val: &value,
} }
s.results = nil s.wk.results = nil
s.script = nil s.script = nil
s.envLock.Lock() s.wk.Lock()
s.env = s.emptyEnv s.wk.env = s.persist
s.envLock.Unlock() s.wk.Unlock()
} }
func (s *Session) ToText(result interface{}) string { func (s *Session) ToText(result interface{}) string {
@ -210,9 +174,9 @@ func (s *Session) ToText(result interface{}) string {
out = fmt.Sprintln("****") out = fmt.Sprintln("****")
if data.val == nil { if data.val == nil {
tags := data.actualResults tags := data.actualResults
tagKeys := make([]string, len(*tags)) tagKeys := make([]string, len(tags))
i := 0 i := 0
for k, _ := range *tags { for k, _ := range tags {
tagKeys[i] = k tagKeys[i] = k
i++ i++
} }
@ -221,7 +185,7 @@ func (s *Session) ToText(result interface{}) string {
if k == "$_" { if k == "$_" {
continue continue
} }
out += fmt.Sprintf("%s : %s\n", k, s.ts.NameOf((*tags)[k])) out += fmt.Sprintf("%s : %s\n", k, s.ts.NameOf(tags[k]))
} }
} else { } else {
if data.val.IsObject() { if data.val.IsObject() {
@ -245,15 +209,15 @@ func (s *Session) BuildJson(result interface{}) {
if data.val == nil { if data.val == nil {
obj := make(map[string]string) obj := make(map[string]string)
tags := data.actualResults tags := data.actualResults
tagKeys := make([]string, len(*tags)) tagKeys := make([]string, len(tags))
i := 0 i := 0
for k, _ := range *tags { for k, _ := range tags {
tagKeys[i] = k tagKeys[i] = k
i++ i++
} }
sort.Strings(tagKeys) sort.Strings(tagKeys)
for _, k := range tagKeys { for _, k := range tagKeys {
obj[k] = s.ts.NameOf((*tags)[k]) obj[k] = s.ts.NameOf(tags[k])
} }
s.dataOutput = append(s.dataOutput, obj) s.dataOutput = append(s.dataOutput, obj)
} else { } else {
@ -273,11 +237,8 @@ func (s *Session) GetJson() ([]interface{}, error) {
if s.err != nil { if s.err != nil {
return nil, s.err return nil, s.err
} }
s.envLock.Lock()
kill := s.kill
s.envLock.Unlock()
select { select {
case <-kill: case <-s.kill:
return nil, ErrKillTimeout return nil, ErrKillTimeout
default: default:
return s.dataOutput, nil return s.dataOutput, nil

View file

@ -21,123 +21,121 @@ import (
"github.com/robertkrimen/otto" "github.com/robertkrimen/otto"
) )
func embedTraversals(env *otto.Otto, ses *Session, obj *otto.Object) { func (wk *worker) embedTraversals(env *otto.Otto, obj *otto.Object) {
obj.Set("In", gremlinFunc("in", obj, env, ses)) obj.Set("In", wk.gremlinFunc("in", obj, env))
obj.Set("Out", gremlinFunc("out", obj, env, ses)) obj.Set("Out", wk.gremlinFunc("out", obj, env))
obj.Set("Is", gremlinFunc("is", obj, env, ses)) obj.Set("Is", wk.gremlinFunc("is", obj, env))
obj.Set("Both", gremlinFunc("both", obj, env, ses)) obj.Set("Both", wk.gremlinFunc("both", obj, env))
obj.Set("Follow", gremlinFunc("follow", obj, env, ses)) obj.Set("Follow", wk.gremlinFunc("follow", obj, env))
obj.Set("FollowR", gremlinFollowR("followr", obj, env, ses)) obj.Set("FollowR", wk.gremlinFollowR("followr", obj, env))
obj.Set("And", gremlinFunc("and", obj, env, ses)) obj.Set("And", wk.gremlinFunc("and", obj, env))
obj.Set("Intersect", gremlinFunc("and", obj, env, ses)) obj.Set("Intersect", wk.gremlinFunc("and", obj, env))
obj.Set("Union", gremlinFunc("or", obj, env, ses)) obj.Set("Union", wk.gremlinFunc("or", obj, env))
obj.Set("Or", gremlinFunc("or", obj, env, ses)) obj.Set("Or", wk.gremlinFunc("or", obj, env))
obj.Set("Back", gremlinBack("back", obj, env, ses)) obj.Set("Back", wk.gremlinBack("back", obj, env))
obj.Set("Tag", gremlinFunc("tag", obj, env, ses)) obj.Set("Tag", wk.gremlinFunc("tag", obj, env))
obj.Set("As", gremlinFunc("tag", obj, env, ses)) obj.Set("As", wk.gremlinFunc("tag", obj, env))
obj.Set("Has", gremlinFunc("has", obj, env, ses)) obj.Set("Has", wk.gremlinFunc("has", obj, env))
obj.Set("Save", gremlinFunc("save", obj, env, ses)) obj.Set("Save", wk.gremlinFunc("save", obj, env))
obj.Set("SaveR", gremlinFunc("saver", obj, env, ses)) obj.Set("SaveR", wk.gremlinFunc("saver", obj, env))
} }
func gremlinFunc(kind string, prevObj *otto.Object, env *otto.Otto, ses *Session) func(otto.FunctionCall) otto.Value { func (wk *worker) gremlinFunc(kind string, prev *otto.Object, env *otto.Otto) func(otto.FunctionCall) otto.Value {
return func(call otto.FunctionCall) otto.Value { return func(call otto.FunctionCall) otto.Value {
call.Otto.Run("var out = {}") call.Otto.Run("var out = {}")
out, _ := call.Otto.Object("out") out, _ := call.Otto.Object("out")
out.Set("_gremlin_type", kind) out.Set("_gremlin_type", kind)
out.Set("_gremlin_values", call.ArgumentList) out.Set("_gremlin_values", call.ArgumentList)
out.Set("_gremlin_prev", prevObj) out.Set("_gremlin_prev", prev)
outStrings := concatStringArgs(call) args := argsOf(call)
if len(*outStrings) > 0 { if len(args) > 0 {
out.Set("string_args", *outStrings) out.Set("string_args", args)
} }
embedTraversals(env, ses, out) wk.embedTraversals(env, out)
if isVertexChain(call.This.Object()) { if isVertexChain(call.This.Object()) {
embedFinals(env, ses, out) wk.embedFinals(env, out)
} }
return out.Value() return out.Value()
} }
} }
func gremlinBack(kind string, prevObj *otto.Object, env *otto.Otto, ses *Session) func(otto.FunctionCall) otto.Value { func (wk *worker) gremlinBack(kind string, prev *otto.Object, env *otto.Otto) func(otto.FunctionCall) otto.Value {
return func(call otto.FunctionCall) otto.Value { return func(call otto.FunctionCall) otto.Value {
call.Otto.Run("var out = {}") call.Otto.Run("var out = {}")
out, _ := call.Otto.Object("out") out, _ := call.Otto.Object("out")
out.Set("_gremlin_type", kind) out.Set("_gremlin_type", kind)
out.Set("_gremlin_values", call.ArgumentList) out.Set("_gremlin_values", call.ArgumentList)
outStrings := concatStringArgs(call) args := argsOf(call)
if len(*outStrings) > 0 { if len(args) > 0 {
out.Set("string_args", *outStrings) out.Set("string_args", args)
} }
var otherChain *otto.Object var otherChain *otto.Object
var thisObj *otto.Object var thisObj *otto.Object
if len(*outStrings) != 0 { if len(args) != 0 {
otherChain, thisObj = reverseGremlinChainTo(call.Otto, prevObj, (*outStrings)[0].(string)) otherChain, thisObj = reverseGremlinChainTo(call.Otto, prev, args[0])
} else { } else {
otherChain, thisObj = reverseGremlinChainTo(call.Otto, prevObj, "") otherChain, thisObj = reverseGremlinChainTo(call.Otto, prev, "")
} }
out.Set("_gremlin_prev", thisObj) out.Set("_gremlin_prev", thisObj)
out.Set("_gremlin_back_chain", otherChain) out.Set("_gremlin_back_chain", otherChain)
embedTraversals(env, ses, out) wk.embedTraversals(env, out)
if isVertexChain(call.This.Object()) { if isVertexChain(call.This.Object()) {
embedFinals(env, ses, out) wk.embedFinals(env, out)
} }
return out.Value() return out.Value()
} }
} }
func gremlinFollowR(kind string, prevObj *otto.Object, env *otto.Otto, ses *Session) func(otto.FunctionCall) otto.Value { func (wk *worker) gremlinFollowR(kind string, prev *otto.Object, env *otto.Otto) func(otto.FunctionCall) otto.Value {
return func(call otto.FunctionCall) otto.Value { return func(call otto.FunctionCall) otto.Value {
call.Otto.Run("var out = {}") call.Otto.Run("var out = {}")
out, _ := call.Otto.Object("out") out, _ := call.Otto.Object("out")
out.Set("_gremlin_type", kind) out.Set("_gremlin_type", kind)
out.Set("_gremlin_values", call.ArgumentList) out.Set("_gremlin_values", call.ArgumentList)
outStrings := concatStringArgs(call) args := argsOf(call)
if len(*outStrings) > 0 { if len(args) > 0 {
out.Set("string_args", *outStrings) out.Set("string_args", args)
} }
if len(call.ArgumentList) == 0 { if len(call.ArgumentList) == 0 {
return prevObj.Value() return prev.Value()
} }
arg := call.Argument(0) arg := call.Argument(0)
if isVertexChain(arg.Object()) { if isVertexChain(arg.Object()) {
return prevObj.Value() return prev.Value()
} }
newChain, _ := reverseGremlinChainTo(call.Otto, arg.Object(), "") newChain, _ := reverseGremlinChainTo(call.Otto, arg.Object(), "")
out.Set("_gremlin_prev", prevObj) out.Set("_gremlin_prev", prev)
out.Set("_gremlin_followr", newChain) out.Set("_gremlin_followr", newChain)
embedTraversals(env, ses, out) wk.embedTraversals(env, out)
if isVertexChain(call.This.Object()) { if isVertexChain(call.This.Object()) {
embedFinals(env, ses, out) wk.embedFinals(env, out)
} }
return out.Value() return out.Value()
} }
} }
func reverseGremlinChainTo(env *otto.Otto, prevObj *otto.Object, tag string) (*otto.Object, *otto.Object) { func reverseGremlinChainTo(env *otto.Otto, prev *otto.Object, tag string) (*otto.Object, *otto.Object) {
env.Run("var _base_object = {}") env.Run("var _base_object = {}")
base, err := env.Object("_base_object") base, err := env.Object("_base_object")
if err != nil { if err != nil {
glog.Error(err) glog.Error(err)
return otto.NullValue().Object(), otto.NullValue().Object() return otto.NullValue().Object(), otto.NullValue().Object()
} }
if isVertexChain(prevObj) { if isVertexChain(prev) {
base.Set("_gremlin_type", "vertex") base.Set("_gremlin_type", "vertex")
} else { } else {
base.Set("_gremlin_type", "morphism") base.Set("_gremlin_type", "morphism")
} }
return reverseGremlinChainHelper(env, prevObj, base, tag) return reverseGremlinChainHelper(env, prev, base, tag)
} }
func reverseGremlinChainHelper(env *otto.Otto, chain *otto.Object, newBase *otto.Object, tag string) (*otto.Object, *otto.Object) { func reverseGremlinChainHelper(env *otto.Otto, chain *otto.Object, newBase *otto.Object, tag string) (*otto.Object, *otto.Object) {
kindVal, _ := chain.Get("_gremlin_type") kindVal, _ := chain.Get("_gremlin_type")
kind, _ := kindVal.ToString() kind := kindVal.String()
if tag != "" { if tag != "" {
if kind == "tag" { if kind == "tag" {
tags := getStringArgs(chain) tags := propertiesOf(chain, "string_args")
for _, t := range tags { for _, t := range tags {
if t == tag { if t == tag {
return newBase, chain return newBase, chain
@ -174,8 +172,7 @@ func reverseGremlinChainHelper(env *otto.Otto, chain *otto.Object, newBase *otto
func debugChain(obj *otto.Object) bool { func debugChain(obj *otto.Object) bool {
val, _ := obj.Get("_gremlin_type") val, _ := obj.Get("_gremlin_type")
x, _ := val.ToString() glog.V(2).Infoln(val)
glog.V(2).Infoln(x)
val, _ = obj.Get("_gremlin_prev") val, _ = obj.Get("_gremlin_prev")
if val.IsObject() { if val.IsObject() {
return debugChain(val.Object()) return debugChain(val.Object())

View file

@ -72,7 +72,7 @@ func (s *Session) InputParses(input string) (query.ParseResult, error) {
return query.Parsed, nil return query.Parsed, nil
} }
func (s *Session) ExecInput(input string, c chan interface{}, limit int) { func (s *Session) ExecInput(input string, c chan interface{}, _ int) {
defer close(c) defer close(c)
var mqlQuery interface{} var mqlQuery interface{}
err := json.Unmarshal([]byte(input), &mqlQuery) err := json.Unmarshal([]byte(input), &mqlQuery)