Skip to content
lvca edited this page Dec 22, 2012 · 3 revisions

Java Native Fluent API for traversing

Introduction

OrientDB is a graph database. This means that the focal point is on relationships (links) and how are managed. The standard SQL language is not enough to work with tree or graphs because it hasn't the concept of recursion. This is the reason why OrientDB provide a new command to traverse trees and graphs: TRAVERSE. Traversing is the operation that cross relationships between records (documents, vertexes, nodes, etc). This operation is much much faster than executing a JOIN in a Relational database.

The main concepts of Traversal are:

  • target, as the starting point where to traverse records. Can be:
  • class
  • cluster
  • set of records, specifying its RecordID
  • sub-command that returns an Iterable<OIdentifiable>. You can nest multiple select and traverse all together
  • fields, the fields to traverse. Use *, any() or all() to traverse all the fields of a document
  • limit, the maximum number of records to retrieve
  • predicate, as the predicate to execute against each document traversed. If the predicate returns true, then the document is returned, otherwise is skipped

During traversing some context variables are managed and can be used by the traverse condition:

  • $depth, as an integer that contain the depth level of nesting in traversal. First level is 0
  • $path, as a string representation of the current position as the sum of traversed nodes
  • $stack, as the stack current node traversed
  • $history, as the entire collection of visited nodes

Below how to traverse using different approaches.

SQL Traverse

The simpler available way to execute a traversal is using the SQL Traverse command. Example to retrieve all the records connected from and to Movie records up to the 5th level of depth:

    for (OIdentifiable id : new OSQLSynchQuery<ODocument>("traverse in, out from Movie while $depth <= 5")) {
      System.out.println(id);
    }

Look at the command syntax for more information.

Native Fluent API

(Available since 1.0) Native API supports fluent execution guaranteeing compact and readable syntax. The main class is OTraverse:

  • target(<iter:Iterable<OIdentifiable>>), to specify the target as any iterable object like collections or arrays of OIdentifiable objects.
  • target(<iter:Iterator<OIdentifiable>>), to specify the target as any iterator object. To specify a class use database.browseClass(<class-name>).iterator()
  • target(<record:OIdentifiable>, <record:OIdentifiable>, ... ), to specify the target as a var ars of OIterable objects
  • field(<field-name:string>), to specify the document's field to traverse. To add multiple field call this method in chain. Example: .field("in").field("out")
  • fields(<field-name:string>, <field-name:string>, ...), to specify multiple fields in one call passing a var args of Strings
  • fields(Collection<field-name:string>), to specify multiple fields in one call passing a collection of String
  • limit(<max:int>), as the maximum number of record returned
  • predicate(<predicate:OCommandPredicate>), to specify a predicate to execute against each traversed record. If the predicate returns true, then the record is returned as result, otherwise false. it's common to create an anonymous class specifying the predicate at the fly
  • predicate(<predicate:OSQLPredicate>), to specify the predicate using the SQL syntax.

In the traverse command context iContext you can read/put any variable. Traverse command updates these variables:

  • depth, as the current depth of nesting
  • path, as the string representation of the current path. You can also display it. Example: select $path from (traverse * from V)
  • stack, as the List of operation in the stack. Use it to access to the history of the traversal. It's a List<OTraverseAbstractProcess<?>> where process implementations are:
  • OTraverseRecordSetProcess, usually the first one it's the base target of traverse
  • OTraverseRecordProcess, represent a traversed record
  • OTraverseFieldProcess, represent a traversal through a record's field
  • OTraverseMultiValueProcess, use on fields that are multivalue: arrays, collections and maps
  • history, as the set of records traversed as a Set<ORID>.

Example using an anonymous OCommandPredicate as predicate

    for (OIdentifiable id : new OTraverse()
                  .field("in").field("out")
                  .target( database.browseClass("Movie").iterator() )
                  .predicate(new OCommandPredicate() {
    
        public boolean evaluate(ORecord<?> iRecord, OCommandContext iContext) {
          return ((Integer) iContext.getVariable("depth")) <= 5;
        }
      })) {
    
      System.out.println(id);
    }

Example using the OSQLPredicate as predicate

    for (OIdentifiable id : new OTraverse()
                  .field("in").field("out")
                  .target(database.browseClass("Movie").iterator())
                  .predicate( new OSQLPredicate("$depth <= 5"))) {
    
      System.out.println(id);
    }

= Other example

OTraverse gets any Iterable, Iterator and Single/Multi OIdentifiable. There's also the limit() clause. To specify multiple fields use fields(). Full example:

    for (OIdentifiable id : new OTraverse()
                  .target(new ORecordId("#6:0"), new ORecordId("#6:1"))
                  .fields("out", "int")
                  .limit(100)
                  .predicate( new OSQLPredicate("$depth <= 10"))) {
    
      System.out.println( id);
    }
Clone this wiki locally