184 lines
5.8 KiB
Go
184 lines
5.8 KiB
Go
// 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 gremlin
|
|
|
|
// Adds special traversal functions to JS Gremlin objects. Most of these just build the chain of objects, and won't often need the session.
|
|
|
|
import (
|
|
"github.com/barakmich/glog"
|
|
"github.com/robertkrimen/otto"
|
|
)
|
|
|
|
func embedTraversals(env *otto.Otto, ses *GremlinSession, obj *otto.Object) {
|
|
obj.Set("In", gremlinFunc("in", obj, env, ses))
|
|
obj.Set("Out", gremlinFunc("out", obj, env, ses))
|
|
obj.Set("Is", gremlinFunc("is", obj, env, ses))
|
|
obj.Set("Both", gremlinFunc("both", obj, env, ses))
|
|
obj.Set("Follow", gremlinFunc("follow", obj, env, ses))
|
|
obj.Set("FollowR", gremlinFollowR("followr", obj, env, ses))
|
|
obj.Set("And", gremlinFunc("and", obj, env, ses))
|
|
obj.Set("Intersect", gremlinFunc("and", obj, env, ses))
|
|
obj.Set("Union", gremlinFunc("or", obj, env, ses))
|
|
obj.Set("Or", gremlinFunc("or", obj, env, ses))
|
|
obj.Set("Back", gremlinBack("back", obj, env, ses))
|
|
obj.Set("Tag", gremlinFunc("tag", obj, env, ses))
|
|
obj.Set("As", gremlinFunc("tag", obj, env, ses))
|
|
obj.Set("Has", gremlinFunc("has", obj, env, ses))
|
|
obj.Set("Save", gremlinFunc("save", obj, env, ses))
|
|
obj.Set("SaveR", gremlinFunc("saver", obj, env, ses))
|
|
}
|
|
|
|
func gremlinFunc(kind string, prevObj *otto.Object, env *otto.Otto, ses *GremlinSession) func(otto.FunctionCall) otto.Value {
|
|
return func(call otto.FunctionCall) otto.Value {
|
|
call.Otto.Run("var out = {}")
|
|
out, _ := call.Otto.Object("out")
|
|
out.Set("_gremlin_type", kind)
|
|
out.Set("_gremlin_values", call.ArgumentList)
|
|
out.Set("_gremlin_prev", prevObj)
|
|
outStrings := concatStringArgs(call)
|
|
if len(*outStrings) > 0 {
|
|
out.Set("string_args", *outStrings)
|
|
}
|
|
embedTraversals(env, ses, out)
|
|
if isVertexChain(call.This.Object()) {
|
|
embedFinals(env, ses, out)
|
|
}
|
|
return out.Value()
|
|
}
|
|
}
|
|
|
|
func gremlinBack(kind string, prevObj *otto.Object, env *otto.Otto, ses *GremlinSession) func(otto.FunctionCall) otto.Value {
|
|
return func(call otto.FunctionCall) otto.Value {
|
|
call.Otto.Run("var out = {}")
|
|
out, _ := call.Otto.Object("out")
|
|
out.Set("_gremlin_type", kind)
|
|
out.Set("_gremlin_values", call.ArgumentList)
|
|
outStrings := concatStringArgs(call)
|
|
if len(*outStrings) > 0 {
|
|
out.Set("string_args", *outStrings)
|
|
}
|
|
var otherChain *otto.Object
|
|
var thisObj *otto.Object
|
|
if len(*outStrings) != 0 {
|
|
otherChain, thisObj = reverseGremlinChainTo(call.Otto, prevObj, (*outStrings)[0].(string))
|
|
} else {
|
|
otherChain, thisObj = reverseGremlinChainTo(call.Otto, prevObj, "")
|
|
}
|
|
out.Set("_gremlin_prev", thisObj)
|
|
out.Set("_gremlin_back_chain", otherChain)
|
|
embedTraversals(env, ses, out)
|
|
if isVertexChain(call.This.Object()) {
|
|
embedFinals(env, ses, out)
|
|
}
|
|
return out.Value()
|
|
|
|
}
|
|
}
|
|
|
|
func gremlinFollowR(kind string, prevObj *otto.Object, env *otto.Otto, ses *GremlinSession) func(otto.FunctionCall) otto.Value {
|
|
return func(call otto.FunctionCall) otto.Value {
|
|
call.Otto.Run("var out = {}")
|
|
out, _ := call.Otto.Object("out")
|
|
out.Set("_gremlin_type", kind)
|
|
out.Set("_gremlin_values", call.ArgumentList)
|
|
outStrings := concatStringArgs(call)
|
|
if len(*outStrings) > 0 {
|
|
out.Set("string_args", *outStrings)
|
|
}
|
|
if len(call.ArgumentList) == 0 {
|
|
return prevObj.Value()
|
|
}
|
|
arg := call.Argument(0)
|
|
if isVertexChain(arg.Object()) {
|
|
return prevObj.Value()
|
|
}
|
|
newChain, _ := reverseGremlinChainTo(call.Otto, arg.Object(), "")
|
|
out.Set("_gremlin_prev", prevObj)
|
|
out.Set("_gremlin_followr", newChain)
|
|
embedTraversals(env, ses, out)
|
|
if isVertexChain(call.This.Object()) {
|
|
embedFinals(env, ses, out)
|
|
}
|
|
return out.Value()
|
|
|
|
}
|
|
}
|
|
|
|
func reverseGremlinChainTo(env *otto.Otto, prevObj *otto.Object, tag string) (*otto.Object, *otto.Object) {
|
|
env.Run("var _base_object = {}")
|
|
base, err := env.Object("_base_object")
|
|
if err != nil {
|
|
glog.Error(err)
|
|
return otto.NullValue().Object(), otto.NullValue().Object()
|
|
}
|
|
if isVertexChain(prevObj) {
|
|
base.Set("_gremlin_type", "vertex")
|
|
} else {
|
|
base.Set("_gremlin_type", "morphism")
|
|
}
|
|
return reverseGremlinChainHelper(env, prevObj, base, tag)
|
|
}
|
|
|
|
func reverseGremlinChainHelper(env *otto.Otto, chain *otto.Object, newBase *otto.Object, tag string) (*otto.Object, *otto.Object) {
|
|
kindVal, _ := chain.Get("_gremlin_type")
|
|
kind, _ := kindVal.ToString()
|
|
|
|
if tag != "" {
|
|
if kind == "tag" {
|
|
tags := getStringArgs(chain)
|
|
for _, t := range tags {
|
|
if t == tag {
|
|
return newBase, chain
|
|
}
|
|
}
|
|
}
|
|
}
|
|
|
|
if kind == "morphism" || kind == "vertex" {
|
|
return newBase, chain
|
|
}
|
|
var newKind string
|
|
switch kind {
|
|
case "in":
|
|
newKind = "out"
|
|
case "out":
|
|
newKind = "in"
|
|
default:
|
|
newKind = kind
|
|
}
|
|
prev, _ := chain.Get("_gremlin_prev")
|
|
env.Run("var out = {}")
|
|
out, _ := env.Object("out")
|
|
out.Set("_gremlin_type", newKind)
|
|
values, _ := chain.Get("_gremlin_values")
|
|
out.Set("_gremlin_values", values)
|
|
back, _ := chain.Get("_gremlin_back_chain")
|
|
out.Set("_gremlin_back_chain", back)
|
|
out.Set("_gremlin_prev", newBase)
|
|
strings, _ := chain.Get("string_args")
|
|
out.Set("string_args", strings)
|
|
return reverseGremlinChainHelper(env, prev.Object(), out, tag)
|
|
}
|
|
|
|
func debugChain(obj *otto.Object) bool {
|
|
val, _ := obj.Get("_gremlin_type")
|
|
x, _ := val.ToString()
|
|
glog.V(2).Infoln(x)
|
|
val, _ = obj.Get("_gremlin_prev")
|
|
if val.IsObject() {
|
|
return debugChain(val.Object())
|
|
}
|
|
return false
|
|
}
|