blob: d1cd8a9ee50a98c0906e47803ea70ab256550140 [file] [log] [blame]
/*
Licensed to the Apache Software Foundation (ASF) under one
or more contributor license agreements. See the NOTICE file
distributed with this work for additional information
regarding copyright ownership. The ASF licenses this file
to you 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 gremlingo
import (
"github.com/google/uuid"
)
// request represents a request to the server.
type request struct {
requestID uuid.UUID
op string
processor string
args map[string]interface{}
}
const sessionProcessor = "session"
const stringOp = "eval"
const stringProcessor = ""
func makeStringRequest(stringGremlin string, traversalSource string, sessionId string, requestOptions RequestOptions) (req request) {
newProcessor := stringProcessor
newArgs := map[string]interface{}{
"gremlin": stringGremlin,
"aliases": map[string]interface{}{
"g": traversalSource,
},
}
if sessionId != "" {
newProcessor = sessionProcessor
newArgs["session"] = sessionId
}
var requestId uuid.UUID
if requestOptions.requestID == uuid.Nil {
requestId = uuid.New()
} else {
requestId = requestOptions.requestID
}
if requestOptions.bindings != nil {
newArgs["bindings"] = requestOptions.bindings
}
if requestOptions.evaluationTimeout != 0 {
newArgs["evaluationTimeout"] = requestOptions.evaluationTimeout
}
if requestOptions.batchSize != 0 {
newArgs["batchSize"] = requestOptions.batchSize
}
if requestOptions.userAgent != "" {
newArgs["userAgent"] = requestOptions.userAgent
}
return request{
requestID: requestId,
op: stringOp,
processor: newProcessor,
args: newArgs,
}
}
const bytecodeOp = "bytecode"
const bytecodeProcessor = "traversal"
const authOp = "authentication"
const authProcessor = "traversal"
func makeBytecodeRequest(bytecodeGremlin *Bytecode, traversalSource string, sessionId string) (req request) {
newProcessor := bytecodeProcessor
newArgs := map[string]interface{}{
"gremlin": *bytecodeGremlin,
"aliases": map[string]interface{}{
"g": traversalSource,
},
}
if sessionId != "" {
newProcessor = sessionProcessor
newArgs["session"] = sessionId
}
for k, v := range extractReqArgs(bytecodeGremlin) {
newArgs[k] = v
}
return request{
requestID: uuid.New(),
op: bytecodeOp,
processor: newProcessor,
args: newArgs,
}
}
// allowedReqArgs contains the arguments that will be extracted from the
// bytecode and sent with the request.
var allowedReqArgs = map[string]bool{
"evaluationTimeout": true,
"batchSize": true,
"requestId": true,
"userAgent": true,
}
// extractReqArgs extracts request arguments from the provided bytecode.
func extractReqArgs(bytecode *Bytecode) map[string]interface{} {
args := make(map[string]interface{})
for _, insn := range bytecode.sourceInstructions {
switch insn.operator {
case "withStrategies":
for k, v := range extractWithStrategiesReqArgs(insn) {
args[k] = v
}
case "with":
if k, v := extractWithReqArg(insn); k != "" {
args[k] = v
}
}
}
return args
}
// extractWithStrategiesReqArgs extracts request arguments from the passed
// "withStrategies" source instruction. Only OptionsStrategy is considered.
func extractWithStrategiesReqArgs(insn instruction) map[string]interface{} {
args := make(map[string]interface{})
for _, strategyInterface := range insn.arguments {
strategy, ok := strategyInterface.(*traversalStrategy)
if !ok {
// (*GraphTraversalSource).WithStrategies accepts
// TraversalStrategy parameters only. Thus, this
// should be unreachable.
continue
}
if strategy.name != decorationNamespace+"OptionsStrategy" {
continue
}
for k, v := range strategy.configuration {
if allowedReqArgs[k] {
args[k] = v
}
}
}
return args
}
// extractWithReqArg extracts a request argument from the passed "with" source
// instruction.
func extractWithReqArg(insn instruction) (key string, value interface{}) {
if len(insn.arguments) != 2 {
// (*GraphTraversalSource).With accepts two parameters. Thus,
// this should be unreachable.
return "", nil
}
key, ok := insn.arguments[0].(string)
if !ok {
return "", nil
}
if !allowedReqArgs[key] {
return "", nil
}
return key, insn.arguments[1]
}
func makeBasicAuthRequest(auth string) (req request) {
return request{
requestID: uuid.New(),
op: authOp,
processor: authProcessor,
args: map[string]interface{}{
"sasl": auth,
},
}
}
func makeCloseSessionRequest(sessionId string) request {
return request{
requestID: uuid.New(),
op: "close",
processor: "session",
args: map[string]interface{}{
"session": sessionId,
},
}
}