blob: eb2566653288223fdc853f2047fbb12821240df8 [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.
*/
import assert from 'assert';
import Bytecode from '../../lib/process/bytecode.js';
import { Vertex } from '../../lib/structure/graph.js';
import { getClient } from '../helper.js';
import { cardinality } from '../../lib/process/traversal.js';
let client, clientCrew;
describe('Client', function () {
before(function () {
client = getClient('gmodern');
clientCrew = helper.getClient('gcrew')
return client.open();
});
after(function () {
clientCrew.close();
return client.close();
});
describe('#submit()', function () {
it('should send bytecode', function () {
return client.submit(new Bytecode().addStep('V', []).addStep('tail', []))
.then(function (result) {
assert.ok(result);
assert.strictEqual(result.length, 1);
assert.ok(result.first().object instanceof Vertex);
});
});
it('should send and parse a script', function () {
return client.submit('g.V().tail()')
.then(function (result) {
assert.ok(result);
assert.strictEqual(result.length, 1);
assert.ok(result.first() instanceof Vertex);
});
});
it('should send and parse a script with bindings', function () {
return client.submit('x + x', { x: 3 })
.then(function (result) {
assert.ok(result);
assert.strictEqual(result.first(), 6);
});
});
it('should send and parse a script with non-native javascript bindings', function () {
return client.submit('card.class.simpleName + ":" + card', { card: cardinality.set } )
.then(function (result) {
assert.ok(result);
assert.strictEqual(result.first(), 'Cardinality:set');
});
});
it('should retrieve the attributes', () => {
return client.submit(new Bytecode().addStep('V', []).addStep('tail', []))
.then(rs => {
assert.ok(rs.attributes instanceof Map);
assert.ok(rs.attributes.get('host'));
});
});
it('should handle Vertex properties for bytecode request', function () {
return client.submit(new Bytecode().addStep('V', [1]))
.then(function (result) {
assert.ok(result);
assert.strictEqual(result.length, 1);
const vertex = result.first().object;
assert.ok(vertex instanceof graphModule.Vertex);
const age = vertex.properties.find(p => p.key === 'age');
const name = vertex.properties.find(p => p.key === 'name');
assert.ok(age);
assert.ok(name);
assert.strictEqual(age.value, 29);
assert.strictEqual(name.value, 'marko');
});
});
it('should skip Vertex properties for bytecode request with tokens', function () {
return client.submit(new Bytecode().addStep('V', [1]), null, {'materializeProperties': 'tokens'})
.then(function (result) {
assert.ok(result);
assert.strictEqual(result.length, 1);
const vertex = result.first().object;
assert.ok(vertex instanceof graphModule.Vertex);
assert.ok(vertex.properties.length === 0);
});
});
it('should handle Vertex properties for gremlin request', function () {
return client.submit('g.V(1)')
.then(function (result) {
assert.ok(result);
assert.strictEqual(result.length, 1);
const vertex = result.first();
assert.ok(vertex instanceof graphModule.Vertex);
const age = vertex.properties.find(p => p.key === 'age');
const name = vertex.properties.find(p => p.key === 'name');
assert.ok(age);
assert.ok(name);
assert.strictEqual(age.value, 29);
assert.strictEqual(name.value, 'marko');
});
});
it('should handle Edge properties for gremlin request', function () {
return client.submit('g.E().has("weight", 0.5).limit(1)')
.then(function (result) {
assert.ok(result);
assert.strictEqual(result.length, 1);
const edge = result.first();
assert.ok(edge instanceof graphModule.Edge);
assert.strictEqual(edge.label, 'knows');
assert.strictEqual(edge.properties[0].value, 0.5);
assert.ok(edge.inV);
assert.ok(edge.outV);
});
});
it('should handle VertexProperty metadata for gremlin request', function () {
return clientCrew.submit('g.V(7).properties("location").limit(1)')
.then(function (result) {
assert.ok(result);
assert.strictEqual(result.length, 1);
const prop = result.first();
assert.ok(prop instanceof graphModule.VertexProperty);
assert.strictEqual(prop.key, 'location');
assert.strictEqual(prop.value, 'centreville');
const startTime = prop.properties.find(p => p.key === 'startTime');
const endTime = prop.properties.find(p => p.key === 'endTime');
assert.ok(startTime);
assert.ok(endTime);
assert.strictEqual(startTime.value, 1990);
assert.strictEqual(endTime.value, 2000);
});
});
it('should skip Vertex properties for gremlin request with tokens', function () {
return client.submit('g.with("materializeProperties", "tokens").V(1)')
.then(function (result) {
assert.ok(result);
assert.strictEqual(result.length, 1);
const vertex = result.first();
assert.ok(vertex instanceof graphModule.Vertex);
assert.ok(vertex.properties.length === 0);
});
});
it('should handle VertexProperties properties for gremlin request', async function () {
const crewClient = getClient('gcrew');
await crewClient.open();
const result = await crewClient.submit('g.V(7)');
assert.ok(result);
assert.strictEqual(result.length, 1);
const vertex = result.first();
assertVertexProperties(vertex);
await crewClient.close();
});
it('should handle VertexProperties properties for bytecode request', async function () {
const crewClient = getClient('gcrew');
await crewClient.open();
const result = await crewClient.submit(new Bytecode().addStep('V', [7]));
assert.ok(result);
assert.strictEqual(result.length, 1);
const vertex = result.first().object;
assertVertexProperties(vertex);
await crewClient.close();
});
it('should be able to stream results from the gremlin server', (done) => {
const output = [];
let calls = 0;
const readable = client.stream('g.V().limit(3)', {}, { batchSize: 2 });
readable.on('data', (data) => {
calls += 1;
data.toArray().forEach(v => output.push(v))
})
readable.on('end', () => {
assert.strictEqual(calls, 2); // limit of 3 with batchSize of 2 should be two function calls
assert.strictEqual(output.length, 3);
assert.ok(output[0] instanceof Vertex);
done();
})
});
it("should be able to iterate stream results async", async () => {
const output = [];
let calls = 0;
const readable = client.stream("g.V().limit(3)", {}, { batchSize: 2 });
for await (const result of readable) {
calls += 1;
result.toArray().forEach((v) => output.push(v));
}
assert.strictEqual(calls, 2); // limit of 3 with batchSize of 2 should be two function calls
assert.strictEqual(output.length, 3);
assert.ok(output[0] instanceof Vertex);
});
it("should get error for malformed requestId for script stream", async () => {
try {
const readable = client.stream('g.V()', {}, {requestId: 'malformed'});
for await (const result of readable) {
assert.fail("malformed requestId should throw");
}
} catch (e) {
assert.ok(e);
assert.ok(e.message);
assert.ok(e.message.includes("is not a valid UUID."));
}
});
it("should get error for malformed requestId for script submit", async () => {
try {
await client.submit('g.V()', {}, {requestId: 'malformed'});
assert.fail("malformed requestId should throw");
} catch (e) {
assert.ok(e);
assert.ok(e.message);
assert.ok(e.message.includes("is not a valid UUID."));
}
});
it("should get error for malformed requestId for bytecode stream", async () => {
try {
const readable = client.stream(new Bytecode().addStep('V', []), {}, {requestId: 'malformed'});
for await (const result of readable) {
assert.fail("malformed requestId should throw");
}
} catch (e) {
assert.ok(e);
assert.ok(e.message);
assert.ok(e.message.includes("is not a valid UUID."));
}
});
it("should get error for malformed requestId for bytecode submit", async () => {
try {
await client.submit(new Bytecode().addStep('V', []), {}, {requestId: 'malformed'});
assert.fail("malformed requestId should throw");
} catch (e) {
assert.ok(e);
assert.ok(e.message);
assert.ok(e.message.includes("is not a valid UUID."));
}
});
it("should reject pending traversal promises if connection closes", async () => {
const closingClient = getClient('gmodern');
await closingClient.open();
const timeout = 10000;
const startTime = Date.now();
let isRejected = false;
const pending = async function submitTraversals() {
while (Date.now() < startTime + timeout) {
try {
await closingClient.submit(new Bytecode().addStep('V', []).addStep('tail', []));
} catch (e) {
isRejected = true;
return;
}
}
};
const pendingPromise = pending();
await closingClient.close();
await pendingPromise;
assert.strictEqual(isRejected, true);
});
it("should end streams on traversals if connection closes", async () => {
const closingClient = getClient('gmodern');
await closingClient.open();
let isRejected = false;
const readable = client.stream('g.V().limit(3)', {}, { batchSize: 2 });
readable.on('end', () => {
isRejected = true;
});
await closingClient.close();
for await (const result of readable) {
// Consume the stream
}
assert.strictEqual(isRejected, true);
});
});
});
function assertVertexProperties(vertex) {
assert.ok(vertex instanceof Vertex);
let locations;
if (vertex.properties instanceof Array) {
locations = vertex.properties.filter(p => p.key == 'location');
} else {
locations = vertex.properties.location
}
assert.strictEqual(locations.length, 3);
const vertexProperty = locations[0];
assert.strictEqual(vertexProperty.value, 'centreville');
const start = vertexProperty.properties.find(p => p.key === 'startTime');
const end = vertexProperty.properties.find(p => p.key === 'endTime');
assert.ok(start);
assert.ok(end);
assert.strictEqual(start.value, 1990);
assert.strictEqual(end.value, 2000);
}