Skip to content

Commit 9d7237f

Browse files
committed
Mid-traversal steps and union as a start
1 parent acaebfc commit 9d7237f

File tree

2 files changed

+102
-6
lines changed

2 files changed

+102
-6
lines changed

book/Section-Beyond-Basic-Queries.adoc

+5-5
Original file line numberDiff line numberDiff line change
@@ -4947,11 +4947,11 @@ tree.getObjectsAtDepth(2)
49474947
We will see the Tree API used again in the "<<btree>>" section later on.
49484948

49494949
[[subgraph]]
4950-
Creating a sub graph
4951-
~~~~~~~~~~~~~~~~~~~~
4950+
Creating a subgraph
4951+
~~~~~~~~~~~~~~~~~~~
49524952

49534953
Using Gremlin, you can create a subgraph which is a subset of the vertices and edges
4954-
in a larger graph you are working with. Once created, to work with a sub graph, you
4954+
in a larger graph you are working with. Once created, to work with a subgraph, you
49554955
create a traversal source object specific to that new graph and distinct from the one
49564956
being used to process the main graph. Subgraphs are created using the 'subgraph'
49574957
traversal step. Note that 'subgraph' works with edges (not vertices) and adds both
@@ -5056,7 +5056,7 @@ As before we can now work with the newly created subgraph.
50565056
[source,groovy]
50575057
----
50585058
// Get a traversal source object so that we can traverse
5059-
// the newly created sub graph.
5059+
// the newly created subgraph.
50605060
sgt = subg.traversal()
50615061
50625062
// What sort of vertices ended up in the subgraph?
@@ -5076,7 +5076,7 @@ multiple 'subgraph' steps.
50765076

50775077
[source,groovy]
50785078
----
5079-
// Create a sub graph only of airports in Europe and routes between those airports
5079+
// Create a subgraph only of airports in Europe and routes between those airports
50805080
50815081
subg = g.V().hasLabel('continent').has('code','EU').
50825082
outE('contains').subgraph('eu-air-routes').inV().as('a').

book/Section-Writing-Gremlin-Queries.adoc

+97-1
Original file line numberDiff line numberDiff line change
@@ -362,7 +362,7 @@ g.V().has('region')
362362
g.V().hasNot('region')
363363
364364
// The above is shorthand for
365-
g.V().not(has('region'))
365+
g.V().not(has('region'))
366366
----
367367

368368
[[count]]
@@ -5555,6 +5555,30 @@ g.V().has('airport','code','DFW').
55555555
[v[8]:221]]
55565556
----
55575557

5558+
Another aspect of 'union' to consider is that it can be used as a start step, which
5559+
means that you can essentially start a traversal from multiple points.
5560+
5561+
[source,groovy]
5562+
----
5563+
g.union(V().has('airport','code','DFW'),
5564+
V().has('airport','code','IAD')).
5565+
values('code').fold()
5566+
5567+
[DFW,IAD]
5568+
----
5569+
5570+
The prior example is meant as a demonstration. Typically, it will make more sense to
5571+
use a `union` this way when you have disparate sets of things you wish to combine to
5572+
a single traversal stream. In this case, you can see that we're just filtering
5573+
airports by their code. It's more likely that you would write that query as follows:
5574+
5575+
[source,groovy]
5576+
----
5577+
g.V().has('airport','code',within('DFW','IAD')).values('code').fold()
5578+
5579+
[DFW,IAD]
5580+
----
5581+
55585582
[[unionidentity]]
55595583
Introducing the 'identity' step
55605584
^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^^
@@ -6289,6 +6313,78 @@ g.V().has('code','AUS').
62896313
Dallas
62906314
----
62916315

6316+
[[midtraversalve]]
6317+
Using 'V' and 'E' mid-traversal
6318+
~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~~
6319+
6320+
All of our queries so far have used the 'V' and 'E' step to help start a traversal.
6321+
We've seen them at the start of a traversal spawned from 'g' and as a spawn of
6322+
anonymous child traversals with '__'. As start steps, they inject the objects that
6323+
flow through the traversal stream. While they are commonly used at the start of a
6324+
traversal, they can also be used in the middle of one.
6325+
6326+
[source,groovy]
6327+
----
6328+
g.V().has('code','AUS').V().has('code', within('DFW','IAD')).
6329+
values('code').fold()
6330+
6331+
6332+
[DFW,IAD]
6333+
----
6334+
6335+
In the above example, we first find the '"AUS"' airport vertex. That start traverser
6336+
triggers the mid-traversal 'V' that does a lookup for '"DFW"' and '"IAD"'. As a
6337+
result, we get two vertices as the result. Each traverser that flows to the
6338+
mid-traversal 'V' will issue that same filter, therefore, if we return two starting
6339+
vertices, we will get some duplication and 4 results as follows:
6340+
6341+
[source,groovy]
6342+
----
6343+
g.V().has('code',,within('SFO','AUS')).V().has('code', within('DFW','IAD')).
6344+
values('code').fold()
6345+
6346+
6347+
[DFW,IAD,DFW,IAD]
6348+
----
6349+
6350+
Mid-traversal 'V' has a similar counterpart for edges with 'E' and behaves in the
6351+
same fashion.
6352+
6353+
[source,groovy]
6354+
----
6355+
g.V().has('code',within('SFO','AUS')).E(3640)
6356+
6357+
e[3640][1-route->24]
6358+
e[3640][1-route->24]
6359+
----
6360+
6361+
You typically find use for these mid-traversal steps when you do a mutation or
6362+
side-effect first, but then want to traverse a wholly new path after that. Let's look
6363+
at a more advanced example to demonstrate this. First, let's find out what the
6364+
longest route is leaving '"IAD"':
6365+
6366+
[source,groovy]
6367+
----
6368+
g.V().has('code','IAD').outE('route').
6369+
values('dist').max()
6370+
6371+
7487
6372+
----
6373+
6374+
Now, let's extend on that query with mid-traversal 'V' to use that result to find
6375+
"all routes from DFW that are longer than the longest route from IAD":
6376+
6377+
6378+
[source,groovy]
6379+
----
6380+
g.V().has('code','IAD').outE('route').
6381+
values('dist').max().as('iad').
6382+
V().has('code','DFW').outE('route').as('r').
6383+
where(values('dist').where(gt('iad'))).
6384+
inV().values('code').fold()
6385+
6386+
[SYD,DXB,HKG,DOH,AUH]
6387+
----
62926388

62936389
[[otherv]]
62946390
Other ways to explore vertices and edges using 'both', 'bothE', 'bothV' and 'otherV'

0 commit comments

Comments
 (0)