module Neo4j module ActiveNode module HasN # A DSL for declared relationships has_n and has_one # This DSL will be used to create accessor methods for relationships. # Instead of using the 'raw' Neo4j::ActiveNode#rels method where one needs to know # the name of relationship and direction one can use the generated accessor methods. # # @example # # class Folder # include Neo4j::ActiveNode # property :name # # Declaring a Many relationship to any other node # has_n(:files) # end # # class File # include Neo4j::ActiveNode # # declaring a incoming relationship from Folder's relationship files # has_one(:folder).from(Folder, :files) # end # # The following methods will be generated: # Folder#files :: returns an Enumerable of outgoing nodes for relationship 'files' # Folder#files_rels :: returns an Enumerable of outgoing relationships for relationship 'files' # File#folder :: for adding one node for the relationship 'files' from the outgoing Folder node # File#folder_rel :: for accessing relationship 'files' from the outgoing Folder node # File#folder :: for accessing nodes from relationship 'files' from the outgoing Folder node # class DeclRel attr_reader :source_class, :dir, :rel_type, :method_id def initialize(method_id, has_one, source_class) @method_id = method_id @has_one = has_one @dir = :outgoing @rel_type = method_id.to_sym @source_class = source_class end def inherit_new base = self dr = DeclRel.new(@method_id, @has_one, @source_class) dr.instance_eval do @dir = base.dir @rel_type = base.rel_type @target_name = base.target_name if base.target_name @source_class = base.source_class end dr end def to_s "DeclRel #{object_id} dir: #{@dir} rel_id: #{@method_id}, rel_type: #{@rel_type}, target_class:#{@target_name}" end # @return [true, false] def has_one? @has_one end # @return [true, false] def has_n? !@has_one end # @return [true,false] def incoming? #:nodoc: @dir == :incoming end # Declares an outgoing relationship type. # It is possible to prefix relationship types so that it's possible to distinguish different incoming relationships. # There is no validation that the added node is of the specified class. # # @example Example # class FolderNode # include Neo4j::ActiveNode # has_n(:files).to(FileNode) # has_one(:root).to("FileSystem") # also possible, if the class is not defined yet # end # # folder = FolderNode.new # # generate a relationship between folder and file of type 'FileNode#files' # folder.files << FileNode.new # # @example relationship with a hash, user defined relationship # # class FolderNode # include Neo4j::ActiveNode # has_n(:files).to('FolderNode#files') # end # # @example without prefix # # class FolderNode # include Neo4j::ActiveNode # has_n(:files).to(:contains) # end # # file = FileNode.new # # create an outgoing relationship of type 'contains' from folder node to file # folder.files << FolderNode.new # # @param [Class, String, Symbol] target the other class to which this relationship goes (if String or Class) or the relationship (if Symbol) # @param [String, Symbol] rel_type the rel_type postfix for the relationships, which defaults to the same as the has_n/one method id # @return self def to(target, rel_type = @method_id) @dir = :outgoing case target when /#/ @target_name, _ = target.to_s.split("#") @rel_type = target.to_sym when Class, String @target_name = target.to_s @rel_type = "#{@source_class}##{rel_type}".to_sym when Symbol @target_name = nil @rel_type = target.to_sym else raise "Expected a class or a symbol for, got #{target}/#{target.class}" end self end # Specifies an incoming relationship. # Will use the outgoing relationship given by the from class. # # @example with prefix FileNode # class FolderNode # include Neo4j::NodeMixin # has_n(:files).to(FileNode) # end # # class FileNode # include Neo4j::NodeMixin # # will only traverse any incoming relationship of type files from node FileNode # has_one(:folder).from(FolderNode.files) # # alternative: has_one(:folder).from(FolderNode, :files) # end # # file = FileNode.new # # create an outgoing relationship of type 'FileNode#files' from folder node to file (FileNode is the prefix). # file.folder = FolderNode.new # # @example without prefix # # class FolderNode # include Neo4j::NodeMixin # has_n(:files) # end # # class FileNode # include Neo4j::NodeMixin # has_one(:folder).from(:files) # will traverse any incoming relationship of type files # end # # file = FileNode.new # # create an outgoing relationship of type 'files' from folder node to file # file.folder = FolderNode.new # # def from(target, rel_type=@method_id) @dir = :incoming case target when /#/ @target_name, _ = target.to_s.split("#") @rel_type = target when Class, String @target_name = target.to_s @rel_type = "#{@target_name}##{rel_type}".to_sym when Symbol @target_name = nil @rel_type = target.to_sym else raise "Expected a class or a symbol for, got #{target}/#{target.class}" end self end # @private def target_name @target_name end def target_class @target_name && @target_name.split("::").inject(Kernel) { |container, name| container.const_get(name.to_s) } end # @private def each_node(node, &block) node.nodes(dir: dir, type: rel_type).each { |n| block.call(n)} end def all_relationships(node) to_enum(:each_rel, node) end def each_rel(node, &block) #:nodoc: node.rels(dir: dir, type: rel_type).each { |rel| block.call(rel) } end def single_relationship(node) node.rel(dir: dir, type: rel_type) end def single_node(node) node.node(dir: dir, type: rel_type) end # @private def create_relationship_to(node, other, relationship_props={}) # :nodoc: from, to = incoming? ? [other, node] : [node, other] from.create_rel(@rel_type, to, relationship_props) end end end end end