
com.tinkerpop.gremlin.GraphProvider Maven / Gradle / Ivy
The newest version!
package com.tinkerpop.gremlin;
import com.tinkerpop.gremlin.process.Traversal;
import com.tinkerpop.gremlin.process.Traverser;
import com.tinkerpop.gremlin.process.graph.AnonymousGraphTraversal;
import com.tinkerpop.gremlin.process.graph.GraphTraversal;
import com.tinkerpop.gremlin.process.graph.util.DefaultGraphTraversal;
import com.tinkerpop.gremlin.process.traverser.B_O_PA_S_SE_SL_Traverser;
import com.tinkerpop.gremlin.process.traverser.B_O_P_PA_S_SE_SL_Traverser;
import com.tinkerpop.gremlin.process.traverser.B_O_Traverser;
import com.tinkerpop.gremlin.process.traverser.O_Traverser;
import com.tinkerpop.gremlin.structure.Edge;
import com.tinkerpop.gremlin.structure.Element;
import com.tinkerpop.gremlin.structure.Graph;
import com.tinkerpop.gremlin.structure.Property;
import com.tinkerpop.gremlin.structure.Vertex;
import com.tinkerpop.gremlin.structure.VertexProperty;
import com.tinkerpop.gremlin.structure.strategy.GraphStrategy;
import com.tinkerpop.gremlin.structure.util.GraphFactory;
import org.apache.commons.configuration.Configuration;
import java.util.Collections;
import java.util.HashMap;
import java.util.HashSet;
import java.util.Map;
import java.util.Set;
/**
* Those developing Gremlin implementations must provide a GraphProvider implementation so that the
* different test suites know how to instantiate their implementations. Implementers may choose to have multiple
* {@code GraphProvider} implementations to mix and match with multiple test suite implementations. For example,
* create one {@code GraphProvider} that has no indices defined and a separate {@code GraphProvider} that has
* indices. Then create separate test suite implementations for each {@code GraphProvider}. This approach will
* have the test suites executed once for each {@code GraphProvider} ensuring that the {@link Graph} implementation
* works under multiple configurations. Consider making these "extra" tests "integration tests" so that they
* don't have to be executed on every run of the build so as to save time. Run the "integration tests" periodically
* to ensure overall compliance.
*
* @author Stephen Mallette (http://stephen.genoprime.com)
*/
public interface GraphProvider {
/**
* Implementations from {@code gremlin-core} that need to be part of the clear process. This does not exempt
* vendors from having to register their extensions to any of these classes, but does prevent them from
* having to register them in addition to their own.
*/
public static final Set CORE_IMPLEMENTATIONS = new HashSet() {{
add(AnonymousGraphTraversal.Tokens.class);
add(DefaultGraphTraversal.class);
add(B_O_PA_S_SE_SL_Traverser.class);
add(B_O_P_PA_S_SE_SL_Traverser.class);
add(B_O_Traverser.class);
add(O_Traverser.class);
}};
/**
* Creates a new {@link com.tinkerpop.gremlin.structure.Graph} instance using the default
* {@link org.apache.commons.configuration.Configuration} from {@link #standardGraphConfiguration(Class, String)}.
* The default implementation converts the passes the
*/
default public Graph standardTestGraph(final Class> test, final String testMethodName) {
return GraphFactory.open(standardGraphConfiguration(test, testMethodName));
}
/**
* Creates a new {@link com.tinkerpop.gremlin.structure.Graph} instance from the Configuration object using
* {@link com.tinkerpop.gremlin.structure.util.GraphFactory}. The assumption here is that the {@code Configuration}
* has been created by one of the {@link #newGraphConfiguration(String, Class, String)} methods and has therefore
* already been modified by the implementation as necessary for {@link Graph} creation.
*/
default public Graph openTestGraph(final Configuration config) {
return openTestGraph(config, null);
}
/**
* Creates a new {@link com.tinkerpop.gremlin.structure.Graph} instance from the Configuration object using
* {@link com.tinkerpop.gremlin.structure.util.GraphFactory}. The assumption here is that the {@code Configuration}
* has been created by one of the {@link #newGraphConfiguration(String, Class, String)} methods and has therefore
* already been modified by the implementation as necessary for {@link Graph} creation.
*/
default public Graph openTestGraph(final Configuration config, final GraphStrategy... strategies) {
final Graph g = GraphFactory.open(config);
return null == strategies ? g : g.strategy(strategies);
}
/**
* Gets the {@link org.apache.commons.configuration.Configuration} object that can construct a {@link com.tinkerpop.gremlin.structure.Graph} instance from {@link com.tinkerpop.gremlin.structure.util.GraphFactory}.
* Note that this method should create a {@link com.tinkerpop.gremlin.structure.Graph} using the {@code graphName} of "standard", meaning it
* should always return a configuration instance that generates the same {@link com.tinkerpop.gremlin.structure.Graph} from the
* {@link com.tinkerpop.gremlin.structure.util.GraphFactory}.
*/
default public Configuration standardGraphConfiguration(final Class> test, final String testMethodName) {
return newGraphConfiguration("standard", test, testMethodName, Collections.emptyMap());
}
/**
* If possible (usually with persisted graph) clear the space on disk given the configuration that would be used
* to construct the graph. The default implementation simply calls
* {@link #clear(com.tinkerpop.gremlin.structure.Graph, org.apache.commons.configuration.Configuration)} with
* a null graph argument.
*/
public default void clear(final Configuration configuration) throws Exception {
clear(null, configuration);
}
/**
* Clears a {@link com.tinkerpop.gremlin.structure.Graph} of all data and settings. Implementations will have
* different ways of handling this. For a brute force approach, implementers can simply delete data directories
* provided in the configuration. Implementers may choose a more elegant approach if it exists.
*
* Implementations should be able to accept an argument of null for the Graph, in which case the only action
* that can be performed is a clear given the configuration. The method will typically be called this way
* as clean up task on setup to ensure that a persisted graph has a clear space to create a test graph.
*/
public void clear(final Graph g, final Configuration configuration) throws Exception;
/**
* Converts an identifier from a test to an identifier accepted by the Graph instance. Test that try to
* utilize an Element identifier will pass it to this method before usage.
*/
default public Object convertId(final Object id) {
return id;
}
/**
* Converts an label from a test to an label accepted by the Graph instance. Test that try to
* utilize a label will pass it to this method before usage.
*/
default public String convertLabel(final String label) {
return label;
}
/**
* When implementing this method ensure that a test suite can override any settings EXCEPT the
* "gremlin.graph" setting which should be defined by the implementer. It should provide a
* {@link org.apache.commons.configuration.Configuration} that will generate a graph unique to that {@code graphName}.
*
* @param graphName a unique test graph name
* @param test the test class
* @param testMethodName the name of the test
* @param configurationOverrides settings to override defaults with.
*/
public Configuration newGraphConfiguration(final String graphName,
final Class> test,
final String testMethodName,
final Map configurationOverrides);
/**
* When implementing this method ensure that a test suite can override any settings EXCEPT the
* "gremlin.graph" setting which should be defined by the implementer. It should provide a
* {@link org.apache.commons.configuration.Configuration} that will generate a graph unique to that {@code graphName}.
*
* @param graphName a unique test graph name
* @param test the test class
* @param testMethodName the name of the test
*/
default public Configuration newGraphConfiguration(final String graphName,
final Class> test,
final String testMethodName) {
return newGraphConfiguration(graphName, test, testMethodName, new HashMap<>());
}
/**
* Tests are annotated with a {@link com.tinkerpop.gremlin.LoadGraphWith} annotation. These annotations tell
* the test what kind of data to preload into the graph instance. It is up to the implementation to load the
* graph with the data specified by that annotation. This method also represents the place where indices should
* be configured according the the {@link Graph} implementation's API. Implementers can use the {@code testClass}
* and {@code testName} arguments to implement test specific configurations to their graphs.
*
* @param g the {@link Graph} instance to load data into constructed by this {@code GraphProvider}
* @param loadGraphWith the annotation for the currently running test - this value may be null if no graph
* data is to be loaded in front of the test.
* @param testClass the test class being executed
* @param testName the name of the test method being executed
*/
public void loadGraphData(final Graph g, final LoadGraphWith loadGraphWith, final Class testClass, final String testName);
/**
* Converts the GraphSON representation of an identifier to the implementation's representation of an identifier.
* When serializing a mapper identifier type to GraphSON an implementer will typically specify a mapper serializer
* in {@link com.tinkerpop.gremlin.structure.Graph.Io}. That will serialize the identifier to a GraphSON representation.
* When the GraphSON is deserialized, the identifier is written to an
* {@link com.tinkerpop.gremlin.structure.util.detached.Attachable} object where it is passed to a user supplied
* conversion {@link java.util.function.Function} that ultimately processes it. It is in this conversion process
* that vendor specific identifier conversion would occur (if desired). This method mimics that conversion by
* providing the mechanism that a test can use to do the conversion.
*
* @param clazz The {@link Element} class that represents the identifier.
* @param id The identifier to convert.
* @param The type of the identifier.
* @return The reconstituted identifier.
*/
public default ID reconstituteGraphSONIdentifier(final Class extends Element> clazz, final Object id) {
return (ID) id;
}
/**
* Get the set of concrete implementations of certain classes and interfaces utilized by the test suite. This
* method should return any implementations or extensions of the following interfaces or classes:
*
* - {@link Edge}
* - {@link Edge.Iterators}
* - {@link Element}
* - {@link Element.Iterators}
* - {@link DefaultGraphTraversal}
* - {@link Graph}
* - {@link Graph.Variables}
* - {@link Graph.Iterators}
* - {@link GraphTraversal}
* - {@link com.tinkerpop.gremlin.process.traverser.B_O_P_PA_S_SE_SL_Traverser}
* - {@link Property}
* - {@link com.tinkerpop.gremlin.process.traverser.B_O_PA_S_SE_SL_Traverser}
* - {@link Traversal}
* - {@link Traverser}
* - {@link Vertex}
* - {@link Vertex.Iterators}
* - {@link VertexProperty}
* - {@link VertexProperty.Iterators}
*
*
* The test suite only enforces registration of the following core structure interfaces (i.e. these classes must
* be registered or the tests will fail to execute):
*
* - {@link Edge}
* - {@link Edge.Iterators}
* - {@link Element}
* - {@link Element.Iterators}
* - {@link Graph}
* - {@link Graph.Variables}
* - {@link Graph.Iterators}
* - {@link Property}
* - {@link Vertex}
* - {@link Vertex.Iterators}
* - {@link VertexProperty}
* - {@link VertexProperty.Iterators}
*
*
* The remaining interfaces and classes should be registered however as failure to do so, might cause failures
* in the Groovy environment testing suite.
*
* Internally speaking, tests that make use of this method should bind in {@link #CORE_IMPLEMENTATIONS} to the
* {@link Set} because these represent {@code gremlin-core} implementations that are likely not registered
* by the vendor implementations.
*/
public Set getImplementations();
}
© 2015 - 2025 Weber Informatics LLC | Privacy Policy