Indexing Backend Overview - andrew-nguyen/titan GitHub Wiki

Titan provides two types of index systems: The standard index and the external index interface which supports an arbitrary number of external indexing backends to provide support for geo, numeric range, and full-text search. Indexes are needed to retrieve vertices or edges by their properties and are very often used as the starting point for a traversal.

The standard index is very fast and always available without any further configuration, but only supports exact index matches. In other words, the standard index can only retrieve vertices and edges by matching one of their properties exactly. The standard index is implemented directly against the configured storage backend.

The external index interface is more flexible and supports retrieving vertices and edges by bounding their geo-location, properties that fall into a numeric range or matching tokens in full text. The external index interface connects to separate systems as index backends for indexing and retrieval similarly to how storage backends are used for persistence. Index backends need to be configured in the graph configuration before they can be used.

The choice of index backend determines which search features are supported, as well as the performance and scalability of the index. Titan currently supports two index backends: Elastic Search and Lucene.

Indexing Vertices and Edges

To retrieve vertices and edges by their property values through an index lookup, the property key must be registered with the index when it is defined, that is, before its first use. The index is registered through the KeyMaker when the key is defined.

For example, the following property key definitions register all four indexed properties from the Graph of the Gods example with the respective index backends.

// 1) Index vertices by their unique name property
graph.makeKey("name").dataType(String.class).indexed(Vertex.class).unique().make();
// 2) Index vertices by their age property
graph.makeKey("age").dataType(Integer.class).indexed("search", Vertex.class).make();
// 3) Index edges by their geo-location 
graph.makeKey("place").dataType(Geoshape.class).indexed("search", Edge.class).make();
// 4) Index vertices and edges by their full-text reason property
graph.makeKey("reason").dataType(String.class).indexed("search",Vertex.class).indexed("search", Edge.class).make();
  1. The name property key is indexed for vertices using Titan’s standard index. The standard index only supports exact match index retrievals but is very fast. When no index name is specified, the standard index is used.
  2. The age property key is indexed for vertices using the configured external index backend with the name “search”. Since the data type of age is a number, property values are indexed as numbers and can be retrieved via numeric range search.
  3. The place property key is indexed for edges using the external index backend named “search”. Since the data type of place is Geoshape, property values are indexed as geo-location and can be retrieved via geo-searches such as circular region or bounding box search (depending on what the index backed supports).
  4. The reason property key is indexed for vertices and edges using the external index backend named “search”. Since the data type of reason is String, property values are indexed as text and be retrieved via full-text search such as string-containment search. When the string is indexed, it is tokenized.

When using Titan’s standard index, the name argument to the indexed() method is optional. An equivalent definition of the name property key, which identifies the standard index by its name, would be:

graph.makeKey("name").dataType(String.class).indexed("standard",Vertex.class).unique().make();

The name “standard” is always reserved for Titan’s standard index backend. External indexing backends may not be configured with this name.

Note, this section assumes that an index backend named search has been defined in the graph configuration. Read “Next Steps” to find out how to configure an index backend.

Querying an Index

After the property keys have been registered with the index backends and vertices or edges have been added to the graph with corresponding property values, those elements can be queried for using index retrievals. Graph.query() constructs a GraphQuery through method chaining and calling vertices() or edges() retrieves the vertices or edges matching the query using index retrievals.
Continuing with the Graph of the Gods, the following are index query examples:

// 1) Find vertices with the name "hercules"
g.query().has("name",EQUAL,"hercules").vertices()
// 2) Find all vertices with an age greater than 50
g.query().has("age",GREATER_THAN,50).vertices()
// or find all vertices between 1000 (inclusive) and 5000 (exclusive) years of age and order by increasing age
g.query().has("age",GREATER_THAN_EQUAL,1000).has("age",LESS_THAN,5000).orderBy("age",Order.ASC).vertices()
// which returns the same result set as the following query but in reverse order
g.query().interval("age",1000,5000).orderBy("age",Order.DESC).vertices()
// 3) Find all edges where the place is at most 50 kilometers from the given latitude-longitude pair
g.query().has("place",WITHIN,Geoshape.circle(37.97,23.72,50)).edges()
// 4) Find all edges where reason contains the word "loves"
g.query().has("reason",CONTAINS,"loves").edges()
// or all edges which contain two words (need to chunk into individual words)
g.query().has("reason",CONTAINS,"loves").has("reason",CONTAINS,"breezes").edges()
// or all edges which contain words that start with "lov"
g.query().has("reason",CONTAINS_PREFIX,"lov").edges()
// or all edges which contain words that match the regular expression "br[ez]*s" in their entirety
g.query().has("reason",CONTAINS_REGEX,"br[ez]*s").edges()
// 5) Find all vertices older than a thousand years and named "saturn"
g.query().has("age",GREATER_THAN,1000).has("name",EQUAL,"saturn").vertices()
  1. The standard index registered for name supports equality searches. In this case, we retrieve all vertices where the name matches “hercules” exactly (i.e. case sensitive).
  2. The external index registered for age supports numeric range search since the data type for age is numeric. Hence, querying for vertices whose age falls into a given range is supported.
  3. The geo-index registered for place supports retrieving edges by specifying a circular shape as the search region. The circular region is defined by the latitude and longitude of its center (first two arguments) and the radius of the circle in kilometers (third argument).
  4. The full-text index registered for reason supports retrieval by string containment. A property matches if it contains the given word (case insensitive). Note, that the text is tokenized when indexed and stop words as well as punctuation are removed. Two query for properties that contain two or more words, split them up into two separate has() clauses.
  5. Multiple has clauses can be combined into a composite index retrieval potentially spread across multiple index backends. In this example, the query contains a name and age constraint. Titan will optimize the query to find the optimal index retrieval plan.

Compare Predicate

The Compare enum specifies the following self-explanatory comparison predicates used for index query construction and used in the examples above:

  • EQUAL
  • NOT_EQUAL
  • GREATER_THAN
  • GREATER_THAN_EQUAL
  • LESS_THAN
  • LESS_THAN_EQUAL

Text Predicate

The Text enum specifies the full-text predicates used to query for matching text or string values. We differentiate between two types of predicates:

  • Text search predicates which match against the individual words inside a text string after it has been tokenized. These predicates are not case sensitive.
    • CONTAINS: is true if (at least) one word inside the text string matches the query string
    • CONTAINS_PREFIX: is true if (at least) one word inside the text string begins with the query string
    • CONTAINS_REGEX: is true if (at least) one word inside the text string matches the given regular expression
  • String search predicates which match against the entire string value
    • PREFIX: if the string value starts with the given query string
    • REGEX: if the string value matches the given regular expression in its entirety

Read more about full-text and string search and how to specify whether to use full-text or string search.

Geo Predicate

The Geo enum specifies the geo-location predicate Geo.WITHIN which holds true if one geometric object contains the other.

Ordering

The order in which the results are returned can be defined using the orderBy() method of graph query constructor returned by Graph.query(). This method expects to parameters:

  • The name of the property key by which to order the results. The results will be ordered by the value of the vertices or edges for this property key.
  • The sort order: either increasing Order.ASC or decreasing Order.DESC

For example, the query graph.query().has("name",CONTAINS,"john").orderBy("age",Order.DESC).limit(10).vertices() retrieves the ten oldest users who have the word “john” in their name.

When using orderBy() it is important to note that:

  • Titan’s standard indexes do not natively support ordering search results. All results will be retrieved and then sorted in-memory. For large result sets, this can be very expensive.
  • Most external indexing backends support ordering natively and efficiently. However, the property key used in the orderBy method must be configured to be indexed in this indexing backend for native result ordering support. This is important in cases where the orderBy key is different from the query keys. If the property key is not indexed, then sorting requires loading all results into memory.

Choosing an Index

Here are some guidelines for choosing the best indexing strategy for a particular use case:

  1. Use the standard index for exact match index retrievals. The standard index does not require configuring or operating an external index system and is often significantly faster than external index backends.
    1. As an exception, use an external index for exact matches when the number of distinct values for the property key is relatively small or if you expect one value to be associated with many elements in the graph (i.e. in case of low selectivity).
  2. Use an external index for numeric range, full-text or geo-spatial indexing. Also, using an external index can speed up orderBy queries.
    1. Use Elasticsearch when there is an expectation that Titan will be distributed across multiple machines.
    2. Lucene performs better in small scale, single machine applications. It performs better in unit tests, for instance.

Result Scoring

Titan’s Graph.query() currently only supports discrete index queries, that is, an element either matches a query or it does not. In other words, this graph query mechanism does not support result scoring where a “match-goodness” score is associated with each result (so called “fuzzy search”).

Some indexing backends support result scoring which may be accessible through Titan’s direct index query functionality.

Next Steps: Configuring an Index Backend

In order to use an external indexing system with Titan, an index backend has to be configured. Titan currently supports two indexing systems:

  • Lucene: The popular Apache Lucene indexing system
  • Elasticsearch: A distributed indexing system based on Apache Lucene

Refer to the respective documentation pages on how to configure these index backends.

Also, read more about how to configure full-text and string search and how to directly query an indexing backend for more search functionality and result scoring support.

⚠️ **GitHub.com Fallback** ⚠️