neo4j-cypher 1.0.0.rc1
This diff represents the content of publicly available package versions that have been released to one of the supported registries. The information contained in this diff is provided for informational purposes only and reflects changes between package versions as they appear in their respective public registries.
- data/Gemfile +15 -0
- data/README.rdoc +71 -0
- data/lib/neo4j-cypher.rb +47 -0
- data/lib/neo4j-cypher/argument.rb +39 -0
- data/lib/neo4j-cypher/clause.rb +76 -0
- data/lib/neo4j-cypher/clause_list.rb +101 -0
- data/lib/neo4j-cypher/context.rb +411 -0
- data/lib/neo4j-cypher/create.rb +99 -0
- data/lib/neo4j-cypher/match.rb +343 -0
- data/lib/neo4j-cypher/mixins.rb +43 -0
- data/lib/neo4j-cypher/node_var.rb +65 -0
- data/lib/neo4j-cypher/operator.rb +130 -0
- data/lib/neo4j-cypher/predicate.rb +64 -0
- data/lib/neo4j-cypher/property.rb +106 -0
- data/lib/neo4j-cypher/rel_var.rb +128 -0
- data/lib/neo4j-cypher/result.rb +43 -0
- data/lib/neo4j-cypher/result_wrapper.rb +47 -0
- data/lib/neo4j-cypher/return.rb +124 -0
- data/lib/neo4j-cypher/root.rb +182 -0
- data/lib/neo4j-cypher/start.rb +88 -0
- data/lib/neo4j-cypher/version.rb +5 -0
- data/lib/neo4j-cypher/where.rb +16 -0
- data/lib/neo4j-cypher/with.rb +41 -0
- data/neo4j-cypher.gemspec +26 -0
- metadata +81 -0
data/Gemfile
ADDED
data/README.rdoc
ADDED
@@ -0,0 +1,71 @@
|
|
1
|
+
== neo4j-cypher {<img src="https://secure.travis-ci.org/andreasronge/neo4j-cypher.png" />}[http://travis-ci.org/andreasronge/neo4j-cypher]
|
2
|
+
|
3
|
+
A Ruby DSL for the Neo4j Cypher query language for both MRI and JRuby.
|
4
|
+
The JRuby neo4j-core gem's cypher dsl has been moved to this gem.
|
5
|
+
|
6
|
+
=== Docs
|
7
|
+
|
8
|
+
See {Neo4j Wiki Cypher}[https://github.com/andreasronge/neo4j/wiki/Neo4j::Core-Cypher]
|
9
|
+
|
10
|
+
See the rspecs ! (> 99% test coverage)
|
11
|
+
|
12
|
+
|
13
|
+
=== Random Examples
|
14
|
+
|
15
|
+
Notice the last value returned is the return value (if possible).
|
16
|
+
Matching relationships can be done with operators: <,>, -, and <=> or with the both, incoming and outgoing methods.
|
17
|
+
|
18
|
+
"START v1=node(3) MATCH v2 = (v1)-[:`r`]->(x) RETURN v2"
|
19
|
+
|
20
|
+
Neo4j::Cypher.query do
|
21
|
+
node(3) > :r > :x
|
22
|
+
end
|
23
|
+
|
24
|
+
|
25
|
+
"START v2=node(1) MATCH (v2)-[v1:`knows`]->(other) WHERE v1.since > 1994 and other.name = "foo" RETURN other"
|
26
|
+
|
27
|
+
Neo4j::Cypher.query do
|
28
|
+
node(1) > (rel(:knows)[:since] > 1994) > (node(:other)[:name] == 'foo'); :other
|
29
|
+
end
|
30
|
+
|
31
|
+
|
32
|
+
"START v2=node(1) MATCH (v2)-[v1:`friends`]->(v3) WHERE (v1.since = 1994) RETURN v3"
|
33
|
+
|
34
|
+
Neo4j::Cypher.query do
|
35
|
+
node(1).outgoing(rel(:friends).where{|r| r[:since] == 1994})
|
36
|
+
end
|
37
|
+
|
38
|
+
|
39
|
+
"START v1=node(2,3,4,1) RETURN count(v1.property?"
|
40
|
+
|
41
|
+
Neo4j::Cypher.query do
|
42
|
+
node(2, 3, 4, 1)[:property?].count
|
43
|
+
end
|
44
|
+
|
45
|
+
"START v1=node(42) MATCH (v1)-[:`favorite`]->(stuff)<-[:`favorite`]-(person) WHERE not((v1)-[:`friend`]-(person)) RETURN person.name,count(stuff) ORDER BY count(stuff) DESC"
|
46
|
+
|
47
|
+
Neo4j::Cypher.query do
|
48
|
+
node(42).where_not { |m| m - :friend - :person } > :favorite > :stuff < :favorite < :person
|
49
|
+
ret(node(:person)[:name], count(:stuff).desc)
|
50
|
+
end
|
51
|
+
|
52
|
+
|
53
|
+
== Complex Example
|
54
|
+
|
55
|
+
"START n=node(42) MATCH (n)-[r]->(m) WITH n,collect(type(r)) as out_types,collect(m) as outgoing MATCH (n)<-[r]-(m) RETURN n,outgoing,out_types,collect(m) as incoming,collect(type(r)) as in_types"
|
56
|
+
|
57
|
+
Neo4j::Cypher.query do
|
58
|
+
n = node(42).as(:n)
|
59
|
+
r = rel('r')
|
60
|
+
m = node(:m)
|
61
|
+
rel_types = r.rel_type.collect
|
62
|
+
end_nodes = m.collect
|
63
|
+
|
64
|
+
n.with_match(rel_types.as(:out_types), end_nodes.as(:outgoing)) { |n, _, _| n < r < m } > r > m
|
65
|
+
|
66
|
+
ret([n,
|
67
|
+
:outgoing,
|
68
|
+
:out_types,
|
69
|
+
end_nodes.as(:incoming),
|
70
|
+
rel_types.as(:in_types)])
|
71
|
+
end
|
data/lib/neo4j-cypher.rb
ADDED
@@ -0,0 +1,47 @@
|
|
1
|
+
require 'neo4j-cypher/version'
|
2
|
+
|
3
|
+
require 'neo4j-cypher/context'
|
4
|
+
require 'neo4j-cypher/mixins'
|
5
|
+
require 'neo4j-cypher/clause'
|
6
|
+
require 'neo4j-cypher/clause_list'
|
7
|
+
require 'neo4j-cypher/argument'
|
8
|
+
require 'neo4j-cypher/root'
|
9
|
+
require 'neo4j-cypher/start'
|
10
|
+
require 'neo4j-cypher/create'
|
11
|
+
require 'neo4j-cypher/match'
|
12
|
+
require 'neo4j-cypher/return'
|
13
|
+
require 'neo4j-cypher/node_var'
|
14
|
+
require 'neo4j-cypher/rel_var'
|
15
|
+
require 'neo4j-cypher/property'
|
16
|
+
require 'neo4j-cypher/predicate'
|
17
|
+
require 'neo4j-cypher/with'
|
18
|
+
require 'neo4j-cypher/operator'
|
19
|
+
require 'neo4j-cypher/where'
|
20
|
+
require 'neo4j-cypher/result_wrapper'
|
21
|
+
require 'neo4j-cypher/result'
|
22
|
+
|
23
|
+
module Neo4j
|
24
|
+
module Cypher
|
25
|
+
|
26
|
+
# Creates a Cypher DSL query.
|
27
|
+
# To create a new cypher query you must initialize it either an String or a Block.
|
28
|
+
#
|
29
|
+
# @example <tt>START n0=node(3) MATCH (n0)--(x) RETURN x</tt> same as
|
30
|
+
# Cypher.query { start n = node(3); match n <=> :x; ret :x }.to_s
|
31
|
+
#
|
32
|
+
# @example <tt>START n0=node(3) MATCH (n0)-[:`r`]->(x) RETURN r</tt> same as
|
33
|
+
# Cypher.query { node(3) > :r > :x; :r }
|
34
|
+
#
|
35
|
+
# @example <tt>START n0=node(3) MATCH (n0)-->(x) RETURN x</tt> same as
|
36
|
+
# Cypher.query { node(3) >> :x; :x }
|
37
|
+
#
|
38
|
+
# @param args the argument for the dsl_block
|
39
|
+
# @yield the block which will be evaluated in the context of this object in order to create an Cypher Query string
|
40
|
+
# @yieldreturn [Return, Object] If the return is not an instance of Return it will be converted it to a Return object (if possible).
|
41
|
+
# @return [Cypher::Result]
|
42
|
+
def self.query(*args, &dsl_block)
|
43
|
+
Result.new(*args, &dsl_block)
|
44
|
+
end
|
45
|
+
|
46
|
+
end
|
47
|
+
end
|
@@ -0,0 +1,39 @@
|
|
1
|
+
module Neo4j
|
2
|
+
module Cypher
|
3
|
+
|
4
|
+
class Argument
|
5
|
+
include Referenceable
|
6
|
+
include Clause
|
7
|
+
|
8
|
+
def initialize(clause_list, expr, var_name)
|
9
|
+
super(clause_list, :argument, EvalContext)
|
10
|
+
var_name ||= self.var_name
|
11
|
+
@expr = var_name
|
12
|
+
as_alias(var_name)
|
13
|
+
@return_value = (expr != var_name.to_s) ? "#{expr} as #{var_name}" : expr
|
14
|
+
end
|
15
|
+
|
16
|
+
def return_value
|
17
|
+
@return_value
|
18
|
+
end
|
19
|
+
|
20
|
+
def self.new_arg_from_clause(clause)
|
21
|
+
Argument.new(clause.clause_list, clause.return_value, clause.as_alias? && clause.var_name)
|
22
|
+
end
|
23
|
+
|
24
|
+
def self.new_arg_from_string(string, clause_list)
|
25
|
+
Argument.new(clause_list, string.to_s, string)
|
26
|
+
end
|
27
|
+
|
28
|
+
class EvalContext
|
29
|
+
include Context
|
30
|
+
include Comparable
|
31
|
+
include MathOperator
|
32
|
+
include MathFunctions
|
33
|
+
include PredicateMethods
|
34
|
+
include Aggregate
|
35
|
+
end
|
36
|
+
|
37
|
+
end
|
38
|
+
end
|
39
|
+
end
|
@@ -0,0 +1,76 @@
|
|
1
|
+
module Neo4j
|
2
|
+
module Cypher
|
3
|
+
|
4
|
+
# Responsible for order of the clauses
|
5
|
+
# Does expect a #clause method when included
|
6
|
+
module Clause
|
7
|
+
|
8
|
+
ORDER = [:start, :match, :create, :where, :with, :foreach, :set, :delete, :return, :order_by, :skip, :limit]
|
9
|
+
NAME = {:start => 'START', :create => 'CREATE', :match => 'MATCH', :where => "WHERE", :with => 'WITH',
|
10
|
+
:return => 'RETURN', :order_by => 'ORDER BY', :skip => 'SKIP', :limit => 'LIMIT', :set => 'SET',
|
11
|
+
:delete => 'DELETE', :foreach => 'FOREACH'}
|
12
|
+
|
13
|
+
attr_accessor :clause_type, :clause_list, :eval_context, :expr, :insert_order
|
14
|
+
|
15
|
+
def initialize(clause_list, clause_type, eval_context = Context::Empty)
|
16
|
+
@clause_type = clause_type
|
17
|
+
@clause_list = clause_list
|
18
|
+
if eval_context.is_a?(Class)
|
19
|
+
@eval_context = eval_context.new(self)
|
20
|
+
else
|
21
|
+
@eval_context = eval_context
|
22
|
+
end
|
23
|
+
self.insert_order = 0
|
24
|
+
clause_list.insert(self)
|
25
|
+
end
|
26
|
+
|
27
|
+
def <=>(other)
|
28
|
+
clause_position == other.clause_position ? insert_order <=> other.insert_order : clause_position <=> other.clause_position
|
29
|
+
end
|
30
|
+
|
31
|
+
def clause_position
|
32
|
+
valid_clause?
|
33
|
+
ORDER.find_index(clause_type)
|
34
|
+
end
|
35
|
+
|
36
|
+
def valid_clause?
|
37
|
+
raise "Unknown clause_type '#{clause_type}' on #{self}" unless ORDER.include?(clause_type)
|
38
|
+
end
|
39
|
+
|
40
|
+
def separator
|
41
|
+
','
|
42
|
+
end
|
43
|
+
|
44
|
+
def match_value=(mv)
|
45
|
+
@match_value = mv
|
46
|
+
end
|
47
|
+
|
48
|
+
def match_value
|
49
|
+
@match_value || expr || var_name
|
50
|
+
end
|
51
|
+
|
52
|
+
# Used in return clause to generate the last part of the return cypher clause string
|
53
|
+
def return_value
|
54
|
+
var_name
|
55
|
+
end
|
56
|
+
|
57
|
+
def prefix
|
58
|
+
NAME[clause_type]
|
59
|
+
end
|
60
|
+
|
61
|
+
def create_clause_args_for(args)
|
62
|
+
args.map do |arg|
|
63
|
+
case arg
|
64
|
+
when Neo4j::Cypher::ReturnItem::EvalContext, Neo4j::Cypher::Property::EvalContext
|
65
|
+
Argument.new_arg_from_clause(arg.clause)
|
66
|
+
when String, Symbol
|
67
|
+
Argument.new_arg_from_string(arg, clause_list)
|
68
|
+
else
|
69
|
+
arg.clause
|
70
|
+
end
|
71
|
+
end
|
72
|
+
end
|
73
|
+
end
|
74
|
+
|
75
|
+
end
|
76
|
+
end
|
@@ -0,0 +1,101 @@
|
|
1
|
+
module Neo4j
|
2
|
+
module Cypher
|
3
|
+
|
4
|
+
class ClauseList
|
5
|
+
attr_accessor :variables
|
6
|
+
include Enumerable
|
7
|
+
|
8
|
+
def initialize(variables = [])
|
9
|
+
@variables = variables
|
10
|
+
@clause_list = []
|
11
|
+
@insert_order = 0
|
12
|
+
end
|
13
|
+
|
14
|
+
def empty?
|
15
|
+
!first
|
16
|
+
end
|
17
|
+
|
18
|
+
def include?(clause_type)
|
19
|
+
@clause_list.find { |c| c.clause_type == clause_type }
|
20
|
+
end
|
21
|
+
|
22
|
+
def each
|
23
|
+
@clause_list.each { |c| yield c }
|
24
|
+
end
|
25
|
+
|
26
|
+
def push
|
27
|
+
raise "Only support stack of depth 2" if @old_clause_list
|
28
|
+
@old_clause_list = @clause_list
|
29
|
+
@clause_list = []
|
30
|
+
self
|
31
|
+
end
|
32
|
+
|
33
|
+
def pop
|
34
|
+
@clause_list = @old_clause_list
|
35
|
+
@clause_list.sort!
|
36
|
+
@old_clause_list = nil
|
37
|
+
self
|
38
|
+
end
|
39
|
+
|
40
|
+
def insert(clause)
|
41
|
+
ctype = clause.clause_type
|
42
|
+
|
43
|
+
if Clause::ORDER.include?(ctype)
|
44
|
+
# which list should we add the cluase to, the root or the sub list ?
|
45
|
+
# ALl the start and return clauses should move to the clause_list
|
46
|
+
c = (@old_clause_list && (ctype == :start || ctype == :return)) ? @old_clause_list : @clause_list
|
47
|
+
c << clause
|
48
|
+
@insert_order += 1
|
49
|
+
clause.insert_order = @insert_order
|
50
|
+
c.sort!
|
51
|
+
end
|
52
|
+
self
|
53
|
+
end
|
54
|
+
|
55
|
+
def last
|
56
|
+
@clause_list.last
|
57
|
+
end
|
58
|
+
|
59
|
+
def delete(clause_or_context)
|
60
|
+
c = clause_or_context.respond_to?(:clause) ? clause_or_context.clause : clause_or_context
|
61
|
+
@clause_list.delete(c)
|
62
|
+
end
|
63
|
+
|
64
|
+
#def debug
|
65
|
+
# puts "ClauseList id: #{object_id}, vars: #{variables.size}"
|
66
|
+
# @clause_list.each_with_index { |c, i| puts " #{i} #{c.clause_type.inspect}, #{c.class} id: #{c.object_id} order #{c.insert_order}" }
|
67
|
+
#end
|
68
|
+
|
69
|
+
def create_variable(var)
|
70
|
+
raise "Already included #{var}" if @variables.include?(var)
|
71
|
+
@variables << var
|
72
|
+
"v#{@variables.size}".to_sym
|
73
|
+
end
|
74
|
+
|
75
|
+
def group_by_clause
|
76
|
+
prev_clause = nil
|
77
|
+
inject([]) do |memo, clause|
|
78
|
+
memo << [] if clause.clause_type != prev_clause
|
79
|
+
prev_clause = clause.clause_type
|
80
|
+
memo.last << clause
|
81
|
+
memo
|
82
|
+
end
|
83
|
+
end
|
84
|
+
|
85
|
+
def join_group(list)
|
86
|
+
list.map { |c| c.to_cypher }.join(list.first.separator)
|
87
|
+
end
|
88
|
+
|
89
|
+
def to_cypher
|
90
|
+
# Sub lists, like in with clause should not have a clause prefix like WHERE or MATCH
|
91
|
+
group_by_clause.map { |list| "#{prefix(list)}#{join_group(list)}" }.join(' ')
|
92
|
+
end
|
93
|
+
|
94
|
+
def prefix(list)
|
95
|
+
@old_clause_list && ![:set, :delete, :create].include?(list.first.clause_type) ? '' : "#{list.first.prefix} "
|
96
|
+
end
|
97
|
+
|
98
|
+
end
|
99
|
+
end
|
100
|
+
|
101
|
+
end
|
@@ -0,0 +1,411 @@
|
|
1
|
+
module Neo4j
|
2
|
+
module Cypher
|
3
|
+
module Context
|
4
|
+
# @return [Neo4j::Cypher:Clause]
|
5
|
+
attr_accessor :clause
|
6
|
+
|
7
|
+
# @param [Neo4j::Cypher:Clause] clause the clause for this eval context
|
8
|
+
def initialize(clause)
|
9
|
+
@clause = clause
|
10
|
+
|
11
|
+
end
|
12
|
+
|
13
|
+
# @return [Array<Neo4j::Cypher:Clause>] the sorted clause list
|
14
|
+
def clause_list
|
15
|
+
@clause.clause_list
|
16
|
+
end
|
17
|
+
|
18
|
+
# Used for eval context for a clause which does not allow any more method chaining.
|
19
|
+
class Empty
|
20
|
+
include Context
|
21
|
+
end
|
22
|
+
|
23
|
+
module Alias
|
24
|
+
# Typically used in a WITH statement for a count.as(:stuff) or node(42).as(:foo)
|
25
|
+
def as(name)
|
26
|
+
clause.as_alias(name)
|
27
|
+
self
|
28
|
+
end
|
29
|
+
end
|
30
|
+
|
31
|
+
module MathFunctions
|
32
|
+
def abs(value=nil)
|
33
|
+
_add_math_func(:abs, value)
|
34
|
+
end
|
35
|
+
|
36
|
+
def sqrt(value=nil)
|
37
|
+
_add_math_func(:sqrt, value)
|
38
|
+
end
|
39
|
+
|
40
|
+
def round(value=nil)
|
41
|
+
_add_math_func(:round, value)
|
42
|
+
end
|
43
|
+
|
44
|
+
def sign(value=nil)
|
45
|
+
_add_math_func(:sign, value)
|
46
|
+
end
|
47
|
+
|
48
|
+
# @private
|
49
|
+
def _add_math_func(name, value)
|
50
|
+
value ||= clause.to_cypher
|
51
|
+
clause_list.delete(clause)
|
52
|
+
ReturnItem.new(clause_list, "#{name}(#{value})").eval_context
|
53
|
+
end
|
54
|
+
end
|
55
|
+
|
56
|
+
module MathOperator
|
57
|
+
def -(other)
|
58
|
+
Operator.new(clause_list, clause, other, '-').eval_context
|
59
|
+
end
|
60
|
+
|
61
|
+
def +(other)
|
62
|
+
Operator.new(clause_list, clause, other, '+').eval_context
|
63
|
+
end
|
64
|
+
end
|
65
|
+
|
66
|
+
module Comparable
|
67
|
+
def <(other)
|
68
|
+
Operator.new(clause_list, clause, other, '<').eval_context
|
69
|
+
end
|
70
|
+
|
71
|
+
def <=(other)
|
72
|
+
Operator.new(clause_list, clause, other, '<=').eval_context
|
73
|
+
end
|
74
|
+
|
75
|
+
def =~(other)
|
76
|
+
Operator.new(clause_list, clause, other, '=~').eval_context
|
77
|
+
end
|
78
|
+
|
79
|
+
def >(other)
|
80
|
+
Operator.new(clause_list, clause, other, '>').eval_context
|
81
|
+
end
|
82
|
+
|
83
|
+
def >=(other)
|
84
|
+
Operator.new(clause_list, clause, other, '>=').eval_context
|
85
|
+
end
|
86
|
+
|
87
|
+
## Only in 1.9
|
88
|
+
if RUBY_VERSION > "1.9.0"
|
89
|
+
eval %{
|
90
|
+
def !=(other)
|
91
|
+
Operator.new(clause_list, clause, other, "<>").eval_context
|
92
|
+
end }
|
93
|
+
end
|
94
|
+
|
95
|
+
def ==(other)
|
96
|
+
Operator.new(clause_list, clause, other, "=").eval_context
|
97
|
+
end
|
98
|
+
end
|
99
|
+
|
100
|
+
module PredicateMethods
|
101
|
+
def all?(&block)
|
102
|
+
self.respond_to?(:iterable)
|
103
|
+
Predicate.new(clause_list, :op => 'all', :clause => :where, :input => input, :iterable => iterable, :predicate_block => block).eval_context
|
104
|
+
end
|
105
|
+
|
106
|
+
def extract(&block)
|
107
|
+
Predicate.new(clause_list, :op => 'extract', :clause => :return_item, :input => input, :iterable => iterable, :predicate_block => block).eval_context
|
108
|
+
end
|
109
|
+
|
110
|
+
def filter(&block)
|
111
|
+
Predicate.new(clause_list, :op => 'filter', :clause => :return_item, :input => input, :iterable => iterable, :predicate_block => block).eval_context
|
112
|
+
end
|
113
|
+
|
114
|
+
def any?(&block)
|
115
|
+
Predicate.new(clause_list, :op => 'any', :clause => :where, :input => input, :iterable => iterable, :predicate_block => block).eval_context
|
116
|
+
end
|
117
|
+
|
118
|
+
def none?(&block)
|
119
|
+
Predicate.new(clause_list, :op => 'none', :clause => :where, :input => input, :iterable => iterable, :predicate_block => block).eval_context
|
120
|
+
end
|
121
|
+
|
122
|
+
def single?(&block)
|
123
|
+
Predicate.new(clause_list, :op => 'single', :clause => :where, :input => input, :iterable => iterable, :predicate_block => block).eval_context
|
124
|
+
end
|
125
|
+
|
126
|
+
def foreach(&block)
|
127
|
+
Predicate.new(clause_list, :op => '', :clause => :foreach, :input => input, :iterable => iterable, :predicate_block => block, :separator => ' FOREACH ').eval_context
|
128
|
+
input.eval_context
|
129
|
+
end
|
130
|
+
end
|
131
|
+
|
132
|
+
module Returnable
|
133
|
+
# Specifies a return statement.
|
134
|
+
# Notice that this is not needed, since the last value of the DSL block will be converted into one or more
|
135
|
+
# return statements.
|
136
|
+
# @param [Symbol, #var_name] returns a list of variables we want to return
|
137
|
+
# @return [ReturnItem]
|
138
|
+
def ret(*returns, &block)
|
139
|
+
options = returns.last.is_a?(Hash) ? returns.pop : {}
|
140
|
+
returns = [self] if returns.empty? # return self unless not specified what to return
|
141
|
+
returns = [RootClause::EvalContext.new(self).instance_exec(self, &block)].flatten if block
|
142
|
+
r = Return.new(clause_list, returns, options, &block).eval_context
|
143
|
+
(self.is_a?(RootClause::EvalContext)) ? r : self
|
144
|
+
end
|
145
|
+
|
146
|
+
end
|
147
|
+
|
148
|
+
module Sortable
|
149
|
+
def _sort_args(prop)
|
150
|
+
return self if prop.nil?
|
151
|
+
prop.is_a?(Symbol) ? Property.new(clause, prop).eval_context : prop
|
152
|
+
end
|
153
|
+
|
154
|
+
def asc(*props)
|
155
|
+
@return_item ||= ReturnItem.new(clause_list, self).eval_context
|
156
|
+
@return_item.asc(_sort_args(props.first))
|
157
|
+
end
|
158
|
+
|
159
|
+
def desc(*props)
|
160
|
+
@return_item ||= ReturnItem.new(clause_list, self).eval_context
|
161
|
+
@return_item.desc(_sort_args(props.first))
|
162
|
+
end
|
163
|
+
|
164
|
+
end
|
165
|
+
|
166
|
+
module ReturnOrder
|
167
|
+
def _sort_args(props)
|
168
|
+
return [self] if props.empty?
|
169
|
+
props.map { |p| p.is_a?(Symbol) ? Property.new(clause, p).eval_context : p }
|
170
|
+
end
|
171
|
+
|
172
|
+
# Specifies an <tt>ORDER BY</tt> cypher query
|
173
|
+
# @param [Property] props the properties which should be sorted
|
174
|
+
# @return self
|
175
|
+
def asc(*props)
|
176
|
+
@order_by ||= OrderBy.new(clause_list, self)
|
177
|
+
clause_list.delete(props.first)
|
178
|
+
@order_by.asc(_sort_args(props))
|
179
|
+
self
|
180
|
+
end
|
181
|
+
|
182
|
+
# Specifies an <tt>ORDER BY</tt> cypher query
|
183
|
+
# @param [Property] props the properties which should be sorted
|
184
|
+
# @return self
|
185
|
+
def desc(*props)
|
186
|
+
@order_by ||= OrderBy.new(clause_list, self)
|
187
|
+
clause_list.delete(props.first)
|
188
|
+
@order_by.desc(_sort_args(props))
|
189
|
+
self
|
190
|
+
end
|
191
|
+
|
192
|
+
# Creates a <tt>SKIP</tt> cypher clause
|
193
|
+
# @param [Fixnum] val the number of entries to skip
|
194
|
+
# @return self
|
195
|
+
def skip(val)
|
196
|
+
Skip.new(clause_list, val, self)
|
197
|
+
self
|
198
|
+
end
|
199
|
+
|
200
|
+
# Creates a <tt>LIMIT</tt> cypher clause
|
201
|
+
# @param [Fixnum] val the number of entries to limit
|
202
|
+
# @return self
|
203
|
+
def limit(val)
|
204
|
+
Limit.new(clause_list, val, self)
|
205
|
+
self
|
206
|
+
end
|
207
|
+
|
208
|
+
end
|
209
|
+
|
210
|
+
module Aggregate
|
211
|
+
def distinct
|
212
|
+
ReturnItem.new(clause_list, "distinct(#{clause.return_value})").eval_context
|
213
|
+
end
|
214
|
+
|
215
|
+
def count
|
216
|
+
ReturnItem.new(clause_list, "count(#{clause.return_value})").eval_context
|
217
|
+
end
|
218
|
+
|
219
|
+
def sum
|
220
|
+
ReturnItem.new(clause_list, "sum(#{clause.return_value})").eval_context
|
221
|
+
end
|
222
|
+
|
223
|
+
def avg
|
224
|
+
ReturnItem.new(clause_list, "avg(#{clause.return_value})").eval_context
|
225
|
+
end
|
226
|
+
|
227
|
+
def min
|
228
|
+
ReturnItem.new(clause_list, "min(#{clause.return_value})").eval_context
|
229
|
+
end
|
230
|
+
|
231
|
+
def max
|
232
|
+
ReturnItem.new(clause_list, "max(#{clause.return_value})").eval_context
|
233
|
+
end
|
234
|
+
|
235
|
+
def collect
|
236
|
+
ReturnItem.new(clause_list, "collect(#{clause.return_value})").eval_context
|
237
|
+
end
|
238
|
+
|
239
|
+
|
240
|
+
def last
|
241
|
+
ReturnItem.new(clause_list, "last(#{clause.return_value})").eval_context
|
242
|
+
end
|
243
|
+
|
244
|
+
def tail
|
245
|
+
ReturnItem.new(clause_list, "tail(#{clause.return_value})").eval_context
|
246
|
+
end
|
247
|
+
|
248
|
+
def head
|
249
|
+
ReturnItem.new(clause_list, "head(#{clause.return_value})").eval_context
|
250
|
+
end
|
251
|
+
|
252
|
+
end
|
253
|
+
|
254
|
+
module Variable
|
255
|
+
def where(&block)
|
256
|
+
x = block.call(self)
|
257
|
+
clause_list.delete(x)
|
258
|
+
Operator.new(clause_list, x.clause, nil, "").unary!
|
259
|
+
self
|
260
|
+
end
|
261
|
+
|
262
|
+
def where_not(&block)
|
263
|
+
x = block.call(self)
|
264
|
+
clause_list.delete(x)
|
265
|
+
Operator.new(clause_list, x.clause, nil, "not").unary!
|
266
|
+
self
|
267
|
+
end
|
268
|
+
|
269
|
+
def [](prop_name)
|
270
|
+
Property.new(clause, prop_name).eval_context
|
271
|
+
end
|
272
|
+
|
273
|
+
# generates a <tt>ID</tt> cypher fragment.
|
274
|
+
def neo_id
|
275
|
+
Property.new(clause, 'ID').to_function!
|
276
|
+
end
|
277
|
+
|
278
|
+
# generates a <tt>has</tt> cypher fragment.
|
279
|
+
def property?(p)
|
280
|
+
p = Property.new(clause, p)
|
281
|
+
Operator.new(clause_list, p, nil, "has").unary!
|
282
|
+
end
|
283
|
+
|
284
|
+
def []=(p, value)
|
285
|
+
left = Property.new(clause, p).eval_context
|
286
|
+
Operator.new(clause_list, left, value, "=", :set)
|
287
|
+
self
|
288
|
+
end
|
289
|
+
|
290
|
+
def del
|
291
|
+
Delete.new(clause_list, clause)
|
292
|
+
self
|
293
|
+
end
|
294
|
+
|
295
|
+
# Can be used instead of [_classname] == klass
|
296
|
+
def is_a?(klass)
|
297
|
+
return super if klass.class != Class || !klass.respond_to?(:_load_wrapper)
|
298
|
+
self[:_classname] == klass.to_s
|
299
|
+
end
|
300
|
+
end
|
301
|
+
|
302
|
+
module Matchable
|
303
|
+
|
304
|
+
def with(*args, &cypher_dsl)
|
305
|
+
With.new(clause_list, :where, self, *args, &cypher_dsl)
|
306
|
+
self
|
307
|
+
end
|
308
|
+
|
309
|
+
def with_match(*args, &cypher_dsl)
|
310
|
+
With.new(clause_list, :match, self, *args, &cypher_dsl)
|
311
|
+
self
|
312
|
+
end
|
313
|
+
|
314
|
+
def create_path(*args, &cypher_dsl)
|
315
|
+
CreatePath.new(clause_list, self, *args, &cypher_dsl)
|
316
|
+
self
|
317
|
+
end
|
318
|
+
|
319
|
+
def create_unique_path(*args, &cypher_dsl)
|
320
|
+
CreatePath.new(clause_list, self, *args, &cypher_dsl).unique!
|
321
|
+
self
|
322
|
+
end
|
323
|
+
|
324
|
+
|
325
|
+
# This operator means related to, without regard to type or direction.
|
326
|
+
# @param [Symbol, #var_name] other either a node (Symbol, #var_name)
|
327
|
+
# @return [MatchRelLeft, MatchNode]
|
328
|
+
def <=>(other)
|
329
|
+
MatchStart.new_match_node(clause, other, :both).eval_context
|
330
|
+
end
|
331
|
+
|
332
|
+
# This operator means outgoing related to
|
333
|
+
# @param [Symbol, #var_name, String] other the relationship
|
334
|
+
# @return [MatchRelLeft, MatchNode]
|
335
|
+
def >(other)
|
336
|
+
MatchStart.new(clause).new_match_rel(other).eval_context
|
337
|
+
end
|
338
|
+
|
339
|
+
# This operator means any direction related to
|
340
|
+
# @param (see #>)
|
341
|
+
# @return [MatchRelLeft, MatchNode]
|
342
|
+
def -(other)
|
343
|
+
MatchStart.new(clause).new_match_rel(other).eval_context
|
344
|
+
end
|
345
|
+
|
346
|
+
# This operator means incoming related to
|
347
|
+
# @param (see #>)
|
348
|
+
# @return [MatchRelLeft, MatchNode]
|
349
|
+
def <(other)
|
350
|
+
MatchStart.new(clause).new_match_rel(other).eval_context
|
351
|
+
end
|
352
|
+
|
353
|
+
# Outgoing relationship to other node
|
354
|
+
# @param [Symbol, #var_name] other either a node (Symbol, #var_name)
|
355
|
+
# @return [MatchRelLeft, MatchNode]
|
356
|
+
def >>(other)
|
357
|
+
MatchStart.new_match_node(clause, other, :outgoing).eval_context
|
358
|
+
end
|
359
|
+
|
360
|
+
# Incoming relationship to other node
|
361
|
+
# @param [Symbol, #var_name] other either a node (Symbol, #var_name)
|
362
|
+
# @return [MatchRelLeft, MatchNode]
|
363
|
+
def <<(other)
|
364
|
+
MatchStart.new_match_node(clause, other, :incoming).eval_context
|
365
|
+
end
|
366
|
+
|
367
|
+
def outgoing(*rel_types)
|
368
|
+
node = _get_or_create_node(rel_types)
|
369
|
+
MatchStart.new(clause).new_match_rels(rel_types).eval_context > node
|
370
|
+
node.eval_context
|
371
|
+
end
|
372
|
+
|
373
|
+
def _get_or_create_node(rel_types)
|
374
|
+
rel_types.last.kind_of?(Matchable) ? rel_types.pop.clause : NodeVar.new(clause.clause_list)
|
375
|
+
end
|
376
|
+
|
377
|
+
def outgoing?(*rel_types)
|
378
|
+
node = _get_or_create_node(rel_types)
|
379
|
+
MatchStart.new(clause).new_match_rels?(rel_types).eval_context > node
|
380
|
+
node.eval_context
|
381
|
+
end
|
382
|
+
|
383
|
+
def incoming(*rel_types)
|
384
|
+
node = _get_or_create_node(rel_types)
|
385
|
+
MatchStart.new(clause).new_match_rels(rel_types).eval_context < node
|
386
|
+
node.eval_context
|
387
|
+
end
|
388
|
+
|
389
|
+
def incoming?(*rel_types)
|
390
|
+
node = _get_or_create_node(rel_types)
|
391
|
+
MatchStart.new(clause).new_match_rels?(rel_types).eval_context < node
|
392
|
+
node.eval_context
|
393
|
+
end
|
394
|
+
|
395
|
+
def both(*rel_types)
|
396
|
+
node = _get_or_create_node(rel_types)
|
397
|
+
MatchStart.new(clause).new_match_rels(rel_types).eval_context - node
|
398
|
+
node.eval_context
|
399
|
+
end
|
400
|
+
|
401
|
+
def both?(*rel_types)
|
402
|
+
node = _get_or_create_node(rel_types)
|
403
|
+
MatchStart.new(clause).new_match_rels?(rel_types).eval_context - node
|
404
|
+
node.eval_context
|
405
|
+
end
|
406
|
+
end
|
407
|
+
|
408
|
+
|
409
|
+
end
|
410
|
+
end
|
411
|
+
end
|