module Neo4j module Core class Label attr_reader :name def initialize(name) @name = name end def create_index(property, options = {}) validate_index_options!(options) properties = property.is_a?(Array) ? property.join(',') : property schema_query("CREATE INDEX ON :`#{@name}`(#{properties})") end def drop_index(property, options = {}) validate_index_options!(options) schema_query("DROP INDEX ON :`#{@name}`(#{property})") end # Creates a neo4j constraint on a property # See http://docs.neo4j.org/chunked/stable/query-constraints.html # @example # label = Neo4j::Label.create(:person, driver) # label.create_constraint(:name, {type: :unique}, driver) # def create_constraint(property, constraints) cypher = case constraints[:type] when :unique, :uniqueness "CREATE CONSTRAINT ON (n:`#{name}`) ASSERT n.`#{property}` IS UNIQUE" else fail "Not supported constraint #{constraints.inspect} for property #{property} (expected :type => :unique)" end schema_query(cypher) end def create_uniqueness_constraint(property, options = {}) create_constraint(property, options.merge(type: :unique)) end # Drops a neo4j constraint on a property # See http://docs.neo4j.org/chunked/stable/query-constraints.html # @example # label = Neo4j::Label.create(:person, driver) # label.create_constraint(:name, {type: :unique}, driver) # label.drop_constraint(:name, {type: :unique}, driver) # def drop_constraint(property, constraint) cypher = case constraint[:type] when :unique, :uniqueness "n.`#{property}` IS UNIQUE" when :exists "exists(n.`#{property}`)" else fail "Not supported constraint #{constraint.inspect}" end schema_query("DROP CONSTRAINT ON (n:`#{name}`) ASSERT #{cypher}") end def drop_uniqueness_constraint(property, options = {}) drop_constraint(property, options.merge(type: :unique)) end def indexes self.class.indexes.select do |definition| definition[:label] == @name.to_sym end end def drop_indexes self.class.drop_indexes end def index?(property) indexes.any? { |definition| definition[:properties] == [property.to_sym] } end def constraints(_options = {}) Neo4j::Transaction.constraints.select do |definition| definition[:label] == @name.to_sym end end def uniqueness_constraints(_options = {}) constraints.select do |definition| definition[:type] == :uniqueness end end def constraint?(property) constraints.any? { |definition| definition[:properties] == [property.to_sym] } end def uniqueness_constraint?(property) uniqueness_constraints.include?([property]) end private # Store schema threads on the driver so that we can easily wait for all # threads on a driver regardless of label def schema_threads self.class.schema_threads end def schema_threads=(array) self.class.set_schema_threads(array) end class << self def indexes Neo4j::Transaction.indexes end def drop_indexes indexes.each do |definition| begin Neo4j::Transaction.query("DROP INDEX ON :`#{definition[:label]}`(#{definition[:properties][0]})") rescue Neo4j::Driver::Exceptions::DatabaseException # This will error on each constraint. Ignore and continue. next end end end def drop_constraints Neo4j::Transaction.named_constraints.each do |constraint| Neo4j::Transaction.query("DROP CONSTRAINT #{constraint.name}") end Neo4j::Transaction.constraints.each do |definition| Neo4j::Transaction.query("DROP CONSTRAINT ON (n:`#{definition[:label]}`) ASSERT n.`#{definition[:properties][0]}` IS UNIQUE") end end def wait_for_schema_changes schema_threads.map(&:join) set_schema_threads(driver, []) end def schema_threads Neo4j::Transaction.instance_variable_get('@_schema_threads') || [] end def set_schema_threads(array) Neo4j::Transaction.instance_variable_set('@_schema_threads', array) end end def schema_query(cypher) Neo4j::Transaction.transaction { |tx| tx.query(cypher, {}) } end def validate_index_options!(options) return unless options[:type] && options[:type] != :exact fail "Type #{options[:type]} is not supported" end end end end