blob: 21089daba4b8d2e7c057bbc130825f6247097864 [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 org.apache.tinkerpop.gremlin.structure.io.graphson;
import org.apache.tinkerpop.gremlin.process.traversal.Path;
import org.apache.tinkerpop.gremlin.process.traversal.step.util.ImmutablePath;
import org.apache.tinkerpop.gremlin.process.traversal.util.TraversalExplanation;
import org.apache.tinkerpop.gremlin.structure.Edge;
import org.apache.tinkerpop.gremlin.structure.Property;
import org.apache.tinkerpop.gremlin.structure.Vertex;
import org.apache.tinkerpop.gremlin.structure.VertexProperty;
import org.apache.tinkerpop.gremlin.structure.util.detached.DetachedEdge;
import org.apache.tinkerpop.gremlin.structure.util.detached.DetachedProperty;
import org.apache.tinkerpop.gremlin.structure.util.detached.DetachedVertex;
import org.apache.tinkerpop.gremlin.structure.util.detached.DetachedVertexProperty;
import org.apache.tinkerpop.shaded.jackson.databind.ObjectMapper;
import org.junit.Test;
import org.junit.runner.RunWith;
import org.junit.runners.Parameterized;
import java.time.Duration;
import java.time.Instant;
import java.time.LocalDate;
import java.time.LocalDateTime;
import java.time.LocalTime;
import java.time.MonthDay;
import java.time.OffsetDateTime;
import java.time.OffsetTime;
import java.time.Period;
import java.time.Year;
import java.time.YearMonth;
import java.time.ZoneOffset;
import java.time.ZonedDateTime;
import java.util.Arrays;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import static org.apache.tinkerpop.gremlin.process.traversal.dsl.graph.__.__;
import static org.junit.Assert.assertEquals;
/**
* @author Stephen Mallette (http://stephen.genoprime.com)
*/
@RunWith(Parameterized.class)
public class GraphSONMapperTest {
@Parameterized.Parameters(name = "{0}")
public static Iterable<Object[]> data() {
return Arrays.asList(new Object[][]{
{"v1", GraphSONMapper.build().version(GraphSONVersion.V1_0).typeInfo(TypeInfo.NO_TYPES).create().createMapper()},
{"v2", GraphSONMapper.build().version(GraphSONVersion.V2_0).addCustomModule(GraphSONXModuleV2.build()).typeInfo(TypeInfo.NO_TYPES).create().createMapper()},
{"v2-default", GraphSONMapper.build().version(GraphSONVersion.V2_0).addDefaultXModule(true).typeInfo(TypeInfo.NO_TYPES).create().createMapper()}, // alternate construction of v2
{"v3", GraphSONMapper.build().version(GraphSONVersion.V3_0).addCustomModule(GraphSONXModuleV3.build()).typeInfo(TypeInfo.NO_TYPES).create().createMapper()},
});
}
@Parameterized.Parameter(0)
public String version;
@Parameterized.Parameter(1)
public ObjectMapper mapper;
@Test
public void shouldHandleVertex() throws Exception {
final Vertex v = new DetachedVertex(123L, "person", Arrays.asList(
DetachedVertexProperty.build().setId(1).setLabel("name").setValue("alice").create(),
DetachedVertexProperty.build().setId(1).setLabel("age").setValue("31").create()));
final String json = mapper.writeValueAsString(v);
// v2 untyped seems to serialize the VertexProperty label. not changing that since it's already been
// introduced a long while back. v3 dips back to v1 style - since it has never existed prior to 3.7.0
// we can make this change
if (version.startsWith("v1") || version.startsWith("v3"))
assertEquals("{\"id\":123,\"label\":\"person\",\"type\":\"vertex\",\"properties\":{\"name\":[{\"id\":1,\"value\":\"alice\"}],\"age\":[{\"id\":1,\"value\":\"31\"}]}}", json);
else if (version.startsWith("v2"))
assertEquals("{\"id\":123,\"label\":\"person\",\"properties\":{\"name\":[{\"id\":1,\"value\":\"alice\",\"label\":\"name\"}],\"age\":[{\"id\":1,\"value\":\"31\",\"label\":\"age\"}]}}", json);
else
throw new IllegalStateException("Version not accounted for in asserts");
}
@Test
public void shouldHandleEdge() throws Exception {
final Edge e = new DetachedEdge(123L, "knows", new HashMap<String,Object>() {{
put("weight", 0.5d);
}}, 1L, "person", 2L, "person");
final String json = mapper.writeValueAsString(e);
// v2 untyped seems to serialize the VertexProperty label. not changing that since it's already been
// introduced a long while back. v3 dips back to v1 style - since it has never existed prior to 3.7.0
// we can make this change
if (version.startsWith("v1") || version.startsWith("v3"))
assertEquals("{\"id\":123,\"label\":\"knows\",\"type\":\"edge\",\"inVLabel\":\"person\",\"outVLabel\":\"person\",\"inV\":2,\"outV\":1,\"properties\":{\"weight\":0.5}}", json);
else if (version.startsWith("v2"))
assertEquals("{\"id\":123,\"label\":\"knows\",\"inVLabel\":\"person\",\"outVLabel\":\"person\",\"inV\":2,\"outV\":1,\"properties\":{\"weight\":{\"key\":\"weight\",\"value\":0.5}}}", json);
else
throw new IllegalStateException("Version not accounted for in asserts");
}
@Test
public void shouldHandleProperty() throws Exception {
final Property p = new DetachedProperty("k", 123);
final String json = mapper.writeValueAsString(p);
assertEquals("{\"key\":\"k\",\"value\":123}", json);
}
@Test
public void shouldHandleVertexProperty() throws Exception {
final DetachedVertex v = new DetachedVertex(321L, "person", Collections.emptyMap());
final VertexProperty p = new DetachedVertexProperty(123L, "name", "alice",
new HashMap<String,Object>() {{
put("current", true);
}}, v);
final String json = mapper.writeValueAsString(p);
assertEquals("{\"id\":123,\"value\":\"alice\",\"label\":\"name\",\"properties\":{\"current\":true}}", json);
}
@Test
public void shouldHandleVertexPropertyNoMeta() throws Exception {
final DetachedVertex v = new DetachedVertex(321L, "person", Collections.emptyMap());
final VertexProperty p = new DetachedVertexProperty(123L, "name", "alice", Collections.emptyMap(), v);
final String json = mapper.writeValueAsString(p);
assertEquals("{\"id\":123,\"value\":\"alice\",\"label\":\"name\"}", json);
}
@Test
public void shouldHandlePath() throws Exception {
final Vertex v = new DetachedVertex(123L, "person", Arrays.asList(
DetachedVertexProperty.build().setId(1).setLabel("name").setValue("alice").create(),
DetachedVertexProperty.build().setId(1).setLabel("age").setValue("31").create()));
final Path p = ImmutablePath.make().extend(v, new HashSet<>(Collections.singletonList("a"))).
extend(123L, new HashSet<>(Collections.singletonList("b"))).
extend("alice", new HashSet<>(Collections.singletonList("c")));
final String json = mapper.writeValueAsString(p);
if (version.startsWith("v1") || version.startsWith("v3"))
assertEquals("{\"labels\":[[\"a\"],[\"b\"],[\"c\"]],\"objects\":[{\"id\":123,\"label\":\"person\",\"type\":\"vertex\",\"properties\":{\"name\":[{\"id\":1,\"value\":\"alice\"}],\"age\":[{\"id\":1,\"value\":\"31\"}]}},123,\"alice\"]}", json);
else
assertEquals("{\"labels\":[[\"a\"],[\"b\"],[\"c\"]],\"objects\":[{\"id\":123,\"label\":\"person\",\"properties\":{\"name\":[{\"id\":1,\"value\":\"alice\",\"label\":\"name\"}],\"age\":[{\"id\":1,\"value\":\"31\",\"label\":\"age\"}]}},123,\"alice\"]}", json);
}
@Test
public void shouldHandleTraversalExplanation() throws Exception {
final TraversalExplanation te = __().out().outV().outE().explain();
final String json = mapper.writeValueAsString(te);
assertEquals("{\"original\":[\"InjectStep([])\",\"VertexStep(OUT,vertex)\",\"EdgeVertexStep(OUT)\",\"VertexStep(OUT,edge)\"],\"intermediate\":[],\"final\":[\"InjectStep([])\",\"VertexStep(OUT,vertex)\",\"EdgeVertexStep(OUT)\",\"VertexStep(OUT,edge)\"]}", json);
}
@Test
public void shouldHandleDuration()throws Exception {
final Duration o = Duration.ZERO;
final String json = mapper.writeValueAsString(o);
assertEquals("\"" + o.toString() + "\"", json);
}
@Test
public void shouldHandleInstant()throws Exception {
final Instant o = Instant.ofEpochMilli(System.currentTimeMillis());
final String json = mapper.writeValueAsString(o);
assertEquals("\"" + o.toString() + "\"", json);
}
@Test
public void shouldHandleLocalDate()throws Exception {
final LocalDate o = LocalDate.now();
final String json = mapper.writeValueAsString(o);
assertEquals("\"" + o.toString() + "\"", json);
}
@Test
public void shouldHandleLocalDateTime()throws Exception {
final LocalDateTime o = LocalDateTime.now();
final String json = mapper.writeValueAsString(o);
assertEquals("\"" + o.toString() + "\"", json);
}
@Test
public void shouldHandleLocalTime()throws Exception {
final LocalTime o = LocalTime.now();
final String json = mapper.writeValueAsString(o);
assertEquals("\"" + o.toString() + "\"", json);
}
@Test
public void shouldHandleMonthDay()throws Exception {
final MonthDay o = MonthDay.now();
final String json = mapper.writeValueAsString(o);
assertEquals("\"" + o.toString() + "\"", json);
}
@Test
public void shouldHandleOffsetDateTime()throws Exception {
final OffsetDateTime o = OffsetDateTime.now();
final String json = mapper.writeValueAsString(o);
assertEquals("\"" + o.toString() + "\"", json);
}
@Test
public void shouldHandleOffsetTime()throws Exception {
final OffsetTime o = OffsetTime.now();
final String json = mapper.writeValueAsString(o);
assertEquals("\"" + o.toString() + "\"", json);
}
@Test
public void shouldHandlePeriod()throws Exception {
final Period o = Period.ofDays(3);
final String json = mapper.writeValueAsString(o);
assertEquals("\"" + o.toString() + "\"", json);
}
@Test
public void shouldHandleYear()throws Exception {
final Year o = Year.now();
final String json = mapper.writeValueAsString(o);
assertEquals("\"" + o.toString() + "\"", json);
}
@Test
public void shouldHandleYearMonth()throws Exception {
final YearMonth o = YearMonth.now();
final String json = mapper.writeValueAsString(o);
assertEquals("\"" + o.toString() + "\"", json);
}
@Test
public void shouldHandleZonedDateTime()throws Exception {
final ZonedDateTime o = ZonedDateTime.now();
final String json = mapper.writeValueAsString(o);
assertEquals("\"" + o.toString() + "\"", json);
}
@Test
public void shouldHandleZoneOffset()throws Exception {
final ZoneOffset o = ZonedDateTime.now().getOffset();
final String json = mapper.writeValueAsString(o);
assertEquals("\"" + o.toString() + "\"", json);
}
}