flockdb 0.6.1 → 0.7.0
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 +3 -0
- data/Gemfile.lock +28 -0
- data/Rakefile +1 -20
- data/flockdb.gemspec +19 -75
- data/lib/flock.rb +17 -19
- data/lib/flock/client.rb +26 -18
- data/lib/flock/gen-rb/flock_d_b.rb +126 -0
- data/lib/flock/gen-rb/flockdb_types.rb +22 -0
- data/lib/flock/mixins/sizeable.rb +2 -6
- data/lib/flock/mock_service.rb +139 -74
- data/lib/flock/operations/execute_operation.rb +1 -1
- data/lib/flock/operations/execute_operations.rb +6 -1
- data/lib/flock/operations/query_term.rb +51 -7
- data/lib/flock/operations/select_operation.rb +0 -1
- data/lib/flock/operations/select_operations.rb +97 -0
- data/lib/flock/operations/simple_operation.rb +17 -2
- data/lib/flock/service.rb +1 -1
- data/lib/flock/version.rb +3 -0
- data/spec/execute_operations_spec.rb +4 -5
- data/spec/flock_spec.rb +114 -0
- data/spec/query_term_spec.rb +14 -2
- data/spec/simple_operation_spec.rb +2 -3
- data/spec/spec_helper.rb +1 -1
- metadata +68 -16
- data/VERSION +0 -1
@@ -103,6 +103,28 @@ module Flock
|
|
103
103
|
::Thrift::Struct.generate_accessors self
|
104
104
|
end
|
105
105
|
|
106
|
+
class Metadata
|
107
|
+
include ::Thrift::Struct, ::Thrift::Struct_Union
|
108
|
+
SOURCE_ID = 1
|
109
|
+
STATE_ID = 2
|
110
|
+
COUNT = 3
|
111
|
+
UPDATED_AT = 4
|
112
|
+
|
113
|
+
FIELDS = {
|
114
|
+
SOURCE_ID => {:type => ::Thrift::Types::I64, :name => 'source_id'},
|
115
|
+
STATE_ID => {:type => ::Thrift::Types::I32, :name => 'state_id'},
|
116
|
+
COUNT => {:type => ::Thrift::Types::I32, :name => 'count'},
|
117
|
+
UPDATED_AT => {:type => ::Thrift::Types::I32, :name => 'updated_at'}
|
118
|
+
}
|
119
|
+
|
120
|
+
def struct_fields; FIELDS; end
|
121
|
+
|
122
|
+
def validate
|
123
|
+
end
|
124
|
+
|
125
|
+
::Thrift::Struct.generate_accessors self
|
126
|
+
end
|
127
|
+
|
106
128
|
class Edge
|
107
129
|
include ::Thrift::Struct, ::Thrift::Struct_Union
|
108
130
|
SOURCE_ID = 1
|
data/lib/flock/mock_service.rb
CHANGED
@@ -7,24 +7,30 @@ module Flock
|
|
7
7
|
Edges::ExecuteOperationType::Archive => :archive,
|
8
8
|
Edges::ExecuteOperationType::Negate => :negate
|
9
9
|
}
|
10
|
+
OP_COLOR_MAP = {:add => 0, :remove => 1, :archive => 2, :negate => 3}
|
10
11
|
|
11
12
|
attr_accessor :timeout, :fixtures
|
12
13
|
|
13
14
|
def clear
|
14
|
-
@
|
15
|
-
|
15
|
+
@graphs = nil
|
16
|
+
end
|
17
|
+
|
18
|
+
def load_yml(file)
|
19
|
+
@data_for_fixture ||= Hash.new do |h, k|
|
20
|
+
h[k] = YAML::load(ERB.new(File.open(k, 'r').read).result(binding)).sort
|
21
|
+
end
|
22
|
+
@data_for_fixture[file]
|
16
23
|
end
|
17
24
|
|
18
25
|
def load(fixtures = nil)
|
19
26
|
fixtures ||= self.fixtures or raise "No flock fixtures specified. either pass fixtures to load, or set Flock::MockService.fixtures."
|
20
|
-
|
21
27
|
clear
|
22
28
|
|
23
29
|
fixtures.each do |fixture|
|
24
30
|
file, graph, source, dest = fixture.values_at(:file, :graph, :source, :destination)
|
25
31
|
|
26
|
-
|
27
|
-
|
32
|
+
load_yml(file).each do |key, row|
|
33
|
+
color_node(row[source], graph, row[dest], OP_COLOR_MAP[:add])
|
28
34
|
end
|
29
35
|
end
|
30
36
|
end
|
@@ -37,21 +43,52 @@ module Flock
|
|
37
43
|
operations = operations.operations
|
38
44
|
operations.each do |operation|
|
39
45
|
term = operation.term
|
46
|
+
position = operation.position
|
40
47
|
graph, source = term.graph_id, term.source_id
|
41
|
-
|
48
|
+
destinations = term.destination_ids && term.destination_ids.unpack('Q*')
|
49
|
+
dest_state = OP_COLOR_MAP[EXEC_OPS[operation.operation_type]]
|
42
50
|
|
43
|
-
source,
|
44
|
-
|
45
|
-
self.send(EXEC_OPS[operation.operation_type], source, graph, dest)
|
51
|
+
source, destinations = destinations, source unless term.is_forward
|
52
|
+
color_node(source, graph, destinations, dest_state, position)
|
46
53
|
end
|
47
54
|
end
|
48
55
|
|
49
56
|
def select(select_operations, page)
|
50
|
-
|
57
|
+
data, next_cursor, prev_cursor = paginate(select_query(select_operations), page)
|
58
|
+
|
59
|
+
result = Flock::Results.new
|
60
|
+
result.ids = data.pack("Q*")
|
61
|
+
result.next_cursor = next_cursor
|
62
|
+
result.prev_cursor = prev_cursor
|
63
|
+
result
|
64
|
+
end
|
65
|
+
|
66
|
+
def select2(queries)
|
67
|
+
queries.map do |query|
|
68
|
+
select(query.operations, query.page)
|
69
|
+
end
|
70
|
+
end
|
71
|
+
|
72
|
+
def select_edges(queries)
|
73
|
+
queries.map do |query|
|
74
|
+
edges, next_cursor, prev_cursor = paginate(simple_query(query.term), query.page)
|
75
|
+
result = Edges::EdgeResults.new
|
76
|
+
result.edges = if query.term.is_forward
|
77
|
+
edges.map(&:dup)
|
78
|
+
else
|
79
|
+
edges.map(&:dup).map do |edge|
|
80
|
+
edge.source_id, edge.destination_id = edge.destination_id, edge.source_id
|
81
|
+
edge
|
82
|
+
end
|
83
|
+
end
|
84
|
+
result.next_cursor = next_cursor
|
85
|
+
result.prev_cursor = prev_cursor
|
86
|
+
result
|
87
|
+
end
|
51
88
|
end
|
52
89
|
|
53
90
|
def contains(source, graph, dest)
|
54
|
-
|
91
|
+
graphs(graph).contains?(source, dest, Edges::EdgeState::Positive)
|
55
92
|
end
|
56
93
|
|
57
94
|
def count(select_operations)
|
@@ -66,10 +103,13 @@ module Flock
|
|
66
103
|
case select_operation.operation_type
|
67
104
|
when Edges::SelectOperationType::SimpleQuery
|
68
105
|
term = select_operation.term
|
69
|
-
|
70
|
-
|
71
|
-
|
72
|
-
|
106
|
+
matching_edges = simple_query(term)
|
107
|
+
ids = if term.is_forward
|
108
|
+
matching_edges.map(&:destination_id)
|
109
|
+
else
|
110
|
+
matching_edges.map(&:source_id)
|
111
|
+
end
|
112
|
+
stack.push(ids)
|
73
113
|
when Edges::SelectOperationType::Intersection
|
74
114
|
stack.push(stack.pop & stack.pop)
|
75
115
|
when Edges::SelectOperationType::Union
|
@@ -83,7 +123,18 @@ module Flock
|
|
83
123
|
stack.pop
|
84
124
|
end
|
85
125
|
|
86
|
-
def
|
126
|
+
def simple_query(term)
|
127
|
+
destination_ids = term.destination_ids && term.destination_ids.unpack("Q*")
|
128
|
+
states = term.state_ids || [Edges::EdgeState::Positive]
|
129
|
+
graph = graphs(term.graph_id)
|
130
|
+
if term.is_forward
|
131
|
+
graph.select_edges(term.source_id, destination_ids, states)
|
132
|
+
else
|
133
|
+
graph.select_edges(destination_ids, term.source_id, states)
|
134
|
+
end
|
135
|
+
end
|
136
|
+
|
137
|
+
def paginate(data, page)
|
87
138
|
return empty_result if page.cursor == Flock::CursorEnd
|
88
139
|
|
89
140
|
start =
|
@@ -95,13 +146,8 @@ module Flock
|
|
95
146
|
rv = data.slice(start, page.count)
|
96
147
|
next_cursor = (start + page.count >= data.size) ? Flock::CursorEnd : start + page.count
|
97
148
|
prev_cursor = (start <= 0) ? Flock::CursorEnd : -start
|
98
|
-
|
99
|
-
|
100
|
-
result.ids = Array(rv).pack("Q*")
|
101
|
-
result.next_cursor = next_cursor
|
102
|
-
result.prev_cursor = prev_cursor
|
103
|
-
result
|
104
|
-
end
|
149
|
+
[Array(rv), next_cursor, prev_cursor]
|
150
|
+
end
|
105
151
|
|
106
152
|
def empty_result
|
107
153
|
@empty_result ||=
|
@@ -114,81 +160,100 @@ module Flock
|
|
114
160
|
end
|
115
161
|
end
|
116
162
|
|
117
|
-
|
118
|
-
|
119
|
-
|
163
|
+
def graphs(graph_id)
|
164
|
+
(@graphs ||= Hash.new do |h, k|
|
165
|
+
h[k] = Graph.new
|
166
|
+
end)[graph_id]
|
120
167
|
end
|
121
168
|
|
122
|
-
def new_edges_hash
|
123
|
-
Hash.new { |h,k| h[k] = Hash.new { |h,k| h[k] = Hash.new { |h2,k2| h2[k2] = [] } } }
|
124
|
-
end
|
125
169
|
|
126
|
-
def
|
127
|
-
|
128
|
-
|
129
|
-
|
130
|
-
def add_edge(source, graph, dest, state)
|
131
|
-
add_row(forward_edges[graph][state], source, dest)
|
132
|
-
add_row(backward_edges[graph][state], dest, source)
|
133
|
-
[source, graph, dest]
|
134
|
-
end
|
170
|
+
def color_node(source, graph_id, dest, dest_state, position = nil)
|
171
|
+
raise ArgumentError unless Edges::EdgeState::VALUE_MAP.keys.include? dest_state
|
172
|
+
raise ArgumentError if source.nil? && dest.nil?
|
173
|
+
position ||= Time.now.to_i
|
135
174
|
|
136
|
-
|
137
|
-
|
138
|
-
|
175
|
+
if source.nil? || dest.nil?
|
176
|
+
existing_edges = graphs(graph_id).select_edges(source, dest)
|
177
|
+
existing_edges.each do |existing_edge|
|
178
|
+
graphs(graph_id).add_edge(dest_state, existing_edge.source_id, existing_edge.destination_id, Time.now, position)
|
179
|
+
end
|
180
|
+
else
|
181
|
+
Array(source).each do |s|
|
182
|
+
Array(dest).each do |d|
|
183
|
+
graphs(graph_id).add_edge(dest_state, s, d, Time.now, position)
|
184
|
+
end
|
185
|
+
end
|
139
186
|
end
|
140
187
|
end
|
141
188
|
|
142
|
-
|
143
|
-
forward = remove_row(forward_edges[graph][state], source, dest)
|
144
|
-
backward = remove_row(backward_edges[graph][state], dest, source)
|
189
|
+
end
|
145
190
|
|
146
|
-
|
191
|
+
class Graph
|
192
|
+
def initialize
|
193
|
+
@by_pair, @by_source, @by_destination = {}, Hash.new { |h, k| h[k] = [] }, Hash.new { |h, k| h[k] = [] }
|
147
194
|
end
|
148
195
|
|
149
|
-
def
|
150
|
-
|
151
|
-
|
152
|
-
|
153
|
-
|
154
|
-
|
155
|
-
|
156
|
-
|
157
|
-
|
158
|
-
|
159
|
-
|
196
|
+
def add_edge(state, source, dest, time, pos)
|
197
|
+
if existing_edge = @by_pair[[source, dest]]
|
198
|
+
if ![Edges::EdgeState::Positive, Edges::EdgeState::Archived].include?(existing_edge.state_id) && state == Edges::EdgeState::Positive
|
199
|
+
existing_edge.position = pos
|
200
|
+
end
|
201
|
+
existing_edge.state_id = state
|
202
|
+
existing_edge.updated_at = time.to_i
|
203
|
+
else
|
204
|
+
edge = make_edge(state, source, dest, time, pos)
|
205
|
+
@by_pair[[source, dest]] = edge
|
206
|
+
@by_source[source] << edge
|
207
|
+
@by_destination[dest] << edge
|
208
|
+
end
|
160
209
|
end
|
161
210
|
|
162
|
-
def
|
163
|
-
|
164
|
-
|
165
|
-
|
211
|
+
def select_edges(source_ids, destination_ids, states = [])
|
212
|
+
source_ids, destination_ids, states = Array(source_ids), Array(destination_ids), Array(states)
|
213
|
+
result = []
|
214
|
+
if source_ids.empty?
|
215
|
+
destination_ids.each do |destination_id|
|
216
|
+
@by_destination[destination_id].each do |edge|
|
217
|
+
next unless states.empty? || states.include?(edge.state_id)
|
166
218
|
|
167
|
-
|
168
|
-
source_states.each do |source_state|
|
169
|
-
remove_node(source, graph, dest, source_state).tap do |deleted|
|
170
|
-
deleted.each {|source, graph, dest| add_edge(source, graph, dest, dest_state) }
|
219
|
+
result << edge
|
171
220
|
end
|
172
221
|
end
|
222
|
+
elsif destination_ids.empty?
|
223
|
+
source_ids.each do |source_id|
|
224
|
+
@by_source[source_id].each do |edge|
|
225
|
+
next unless states.empty? || states.include?(edge.state_id)
|
173
226
|
|
227
|
+
result << edge
|
228
|
+
end
|
229
|
+
end
|
174
230
|
else
|
175
|
-
|
231
|
+
source_ids.each do |source_id|
|
232
|
+
destination_ids.each do |destination_id|
|
233
|
+
next unless existing_edge = @by_pair[[source_id, destination_id]]
|
234
|
+
next unless states.empty? || states.include?(existing_edge.state_id)
|
176
235
|
|
177
|
-
|
178
|
-
dests.each do |d|
|
179
|
-
add_edge(s, graph, d, dest_state)
|
180
|
-
source_states.each { |state| remove_edge(s, graph, d, state) }
|
236
|
+
result << existing_edge
|
181
237
|
end
|
182
238
|
end
|
183
239
|
end
|
240
|
+
result
|
184
241
|
end
|
185
242
|
|
186
|
-
|
187
|
-
|
188
|
-
|
243
|
+
def contains?(source, dest, state)
|
244
|
+
select_edges(source, dest, state).any?
|
245
|
+
end
|
189
246
|
|
190
|
-
|
191
|
-
|
247
|
+
private
|
248
|
+
def make_edge(state, source, dest, time, pos)
|
249
|
+
Edges::Edge.new.tap do |edge|
|
250
|
+
edge.source_id = source
|
251
|
+
edge.destination_id = dest
|
252
|
+
edge.updated_at = time.to_i
|
253
|
+
edge.position = pos
|
254
|
+
edge.count = 1
|
255
|
+
edge.state_id = state
|
256
|
+
end
|
192
257
|
end
|
193
258
|
end
|
194
259
|
end
|
@@ -6,7 +6,12 @@ module Flock
|
|
6
6
|
|
7
7
|
Flock::Edges::ExecuteOperationType::VALUE_MAP.each do |op_id, op|
|
8
8
|
op = op.downcase
|
9
|
-
class_eval
|
9
|
+
class_eval <<-EOT, __FILE__, __LINE__ + 1
|
10
|
+
def #{op}(s, g, d, p = nil)
|
11
|
+
@operations << ExecuteOperation.new(#{op_id}, Flock::QueryTerm.new([s, g, d]), p)
|
12
|
+
self
|
13
|
+
end
|
14
|
+
EOT
|
10
15
|
end
|
11
16
|
|
12
17
|
def apply
|
@@ -1,20 +1,27 @@
|
|
1
1
|
module Flock
|
2
2
|
class QueryTerm
|
3
|
+
# symbol => state_id map
|
4
|
+
STATES = Flock::Edges::EdgeState::VALUE_MAP.inject({}) do |states, (id, name)|
|
5
|
+
states.update name.downcase.to_sym => id
|
6
|
+
end.freeze
|
7
|
+
|
3
8
|
attr_accessor :source, :graph, :destination, :states
|
4
9
|
|
5
|
-
def initialize(query)
|
6
|
-
|
7
|
-
|
8
|
-
|
9
|
-
|
10
|
-
raise ArgumentError
|
11
|
-
end
|
10
|
+
def initialize(query, graphs = nil)
|
11
|
+
raise ArgumentError if query.size < 3
|
12
|
+
@graphs = graphs
|
13
|
+
|
14
|
+
@source, @graph, @destination, @states = *query_args(query)
|
12
15
|
end
|
13
16
|
|
14
17
|
def forward?
|
15
18
|
@source.is_a? Numeric
|
16
19
|
end
|
17
20
|
|
21
|
+
def unapply
|
22
|
+
[@source, @graph, @destination, @states]
|
23
|
+
end
|
24
|
+
|
18
25
|
def to_thrift
|
19
26
|
term = Edges::QueryTerm.new
|
20
27
|
term.graph_id = @graph
|
@@ -33,5 +40,42 @@ module Flock
|
|
33
40
|
|
34
41
|
term
|
35
42
|
end
|
43
|
+
|
44
|
+
def ==(other)
|
45
|
+
self.source == other.source &&
|
46
|
+
self.graph == other.graph &&
|
47
|
+
self.destination == other.destination &&
|
48
|
+
self.states == other.states
|
49
|
+
end
|
50
|
+
|
51
|
+
private
|
52
|
+
|
53
|
+
def lookup_graph(key)
|
54
|
+
if @graphs.nil? or key.is_a? Integer
|
55
|
+
key
|
56
|
+
else
|
57
|
+
@graphs[key] or raise UnknownGraphError.new(key)
|
58
|
+
end
|
59
|
+
end
|
60
|
+
|
61
|
+
def lookup_states(states)
|
62
|
+
states = states.flatten.compact.map do |s|
|
63
|
+
if s.is_a? Integer
|
64
|
+
s
|
65
|
+
else
|
66
|
+
STATES[s] or raise UnknownStateError.new(s)
|
67
|
+
end
|
68
|
+
end
|
69
|
+
end
|
70
|
+
|
71
|
+
def query_args(args)
|
72
|
+
source, graph, destination, *states = *((args.length == 1) ? args.first : args)
|
73
|
+
[node_arg(source), lookup_graph(graph), node_arg(destination), lookup_states(states)]
|
74
|
+
end
|
75
|
+
|
76
|
+
def node_arg(node)
|
77
|
+
return node.map {|n| n.to_i if n } if node.respond_to? :map
|
78
|
+
node.to_i if node
|
79
|
+
end
|
36
80
|
end
|
37
81
|
end
|