module Neo4j module Core # This module contains a number of mixins and classes used by the neo4j.rb cypher DSL. # The Cypher DSL is evaluated in the context of {Neo4j::Cypher} which contains a number of methods (e.g. {Neo4j::Cypher#node}) # which returns classes from this module. module Cypher module MathFunctions def abs(value=nil) _add_math_func(:abs, value) end def sqrt(value=nil) _add_math_func(:sqrt, value) end def round(value=nil) _add_math_func(:round, value) end def sign(value=nil) _add_math_func(:sign, value) end # @private def _add_math_func(name, value=nil) value ||= self.respond_to?(:var_name) ? self.var_name : to_s expressions.delete(self) Property.new(expressions, nil, name).to_function!(value) end end module MathOperator def -(other) ExprOp.new(self, other, '-') end def +(other) ExprOp.new(self, other, '+') end end module Comparable def <(other) ExprOp.new(self, other, '<') end def <=(other) ExprOp.new(self, other, '<=') end def =~(other) ExprOp.new(self, other, '=~') end def >(other) ExprOp.new(self, other, '>') end def >=(other) ExprOp.new(self, other, '>=') end ## Only in 1.9 if RUBY_VERSION > "1.9.0" eval %{ def !=(other) other.is_a?(String) ? ExprOp.new(self, other, "!=") : super end } end def ==(other) if other.is_a?(Fixnum) || other.is_a?(String) || other.is_a?(Regexp) ExprOp.new(self, other, "=") else super end end end module PredicateMethods def all?(&block) self.respond_to?(:iterable) Predicate.new(expressions, :op => 'all', :clause => :where, :input => input, :iterable => iterable, :predicate_block => block) end def extract(&block) Predicate.new(expressions, :op => 'extract', :clause => :return, :input => input, :iterable => iterable, :predicate_block => block) end def filter(&block) Predicate.new(expressions, :op => 'filter', :clause => :return, :input => input, :iterable => iterable, :predicate_block => block) end def any?(&block) Predicate.new(@expressions, :op => 'any', :clause => :where, :input => input, :iterable => iterable, :predicate_block => block) end def none?(&block) Predicate.new(@expressions, :op => 'none', :clause => :where, :input => input, :iterable => iterable, :predicate_block => block) end def single?(&block) Predicate.new(@expressions, :op => 'single', :clause => :where, :input => input, :iterable => iterable, :predicate_block => block) end end module Variable attr_accessor :return_method def distinct self.return_method = {:name => 'distinct', :bracket => false} self end def [](prop_name) Property.new(expressions, self, prop_name) end def as(v) @var_name = v self end # generates a ID cypher fragment. def neo_id Property.new(@expressions, self, 'ID').to_function! end # generates a has cypher fragment. def property?(p) p = Property.new(expressions, self, p) p.binary_operator("has") end # generates a is null cypher fragment. def exist? p = Property.new(expressions, self, p) p.binary_operator("", " is null") end # Can be used instead of [_classname] == klass def is_a?(klass) return super if klass.class != Class || !klass.respond_to?(:_load_wrapper) self[:_classname] == klass.to_s end def count #expressions.delete(self) ExprOp.new(self, nil, 'count') end end module Matchable def where(&block) x = block.call(self) expressions.delete(x) ExprOp.new(x, nil, "").binary! self end def where_not(&block) x = block.call(self) expressions.delete(x) ExprOp.new(x, nil, "not").binary! self end # This operator means related to, without regard to type or direction. # @param [Symbol, #var_name] other either a node (Symbol, #var_name) # @return [MatchRelLeft, MatchNode] def <=>(other) MatchNode.new(self, other, expressions, :both) end # This operator means outgoing related to # @param [Symbol, #var_name, String] other the relationship # @return [MatchRelLeft, MatchNode] def >(other) MatchRelLeft.new(self, other, expressions, :outgoing) end # This operator means any direction related to # @param (see #>) # @return [MatchRelLeft, MatchNode] def -(other) MatchRelLeft.new(self, other, expressions, :both) end # This operator means incoming related to # @param (see #>) # @return [MatchRelLeft, MatchNode] def <(other) MatchRelLeft.new(self, other, expressions, :incoming) end # Outgoing relationship to other node # @param [Symbol, #var_name] other either a node (Symbol, #var_name) # @return [MatchRelLeft, MatchNode] def >>(other) MatchNode.new(self, other, expressions, :outgoing) end def outgoing(rel_type) node = NodeVar.new(@expressions, @variables) MatchRelLeft.new(self, ":`#{rel_type}`", expressions, :outgoing) > node node end def incoming(rel_type) node = NodeVar.new(@expressions, @variables) MatchRelLeft.new(self, ":`#{rel_type}`", expressions, :incoming) < node node end def both(rel_type) node = NodeVar.new(@expressions, @variables) MatchRelLeft.new(self, ":`#{rel_type}`", expressions, :both) < node node end # Incoming relationship to other node # @param [Symbol, #var_name] other either a node (Symbol, #var_name) # @return [MatchRelLeft, MatchNode] def <<(other) MatchNode.new(self, other, expressions, :incoming) end end class Expression attr_reader :expressions attr_accessor :separator, :clause def initialize(expressions, clause) @clause = clause @expressions = expressions insert_last(clause) @separator = "," end def insert_last(clause) curr_clause = clause while (i = @expressions.reverse.index { |e| e.clause == curr_clause }).nil? && curr_clause != :start curr_clause = prev_clause(curr_clause) end if i.nil? @expressions << self else pos = @expressions.size - i @expressions.insert(pos, self) end end def prev_clause(clause) {:limit => :skip, :skip => :order_by, :order_by => :return, :return => :where, :where => :match, :match => :start}[clause] end def prefixes {:start => "START", :where => " WHERE", :match => " MATCH", :return => " RETURN", :order_by => " ORDER BY", :skip => " SKIP", :limit => " LIMIT"} end def prefix prefixes[clause] end def valid? true end end # A property is returned from a Variable by using the [] operator. # # It has a number of useful method like # count, sum, avg, min, max, collect, head, last, tail, # # @example # n=node(2, 3, 4); n[:name].collect # # same as START n0=node(2,3,4) RETURN collect(n0.property) class Property # @private attr_reader :expressions, :var_name, :var_expr include Comparable include MathOperator include MathFunctions include PredicateMethods def initialize(expressions, var_expr, prop_name) @var_expr = var_expr @var = var_expr.respond_to?(:var_name) ? var_expr.var_name : var_expr @expressions = expressions @prop_name = prop_name @var_name = @prop_name ? "#{@var.to_s}.#{@prop_name}" : @var.to_s end # @private def to_function!(var = @var.to_s) @var_name = "#{@prop_name}(#{var})" self end # Make it possible to rename a property with a different name (AS) def as(new_name) @var_name = "#{@var_name} AS #{new_name}" end # required by the Predicate Methods Module # @see PredicateMethods # @private def iterable var_name end def input self end # @private def in?(values) binary_operator("", " IN [#{values.map { |x| %Q["#{x}"] }.join(',')}]") end # Only return distinct values/nodes/rels/paths def distinct @var_name = "distinct #{@var_name}" self end def length @prop_name = "length" to_function! self end %w[count sum avg min max collect head last tail].each do |meth_name| define_method(meth_name) do function(meth_name.to_s) end end # @private def function(func_name_pre, func_name_post = "") ExprOp.new(self, nil, func_name_pre, func_name_post) end # @private def binary_operator(op, post_fix = "") ExprOp.new(self, nil, op, post_fix).binary! end end class Start < Expression # @private attr_reader :var_name include Variable include Matchable def initialize(var_name, expressions) @var_name = "#{var_name}#{expressions.size}" super(expressions, :start) end end # Can be created from a node dsl method. class StartNode < Start # @private attr_reader :nodes def initialize(nodes, expressions) super("n", expressions) @nodes = nodes.map { |n| n.respond_to?(:neo_id) ? n.neo_id : n } end def to_s "#{var_name}=node(#{nodes.join(',')})" end end # Can be created from a rel dsl method. class StartRel < Start # @private attr_reader :rels def initialize(rels, expressions) super("r", expressions) @rels = rels.map { |n| n.respond_to?(:neo_id) ? n.neo_id : n } end def to_s "#{var_name}=relationship(#{rels.join(',')})" end end class NodeQuery < Start attr_reader :index_name, :query def initialize(index_class, query, index_type, expressions) super("n", expressions) @index_name = index_class.index_name_for_type(index_type) @query = query end def to_s "#{var_name}=node:#{index_name}(#{query})" end end class NodeLookup < Start attr_reader :index_name, :query def initialize(index_class, key, value, expressions) super("n", expressions) index_type = index_class.index_type(key.to_s) raise "No index on #{index_class} property #{key}" unless index_type @index_name = index_class.index_name_for_type(index_type) @query = %Q[#{key}="#{value}"] end def to_s %Q[#{var_name}=node:#{index_name}(#{query})] end end # The return statement in the cypher query class Return < Expression attr_reader :var_name def initialize(name_or_ref, expressions, opts = {}) super(expressions, :return) @name_or_ref = name_or_ref @name_or_ref.referenced! if @name_or_ref.respond_to?(:referenced!) @var_name = @name_or_ref.respond_to?(:var_name) ? @name_or_ref.var_name : @name_or_ref.to_s opts.each_pair { |k, v| self.send(k, v) } end # @private def return_method @name_or_ref.respond_to?(:return_method) && @name_or_ref.return_method end # @private def as_return_method if return_method[:bracket] "#{return_method[:name]}(#@var_name)" else "#{return_method[:name]} #@var_name" end end # Specifies an ORDER BY cypher query # @param [Property] props the properties which should be sorted # @return self def asc(*props) @order_by ||= OrderBy.new(expressions) expressions.delete(props.first) @order_by.asc(props) self end # Specifies an ORDER BY cypher query # @param [Property] props the properties which should be sorted # @return self def desc(*props) @order_by ||= OrderBy.new(expressions) expressions.delete(props.first) @order_by.desc(props) self end # Creates a SKIP cypher clause # @param [Fixnum] val the number of entries to skip # @return self def skip(val) Skip.new(expressions, val) self end # Creates a LIMIT cypher clause # @param [Fixnum] val the number of entries to limit # @return self def limit(val) Limit.new(expressions, val) self end def to_s return_method ? as_return_method : var_name.to_s end end # Can be used to skip result from a return clause class Skip < Expression def initialize(expressions, value) super(expressions, :skip) @value = value end def to_s @value end end # Can be used to limit result from a return clause class Limit < Expression def initialize(expressions, value) super(expressions, :limit) @value = value end def to_s @value end end class OrderBy < Expression def initialize(expressions) super(expressions, :order_by) @orders = [] end def asc(props) @orders << [:asc, props] end def desc(props) @orders << [:desc, props] end def to_s @orders.map do |pair| if pair[0] == :asc pair[1].map(&:var_name).join(', ') else pair[1].map(&:var_name).join(', ') + " DESC" end end.join(', ') end end # Created from a node's match operator like >> or <. class Match < Expression # @private attr_reader :dir, :expressions, :left, :right, :var_name, :dir_op # @private attr_accessor :algorithm, :next, :prev include Variable def initialize(left, right, expressions, dir, dir_op) super(expressions, :match) @var_name = "m#{expressions.size}" @dir = dir @dir_op = dir_op @prev = left if left.is_a?(Match) @left = left @right = right end # Generates a x in nodes(m3) cypher expression. # # @example # p.nodes.all? { |x| x[:age] > 30 } def nodes Entities.new(@expressions, "nodes", self) end # Generates a x in relationships(m3) cypher expression. # # @example # p.relationships.all? { |x| x[:age] > 30 } def rels Entities.new(@expressions, "relationships", self) end # returns the length of the path def length self.return_method = {:name => 'length', :bracket => true} self end # @private def find_match_start c = self while (c.prev) do c = c.prev end c end # @private def left_var_name @left.respond_to?(:var_name) ? @left.var_name : @left.to_s end # @private def right_var_name @right.respond_to?(:var_name) ? @right.var_name : @right.to_s end # @private def right_expr c = @right r = while (c) break c.var_expr if c.respond_to?(:var_expr) c = c.respond_to?(:left_expr) && c.left_expr end || @right r.respond_to?(:expr) ? r.expr : right_var_name end # @private def referenced! @referenced = true end # @private def referenced? !!@referenced end # @private def to_s curr = find_match_start result = (referenced? || curr.referenced?) ? "#{var_name} = " : "" result << (algorithm ? "#{algorithm}(" : "") begin result << curr.expr end while (curr = curr.next) result << ")" if algorithm result end end # The left part of a match clause, e.g. node < rel(':friends') # Can return {MatchRelRight} using a match operator method. class MatchRelLeft < Match def initialize(left, right, expressions, dir) super(left, right, expressions, dir, dir == :incoming ? '<-' : '-') end # @param [Symbol,NodeVar,String] other part of the match cypher statement. # @return [MatchRelRight] the right part of an relationship cypher query. def >(other) expressions.delete(self) self.next = MatchRelRight.new(self, other, expressions, :outgoing) end # @see #> # @return (see #>) def <(other) expressions.delete(self) self.next = MatchRelRight.new(self, other, expressions, :incoming) end # @see #> # @return (see #>) def -(other) expressions.delete(self) self.next = MatchRelRight.new(self, other, expressions, :both) end # @return [String] a cypher string for this match. def expr if prev # we have chained more then one relationships in a match expression "#{dir_op}[#{right_expr}]" else # the right is an relationship and could be an expressions, e.g "r?" "(#{left_var_name})#{dir_op}[#{right_expr}]" end end end class MatchRelRight < Match # @param left the left part of the query # @param [Symbol,NodeVar,String] right part of the match cypher statement. def initialize(left, right, expressions, dir) super(left, right, expressions, dir, dir == :outgoing ? '->' : '-') end # @param [Symbol,NodeVar,String] other part of the match cypher statement. # @return [MatchRelLeft] the right part of an relationship cypher query. def >(other) expressions.delete(self) self.next = MatchRelLeft.new(self, other, expressions, :outgoing) end # @see #> # @return (see #>) def <(other) expressions.delete(self) self.next = MatchRelLeft.new(self, other, expressions, :incoming) end # @see #> # @return (see #>) def -(other) expressions.delete(self) self.next = MatchRelLeft.new(self, other, expressions, :both) end def <<(other) expressions.delete(self) self.next = MatchNode.new(self, other, expressions, :incoming) end def >>(other) expressions.delete(self) self.next = MatchNode.new(self, other, expressions, :outgoing) end # @return [String] a cypher string for this match. def expr "#{dir_op}(#{right_expr})" end # negate this match def not expressions.delete(self) ExprOp.new(left, nil, "not").binary! end if RUBY_VERSION > "1.9.0" eval %{ def ! expressions.delete(self) ExprOp.new(left, nil, "not").binary! end } end end # The right part of a match clause (node_b), e.g. node_a > rel(':friends') > node_b # class MatchNode < Match attr_reader :dir_op def initialize(left, right, expressions, dir) dir_op = case dir when :outgoing then "-->" when :incoming then "<--" when :both then "--" end super(left, right, expressions, dir, dir_op) end # @return [String] a cypher string for this match. def expr if prev # we have chained more then one relationships in a match expression "#{dir_op}(#{right_expr})" else # the right is an relationship and could be an expressions, e.g "r?" "(#{left_var_name})#{dir_op}(#{right_expr})" end end def <<(other) expressions.delete(self) self.next = MatchNode.new(self, other, expressions, :incoming) end def >>(other) expressions.delete(self) self.next = MatchNode.new(self, other, expressions, :outgoing) end # @param [Symbol,NodeVar,String] other part of the match cypher statement. # @return [MatchRelRight] the right part of an relationship cypher query. def >(other) expressions.delete(self) self.next = MatchRelLeft.new(self, other, expressions, :outgoing) end # @see #> # @return (see #>) def <(other) expressions.delete(self) self.next = MatchRelLeft.new(self, other, expressions, :incoming) end # @see #> # @return (see #>) def -(other) expressions.delete(self) self.next = MatchRelLeft.new(self, other, expressions, :both) end end # Represents an unbound node variable used in match statements class NodeVar include Variable include Matchable # @return the name of the variable attr_reader :var_name attr_reader :expressions def initialize(expressions, variables) variables ||= [] @var_name = "v#{variables.size}" variables << self @variables = variables @expressions = expressions end # @return [String] a cypher string for this node variable def to_s var_name end # @private def expr to_s end end # represent an unbound relationship variable used in match,where,return statement class RelVar include Variable attr_reader :var_name, :expr, :expressions def initialize(expressions, variables, expr) variables << self @expr = expr @expressions = expressions guess = expr ? /([[:alpha:]_]*)/.match(expr)[1] : "" @auto_var_name = "v#{variables.size}" @var_name = guess.empty? ? @auto_var_name : guess end def rel_type Property.new(@expressions, self, 'type').to_function! end def [](p) if @expr.to_s[0..0] == ':' @var_name = @auto_var_name @expr = "#{@var_name}#{@expr}" end super end # @return [String] a cypher string for this relationship variable def to_s var_name end end class ExprOp < Expression attr_reader :left, :right, :op, :neg, :post_fix, :left_expr, :right_expr include MathFunctions def initialize(left_expr, right_expr, op, post_fix = "") super(left_expr.expressions, :where) @left_expr = left_expr @right_expr = right_expr @op = op @post_fix = post_fix self.expressions.delete(left_expr) self.expressions.delete(right_expr) @left = quote(left_expr) if regexp?(right_expr) @op = "=~" @right = to_regexp(right_expr) else @right = right_expr && quote(right_expr) end @neg = nil end def separator " and " end def quote(val) if val.respond_to?(:var_name) && !val.kind_of?(Match) val.var_name else val.is_a?(String) ? %Q["#{val}"] : val end end def regexp?(right) @op == "=~" || right.is_a?(Regexp) end def to_regexp(val) %Q[/#{val.respond_to?(:source) ? val.source : val.to_s}/] end def count ExprOp.new(self, nil, 'count') end def &(other) ExprOp.new(self, other, "and") end def |(other) ExprOp.new(self, other, "or") end def -@ @neg = "not" self end def not @neg = "not" self end # Only in 1.9 if RUBY_VERSION > "1.9.0" eval %{ def ! @neg = "not" self end } end def left_to_s left.is_a?(ExprOp) ? "(#{left})" : left end def right_to_s right.is_a?(ExprOp) ? "(#{right})" : right end def binary! @binary = true self end def valid? # it is only valid in a where clause if it's either binary or it has right and left values @binary ? @left : @left && @right end def to_s if @right neg ? "#{neg}(#{left_to_s} #{op} #{right_to_s})" : "#{left_to_s} #{op} #{right_to_s}" else # binary operator neg ? "#{neg}#{op}(#{left_to_s}#{post_fix})" : "#{op}(#{left_to_s}#{post_fix})" end end end class Where < Expression def initialize(expressions, where_statement = nil) super(expressions, :where) @where_statement = where_statement end def to_s @where_statement.to_s end end class Predicate < Expression attr_accessor :params def initialize(expressions, params) @params = params @identifier = :x params[:input].referenced! if params[:input].respond_to?(:referenced!) super(expressions, params[:clause]) end def identifier(i) @identifier = i self end def to_s input = params[:input] if input.kind_of?(Property) yield_param = Property.new([], @identifier, nil) args = "" else yield_param = NodeVar.new([], []).as(@identifier.to_sym) args = "(#{input.var_name})" end context = Neo4j::Cypher.new(yield_param, ¶ms[:predicate_block]) context.expressions.each { |e| e.clause = nil } if params[:clause] == :return where_or_colon = ':' else where_or_colon = 'WHERE' end predicate_value = context.to_s[1..-1] # skip separator , "#{params[:op]}(#@identifier in #{params[:iterable]}#{args} #{where_or_colon} #{predicate_value})" end end class Entities include PredicateMethods attr_reader :input, :expressions, :iterable def initialize(expressions, iterable, input) @iterable = iterable @input = input @expressions = expressions end end end end end