redgraph 0.1.1 → 0.2.1

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.
@@ -0,0 +1,105 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Redgraph
4
+ class Graph
5
+ module EdgeMethods
6
+ # Adds an edge. If successul it returns the created object, otherwise false
7
+ #
8
+ def add_edge(edge)
9
+ merge_or_add_edge(edge, :create)
10
+ end
11
+
12
+ # Merges (creates an edge unless one with the same type and properties exists) an edge.
13
+ # If successul it returns the object, otherwise false
14
+ #
15
+ def merge_edge(edge)
16
+ merge_or_add_edge(edge, :merge)
17
+ end
18
+
19
+ # Finds edges. Options:
20
+ #
21
+ # - type
22
+ # - src
23
+ # - dest
24
+ # - properties
25
+ # - order
26
+ # - limit
27
+ # - skip
28
+ #
29
+ def edges(type: nil, src: nil, dest: nil, properties: nil, order: nil, limit: nil, skip: nil)
30
+ _order = if order
31
+ raise MissingAliasPrefixError unless order.include?("edge.")
32
+ "ORDER BY #{order}"
33
+ end
34
+ _limit = "LIMIT #{limit}" if limit
35
+ _skip = "SKIP #{skip}" if skip
36
+
37
+ _where = if src || dest
38
+ clauses = [
39
+ ("ID(src) = #{src.id}" if src),
40
+ ("ID(dest) = #{dest.id}" if dest)
41
+ ].compact.join(" AND ")
42
+ "WHERE #{clauses}"
43
+ end
44
+
45
+ edge = Edge.new(type: type, src: src, dest: dest, properties: properties)
46
+
47
+ cmd = "MATCH #{edge.to_query_string} #{_where}
48
+ RETURN src, edge, dest #{_order} #{_skip} #{_limit}"
49
+ result = _query(cmd)
50
+
51
+ result.resultset.map do |item|
52
+ src = node_from_resultset_item(item["src"])
53
+ dest = node_from_resultset_item(item["dest"])
54
+ edge = edge_from_resultset_item(item["edge"])
55
+
56
+ edge.src = src
57
+ edge.dest = dest
58
+
59
+ edge
60
+ end
61
+ end
62
+
63
+ # Counts edges. Options:
64
+ #
65
+ # - type: filter by type
66
+ # - properties: filter by properties
67
+ #
68
+ def count_edges(type: nil, properties: nil)
69
+ edge = Edge.new(type: type, properties: properties)
70
+
71
+ cmd = "MATCH #{edge.to_query_string} RETURN COUNT(edge)"
72
+ query(cmd).flatten[0]
73
+ end
74
+
75
+
76
+ private
77
+
78
+ def edge_from_resultset_item(item)
79
+ (edge_id, type_id, _src_id, _dest_id, props) = item
80
+ attrs = HashWithIndifferentAccess.new
81
+
82
+ props.each do |(index, type, value)|
83
+ attrs[get_property(index)] = value
84
+ end
85
+
86
+ Edge.new.tap do |edge|
87
+ edge.id = edge_id
88
+ edge.type = get_relationship_type(type_id)
89
+ edge.properties = attrs
90
+ end
91
+ end
92
+
93
+ def merge_or_add_edge(edge, verb = :create)
94
+ verb = verb == :create ? "CREATE" : "MERGE"
95
+ result = _query("MATCH (src), (dest)
96
+ WHERE ID(src) = #{edge.src.id} AND ID(dest) = #{edge.dest.id}
97
+ #{verb} #{edge.to_query_string} RETURN ID(edge)")
98
+ return false if result.stats[:relationships_created] != 1
99
+ id = result.resultset.first["ID(edge)"]
100
+ edge.id = id
101
+ edge
102
+ end
103
+ end
104
+ end
105
+ end
@@ -0,0 +1,123 @@
1
+ # frozen_string_literal: true
2
+
3
+ module Redgraph
4
+ class Graph
5
+ module NodeMethods
6
+ # Adds a node. If successul it returns the created object, otherwise false
7
+ #
8
+ def add_node(node)
9
+ merge_or_add_node(node, :create)
10
+ end
11
+
12
+ # Merges (creates a node unless one with the same label and properties exists). If successul
13
+ # it returns the object, otherwise false
14
+ #
15
+ def merge_node(node)
16
+ merge_or_add_node(node, :merge)
17
+ end
18
+
19
+ def find_node_by_id(id)
20
+ result = _query("MATCH (node) WHERE ID(node) = #{id} RETURN node")
21
+ return nil if result.resultset.empty?
22
+ (node_id, labels, properties) = result.resultset.first["node"]
23
+ attrs = {}
24
+
25
+ properties.each do |(index, type, value)|
26
+ attrs[get_property(index)] = value
27
+ end
28
+ Node.new(label: get_label(labels.first), properties: attrs).tap do |node|
29
+ node.id = node_id
30
+ end
31
+ end
32
+
33
+
34
+ # Returns nodes. Options:
35
+ #
36
+ # - label: filter by label
37
+ # - properties: filter by properties
38
+ # - order: node.name ASC, node.year DESC
39
+ # - limit: number of items
40
+ # - skip: items offset (useful for pagination)
41
+ #
42
+ def nodes(label: nil, properties: nil, order: nil, limit: nil, skip: nil)
43
+ _label = ":`#{label}`" if label
44
+ _order = if order
45
+ raise MissingAliasPrefixError unless order.include?("node.")
46
+ "ORDER BY #{order}"
47
+ end
48
+ _limit = "LIMIT #{limit}" if limit
49
+ _skip = "SKIP #{skip}" if skip
50
+
51
+ node = Node.new(label: label, properties: properties)
52
+
53
+ cmd = "MATCH #{node.to_query_string} RETURN node #{_order} #{_skip} #{_limit}"
54
+
55
+ result = _query(cmd)
56
+
57
+ result.resultset.map do |item|
58
+ node_from_resultset_item(item["node"])
59
+ end
60
+ end
61
+
62
+ # Counts nodes. Options:
63
+ #
64
+ # - label: filter by label
65
+ # - properties: filter by properties
66
+ #
67
+ def count_nodes(label: nil, properties: nil)
68
+ node = Node.new(label: label, properties: properties)
69
+
70
+ cmd = "MATCH #{node.to_query_string} RETURN COUNT(node)"
71
+ # RedisGraph bug: if there are no matches COUNT returns zero rows
72
+ # https://github.com/RedisGraph/RedisGraph/issues/1455
73
+ query(cmd).flatten[0] || 0
74
+ end
75
+
76
+ def update_node(node)
77
+ return false unless node.persisted?
78
+ _set = node.properties.map do |(key, val)|
79
+ "node.#{key} = #{escape_value(val)}"
80
+ end.join(", ")
81
+
82
+ cmd = "MATCH (node) WHERE ID(node) = #{node.id} SET #{_set} RETURN node"
83
+ result = _query(cmd)
84
+ node_from_resultset_item(result.resultset.first["node"])
85
+ end
86
+
87
+ def destroy_node(node)
88
+ return false unless node.persisted?
89
+ cmd = "MATCH (node) WHERE ID(node) = #{node.id} DELETE node"
90
+ result = _query(cmd)
91
+ result.stats["nodes_deleted"] == 1
92
+ end
93
+
94
+ private
95
+
96
+ # Builds a Node object from the raw data
97
+ #
98
+ def node_from_resultset_item(item)
99
+ (node_id, labels, props) = item
100
+ attrs = HashWithIndifferentAccess.new
101
+
102
+ props.each do |(index, type, value)|
103
+ attrs[get_property(index)] = value
104
+ end
105
+ Node.new(label: get_label(labels.first), properties: attrs).tap do |node|
106
+ node.id = node_id
107
+ end
108
+ end
109
+
110
+ def merge_or_add_node(node, verb = :create)
111
+ verb = verb == :create ? "CREATE" : "MERGE"
112
+ result = _query("#{verb} #{node.to_query_string} RETURN ID(node)")
113
+ # Should we treat this case differently?
114
+ # return false if result.stats[:nodes_created] != 1
115
+ id = result.resultset.first["ID(node)"]
116
+ node.id = id
117
+ node
118
+ end
119
+
120
+
121
+ end
122
+ end
123
+ end
data/lib/redgraph/node.rb CHANGED
@@ -2,19 +2,27 @@
2
2
 
3
3
  module Redgraph
4
4
  class Node
5
+ include Util
6
+
5
7
  attr_accessor :id, :label, :properties
6
8
 
7
- def initialize(label:, properties: {})
9
+ def initialize(label: nil, properties: nil, id: nil)
10
+ @id = id
8
11
  @label = label
9
- @properties = properties
12
+ @properties = (properties || {}).with_indifferent_access
10
13
  end
11
14
 
12
15
  def persisted?
13
- !id.nil?
16
+ id.present?
14
17
  end
15
18
 
16
19
  def ==(other)
17
20
  super || other.instance_of?(self.class) && !id.nil? && other.id == id
18
21
  end
22
+
23
+ def to_query_string(item_alias: 'node')
24
+ _label = ":#{label}" if label
25
+ "(#{item_alias}#{_label} #{properties_to_string(properties)})"
26
+ end
19
27
  end
20
28
  end
@@ -0,0 +1,99 @@
1
+ # frozen_string_literal: true
2
+ require_relative 'node_model/class_methods'
3
+ require_relative 'node_model/graph_manipulation'
4
+ require_relative 'node_model/persistence'
5
+
6
+ module Redgraph
7
+ # This mixin allows you to use an interface similar to ActiveRecord
8
+ #
9
+ # class Actor
10
+ # include Redgraph::NodeModel
11
+ #
12
+ # self.graph = Redgraph::Graph.new("movies", url: $REDIS_URL)
13
+ # self.label = "actor" # optional, if missing it will be extracted from the class name
14
+ # attribute :name
15
+ # end
16
+ #
17
+ # You will then be able to
18
+ #
19
+ # john = Actor.find(123)
20
+ # total = Actor.count
21
+ #
22
+ # When you create a record it will automatically set the _type property with the class name.
23
+ # This allows reifying the node into the corresponding NodeModel class.
24
+ #
25
+ module NodeModel
26
+ extend ActiveSupport::Concern
27
+
28
+ included do |base|
29
+ include Persistence
30
+ include GraphManipulation
31
+
32
+ @attribute_names = [:id]
33
+
34
+ attr_accessor :id, :_type
35
+
36
+ class << self
37
+ attr_reader :attribute_names
38
+ attr_accessor :graph
39
+
40
+ def attribute(name)
41
+ @attribute_names << name
42
+ attr_accessor(name)
43
+ end
44
+
45
+ private
46
+
47
+ def inherited(subclass)
48
+ super
49
+ subclass.instance_variable_set(:@attribute_names, @attribute_names.dup)
50
+ subclass.instance_variable_set(:@graph, @graph.dup)
51
+ end
52
+ end
53
+ end
54
+
55
+ def initialize(**args)
56
+ absent_attributes = args.keys.map(&:to_sym) - self.class.attribute_names - [:_type]
57
+
58
+ if absent_attributes.any?
59
+ raise ArgumentError, "Unknown attribute #{absent_attributes}"
60
+ end
61
+
62
+ args.each do |name, value|
63
+ instance_variable_set("@#{name}", value)
64
+ end
65
+ end
66
+
67
+ # The current graph
68
+ #
69
+ def graph
70
+ self.class.graph
71
+ end
72
+
73
+ def label
74
+ self.class.label
75
+ end
76
+
77
+ # Object attributes as a hash
78
+ #
79
+ def attributes
80
+ self.class.attribute_names.to_h { |name| [name, public_send(name)] }
81
+ end
82
+
83
+ def assign_attributes(attrs = {})
84
+ attrs.each do |name, value|
85
+ instance_variable_set("@#{name}", value)
86
+ end
87
+ end
88
+
89
+ def to_node
90
+ props = attributes.except(:id).merge(_type: self.class.name)
91
+ Redgraph::Node.new(id: id, label: label, properties: props)
92
+ end
93
+
94
+ def ==(other)
95
+ attributes == other.attributes && id == other.id
96
+ end
97
+
98
+ end
99
+ end
@@ -0,0 +1,83 @@
1
+ module Redgraph
2
+ module NodeModel
3
+ module ClassMethods
4
+ # Returns an array of nodes. Options:
5
+ #
6
+ # - label: filter by label
7
+ # - properties: filter by properties
8
+ # - order: node.name ASC, node.year DESC
9
+ # - limit: number of items
10
+ # - skip: items offset (useful for pagination)
11
+ #
12
+ def all(label: nil, properties: {}, limit: nil, skip: nil, order: nil)
13
+ graph.nodes(label: label, properties: properties_plus_type(properties),
14
+ limit: limit, skip: skip, order: nil).map do |node|
15
+ reify_from_node(node)
16
+ end
17
+ end
18
+
19
+ # Returns the number of nodes with the current label. Options:
20
+ #
21
+ # - properties: filter by properties
22
+ #
23
+ def count(label: nil, properties: nil)
24
+ graph.count_nodes(label: label, properties: properties_plus_type(properties))
25
+ end
26
+
27
+ # Finds a node by id. Returns nil if not found
28
+ #
29
+ def find(id)
30
+ node = graph.find_node_by_id(id)
31
+ return unless node
32
+ reify_from_node(node)
33
+ end
34
+
35
+ # Sets the label for this class of nodes. If missing it will be computed from the class name
36
+ def label=(x)
37
+ @label = x
38
+ end
39
+
40
+ # Current label
41
+ #
42
+ def label
43
+ @label ||= default_label
44
+ end
45
+
46
+ # Converts a Node object into NodeModel
47
+ #
48
+ def reify_from_node(node)
49
+ klass = node.properties[:_type].to_s.safe_constantize || self
50
+ klass.new(id: node.id, **node.properties)
51
+ end
52
+
53
+ # Runs a query on the graph, but converts the nodes to the corresponding ActiveModel class
54
+ # if available - otherwise they stay NodeObjects.
55
+ #
56
+ # Returns an array of rows.
57
+ #
58
+ def query(cmd)
59
+ raise MissingGraphError unless graph
60
+
61
+ graph.query(cmd).map do |row|
62
+ row.map do |item|
63
+ item.is_a?(Node) ? reify_from_node(item) : item
64
+ end
65
+ end
66
+ end
67
+
68
+ def create(properties)
69
+ new(**properties).add_to_graph
70
+ end
71
+
72
+ private
73
+
74
+ def default_label
75
+ name.demodulize.underscore
76
+ end
77
+
78
+ def properties_plus_type(properties = {})
79
+ {_type: name}.merge(properties || {})
80
+ end
81
+ end
82
+ end
83
+ end
@@ -0,0 +1,24 @@
1
+ module Redgraph
2
+ module NodeModel
3
+ module GraphManipulation
4
+ # Adds a relation between the node and another node.
5
+ #
6
+ # - type: type of relation
7
+ # - node: the destination node
8
+ # - properties: optional properties hash
9
+ # - allow_duplicates: if false it will create a relation between two nodes with the same type
10
+ # and properties only if not present
11
+ #
12
+ def add_relation(type:, node:, properties: nil, allow_duplicates: true)
13
+ edge = Edge.new(type: type, src: to_node, dest: node.to_node, properties: properties)
14
+ allow_duplicates ? graph.add_edge(edge) : graph.merge_edge(edge)
15
+ end
16
+
17
+ # Runs a custom query on the graph
18
+ #
19
+ def query(cmd)
20
+ self.class.query(cmd)
21
+ end
22
+ end
23
+ end
24
+ end