eb_nested_set 0.3.3
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/LICENSE +20 -0
- data/README.md +41 -0
- data/Rakefile +87 -0
- data/init.rb +1 -0
- data/lib/eb_nested_set.rb +349 -0
- data/spec/db/test.sqlite3 +0 -0
- data/spec/directory_spec.rb +54 -0
- data/spec/employee_spec.rb +74 -0
- data/spec/nested_set_behavior.rb +586 -0
- data/spec/spec_helper.rb +83 -0
- metadata +64 -0
data/LICENSE
ADDED
@@ -0,0 +1,20 @@
|
|
1
|
+
Copyright (c) 2008 Jonas Nicklas
|
2
|
+
|
3
|
+
Permission is hereby granted, free of charge, to any person obtaining
|
4
|
+
a copy of this software and associated documentation files (the
|
5
|
+
"Software"), to deal in the Software without restriction, including
|
6
|
+
without limitation the rights to use, copy, modify, merge, publish,
|
7
|
+
distribute, sublicense, and/or sell copies of the Software, and to
|
8
|
+
permit persons to whom the Software is furnished to do so, subject to
|
9
|
+
the following conditions:
|
10
|
+
|
11
|
+
The above copyright notice and this permission notice shall be
|
12
|
+
included in all copies or substantial portions of the Software.
|
13
|
+
|
14
|
+
THE SOFTWARE IS PROVIDED "AS IS", WITHOUT WARRANTY OF ANY KIND,
|
15
|
+
EXPRESS OR IMPLIED, INCLUDING BUT NOT LIMITED TO THE WARRANTIES OF
|
16
|
+
MERCHANTABILITY, FITNESS FOR A PARTICULAR PURPOSE AND
|
17
|
+
NONINFRINGEMENT. IN NO EVENT SHALL THE AUTHORS OR COPYRIGHT HOLDERS BE
|
18
|
+
LIABLE FOR ANY CLAIM, DAMAGES OR OTHER LIABILITY, WHETHER IN AN ACTION
|
19
|
+
OF CONTRACT, TORT OR OTHERWISE, ARISING FROM, OUT OF OR IN CONNECTION
|
20
|
+
WITH THE SOFTWARE OR THE USE OR OTHER DEALINGS IN THE SOFTWARE.
|
data/README.md
ADDED
@@ -0,0 +1,41 @@
|
|
1
|
+
# EvenBetterNestedSet
|
2
|
+
|
3
|
+
A nested set is a datastruture in a database, sort of like a tree, but unlike a tree it allows you to find all descendants of a node with a single query. Loading a deeply nested structure with nested sets is therefore a lot more efficient than using a tree. So what's the disadvantage? Nested sets are a lot harder to maintain, since inserting and moving records requires management and it is easy to corrupt the dataset. Enter: EvenBetterNestedSet. Amount of micromanaging you need to do: 0. EvenBetterNestedSet does it all for you.
|
4
|
+
|
5
|
+
## Declaring nested sets
|
6
|
+
|
7
|
+
This is how you declare a nested set:
|
8
|
+
|
9
|
+
class Directory < ActiveRecord::Base
|
10
|
+
|
11
|
+
acts_as_nested_set
|
12
|
+
|
13
|
+
end
|
14
|
+
|
15
|
+
The directories table should have the columns 'parent_id', 'left' and 'right'.
|
16
|
+
|
17
|
+
Now just set the parent to wherever you want your node to be located and EvenBetterNestedSet will do the rest for you.
|
18
|
+
|
19
|
+
d = Directory.new
|
20
|
+
|
21
|
+
d.children.create!(:name => 'blah')
|
22
|
+
d.children.create!(:name => 'gurr')
|
23
|
+
d.children.create!(:name => 'doh')
|
24
|
+
|
25
|
+
d.bounds #=> 1..8
|
26
|
+
d.children[1].bounds #=> 4..5
|
27
|
+
d.children[1].name #=> 'gurr'
|
28
|
+
d.children[1].parent #=> d
|
29
|
+
|
30
|
+
c = Directory.create!(:name => 'test', :parent => d.directory[1]
|
31
|
+
|
32
|
+
## Finding with nested sets
|
33
|
+
|
34
|
+
EvenBetterNestedSet will not automatically cache children for you, because it assumes that this is not always the preferred behaviour. If you want to cache children to a nested set, just do:
|
35
|
+
|
36
|
+
d = Directory.find(42)
|
37
|
+
d.cache_nested_set
|
38
|
+
|
39
|
+
or more conveniently:
|
40
|
+
|
41
|
+
d = Directory.find_with_nested_set(42)
|
data/Rakefile
ADDED
@@ -0,0 +1,87 @@
|
|
1
|
+
require 'rubygems'
|
2
|
+
require 'rake/gempackagetask'
|
3
|
+
require 'rubygems/specification'
|
4
|
+
require 'date'
|
5
|
+
require 'spec/rake/spectask'
|
6
|
+
|
7
|
+
GEM = "eb_nested_set"
|
8
|
+
GEM_VERSION = "0.3.3"
|
9
|
+
AUTHOR = "Jonas Nicklas"
|
10
|
+
EMAIL = "jonas.nicklas@gmail.com"
|
11
|
+
HOMEPAGE = "http://github.com/jnicklas/even_better_nested_set/tree/master"
|
12
|
+
SUMMARY = "A cool acts_as_nested_set alternative"
|
13
|
+
|
14
|
+
spec = Gem::Specification.new do |s|
|
15
|
+
s.name = GEM
|
16
|
+
s.version = GEM_VERSION
|
17
|
+
s.platform = Gem::Platform::RUBY
|
18
|
+
s.has_rdoc = true
|
19
|
+
s.extra_rdoc_files = ['README.md', 'LICENSE']
|
20
|
+
s.summary = SUMMARY
|
21
|
+
s.description = s.summary
|
22
|
+
s.author = AUTHOR
|
23
|
+
s.email = EMAIL
|
24
|
+
s.homepage = HOMEPAGE
|
25
|
+
s.require_path = 'lib'
|
26
|
+
s.autorequire = GEM
|
27
|
+
s.files = %w(LICENSE README.md Rakefile init.rb) + Dir.glob("{lib,spec}/**/*")
|
28
|
+
end
|
29
|
+
|
30
|
+
Rake::GemPackageTask.new(spec) do |pkg|
|
31
|
+
pkg.gem_spec = spec
|
32
|
+
end
|
33
|
+
|
34
|
+
desc "install the plugin locally"
|
35
|
+
task :install => [:package] do
|
36
|
+
sh %{sudo gem install pkg/#{GEM}-#{GEM_VERSION} --no-update-sources}
|
37
|
+
end
|
38
|
+
|
39
|
+
desc "create a gemspec file"
|
40
|
+
task :make_spec do
|
41
|
+
File.open("#{GEM}.gemspec", "w") do |file|
|
42
|
+
file.puts spec.to_ruby
|
43
|
+
end
|
44
|
+
end
|
45
|
+
|
46
|
+
namespace :jruby do
|
47
|
+
|
48
|
+
desc "Run :package and install the resulting .gem with jruby"
|
49
|
+
task :install => :package do
|
50
|
+
sh %{#{SUDO} jruby -S gem install pkg/#{GEM}-#{GEM_VERSION}.gem --no-rdoc --no-ri}
|
51
|
+
end
|
52
|
+
|
53
|
+
end
|
54
|
+
|
55
|
+
spec_files = FileList['spec/*_spec.rb']
|
56
|
+
|
57
|
+
desc 'Default: run unit tests.'
|
58
|
+
task :default => 'spec'
|
59
|
+
|
60
|
+
task :specs => :spec
|
61
|
+
desc "Run all examples"
|
62
|
+
Spec::Rake::SpecTask.new('spec') do |t|
|
63
|
+
t.spec_opts = ['--color']
|
64
|
+
t.spec_files = spec_files
|
65
|
+
end
|
66
|
+
|
67
|
+
namespace :spec do
|
68
|
+
desc "Run all examples with RCov"
|
69
|
+
Spec::Rake::SpecTask.new('rcov') do |t|
|
70
|
+
t.spec_files = spec_files
|
71
|
+
t.rcov = true
|
72
|
+
t.rcov_dir = "doc/coverage"
|
73
|
+
t.rcov_opts = ['--exclude', 'spec,rspec-*,rcov-*,gems']
|
74
|
+
t.spec_opts = ['--color']
|
75
|
+
end
|
76
|
+
|
77
|
+
desc "Generate an html report"
|
78
|
+
Spec::Rake::SpecTask.new('report') do |t|
|
79
|
+
t.spec_files = spec_files
|
80
|
+
t.rcov = true
|
81
|
+
t.rcov_dir = "doc/coverage"
|
82
|
+
t.rcov_opts = ['--exclude', 'spec']
|
83
|
+
t.spec_opts = ['--color', "--format", "html:doc/reports/specs.html"]
|
84
|
+
t.fail_on_error = false
|
85
|
+
end
|
86
|
+
|
87
|
+
end
|
data/init.rb
ADDED
@@ -0,0 +1 @@
|
|
1
|
+
require File.join(File.dirname(__FILE__), 'lib', 'even_better_nested_set')
|
@@ -0,0 +1,349 @@
|
|
1
|
+
module EvenBetterNestedSet
|
2
|
+
|
3
|
+
def self.included(base)
|
4
|
+
super
|
5
|
+
base.extend ClassMethods
|
6
|
+
end
|
7
|
+
|
8
|
+
class NestedSetError < StandardError; end
|
9
|
+
class IllegalAssignmentError < NestedSetError; end
|
10
|
+
|
11
|
+
module NestedSet
|
12
|
+
|
13
|
+
def self.included(base)
|
14
|
+
super
|
15
|
+
base.extend ClassMethods
|
16
|
+
end
|
17
|
+
|
18
|
+
module ClassMethods
|
19
|
+
|
20
|
+
attr_accessor :nested_set_options
|
21
|
+
|
22
|
+
def find_last_root
|
23
|
+
find(:first, :order => "#{nested_set_column(:right)} DESC", :conditions => { :parent_id => nil })
|
24
|
+
end
|
25
|
+
|
26
|
+
def find_boundaries(id)
|
27
|
+
query = "SELECT #{nested_set_column(:left)}, #{nested_set_column(:right)}" +
|
28
|
+
"FROM #{quote_db_property(table_name)}" +
|
29
|
+
"WHERE #{quote_db_property(primary_key)} = #{id}"
|
30
|
+
connection.select_rows(query).first
|
31
|
+
end
|
32
|
+
|
33
|
+
def nested_set
|
34
|
+
sort_nodes_to_nested_set(find(:all, :order => "#{nested_set_column(:left)} ASC"))
|
35
|
+
end
|
36
|
+
|
37
|
+
def find_with_nested_set(*args)
|
38
|
+
result = find(*args)
|
39
|
+
if result.respond_to?(:cache_nested_set)
|
40
|
+
result.cache_nested_set
|
41
|
+
elsif result.respond_to?(:each)
|
42
|
+
result.each do |node|
|
43
|
+
node.cache_nested_set
|
44
|
+
end
|
45
|
+
end
|
46
|
+
result
|
47
|
+
end
|
48
|
+
|
49
|
+
def sort_nodes_to_nested_set(nodes)
|
50
|
+
roots = []
|
51
|
+
hashmap = {}
|
52
|
+
for node in nodes.sort_by { |n| n.left }
|
53
|
+
# if the parent is not in the hashmap, parent will be nil, therefore node will be a root node
|
54
|
+
# in that case
|
55
|
+
parent = node.parent_id ? hashmap[node.parent_id] : nil
|
56
|
+
|
57
|
+
# make sure this is called at least once on every node, so leaves know that they have *no* children
|
58
|
+
node.cache_children()
|
59
|
+
|
60
|
+
if parent
|
61
|
+
node.cache_parent(parent)
|
62
|
+
parent.cache_children(node)
|
63
|
+
else
|
64
|
+
roots << node
|
65
|
+
end
|
66
|
+
|
67
|
+
hashmap[node.id] = node
|
68
|
+
end
|
69
|
+
return roots
|
70
|
+
end
|
71
|
+
|
72
|
+
def nested_set_column(name)
|
73
|
+
quote_db_property(nested_set_options[name])
|
74
|
+
end
|
75
|
+
|
76
|
+
# Recalculates the left and right values for the entire tree
|
77
|
+
def recalculate_nested_set
|
78
|
+
transaction do
|
79
|
+
left = 1
|
80
|
+
roots.each do |root|
|
81
|
+
left = root.recalculate_nested_set(left)
|
82
|
+
end
|
83
|
+
end
|
84
|
+
end
|
85
|
+
|
86
|
+
def quote_db_property(property)
|
87
|
+
"`#{property}`".gsub('.','`.`')
|
88
|
+
end
|
89
|
+
|
90
|
+
end
|
91
|
+
|
92
|
+
def root?
|
93
|
+
not parent_id?
|
94
|
+
end
|
95
|
+
|
96
|
+
def descendant_of?(node)
|
97
|
+
node.left < self.left && self.right < node.right
|
98
|
+
end
|
99
|
+
|
100
|
+
def root
|
101
|
+
transaction do
|
102
|
+
reload_boundaries
|
103
|
+
@root ||= base_class.roots.find(:first, :conditions => ["#{nested_set_column(:left)} <= ? AND #{nested_set_column(:right)} >= ?", left, right])
|
104
|
+
end
|
105
|
+
end
|
106
|
+
|
107
|
+
alias_method :patriarch, :root
|
108
|
+
|
109
|
+
def ancestors(force_reload=false)
|
110
|
+
@ancestors = nil if force_reload
|
111
|
+
@ancestors ||= base_class.find(
|
112
|
+
:all,:conditions => ["#{nested_set_column(:left)} < ? AND #{nested_set_column(:right)} > ?", left, right],
|
113
|
+
:order => "#{nested_set_column(:left)} DESC"
|
114
|
+
)
|
115
|
+
end
|
116
|
+
|
117
|
+
def lineage(force_reload=false)
|
118
|
+
[self, *ancestors(force_reload)]
|
119
|
+
end
|
120
|
+
|
121
|
+
def kin
|
122
|
+
patriarch.family
|
123
|
+
end
|
124
|
+
|
125
|
+
def descendants
|
126
|
+
base_class.descendants(self)
|
127
|
+
end
|
128
|
+
|
129
|
+
def cache_nested_set
|
130
|
+
@cached_children || base_class.sort_nodes_to_nested_set(family)
|
131
|
+
end
|
132
|
+
|
133
|
+
def family
|
134
|
+
[self, *descendants]
|
135
|
+
end
|
136
|
+
|
137
|
+
def family_ids(force_reload=false)
|
138
|
+
return @family_ids unless @family_ids.nil? or force_reload
|
139
|
+
|
140
|
+
transaction do
|
141
|
+
reload_boundaries
|
142
|
+
query = "SELECT id FROM #{self.class.quote_db_property(base_class.table_name)} " +
|
143
|
+
"WHERE #{nested_set_column(:left)} >= #{left} AND #{nested_set_column(:right)} <= #{right} " +
|
144
|
+
"ORDER BY #{nested_set_column(:left)}"
|
145
|
+
@family_ids = base_class.connection.select_values(query).map(&:to_i)
|
146
|
+
end
|
147
|
+
end
|
148
|
+
|
149
|
+
def generation
|
150
|
+
root? ? base_class.roots : parent.children
|
151
|
+
end
|
152
|
+
|
153
|
+
def siblings
|
154
|
+
generation - [self]
|
155
|
+
end
|
156
|
+
|
157
|
+
def level
|
158
|
+
if root?
|
159
|
+
0
|
160
|
+
elsif @ancestors
|
161
|
+
@ancestors.size
|
162
|
+
else
|
163
|
+
base_class.count :conditions => ["#{nested_set_column(:left)} < ? AND #{nested_set_column(:right)} > ?", left, right]
|
164
|
+
end
|
165
|
+
end
|
166
|
+
|
167
|
+
def bounds
|
168
|
+
left..right
|
169
|
+
end
|
170
|
+
|
171
|
+
def children
|
172
|
+
@cached_children || uncached_children
|
173
|
+
end
|
174
|
+
|
175
|
+
def children?
|
176
|
+
children.empty?
|
177
|
+
end
|
178
|
+
|
179
|
+
def cache_parent(parent) #:nodoc:
|
180
|
+
self.parent = parent
|
181
|
+
end
|
182
|
+
|
183
|
+
def cache_children(*nodes) #:nodoc:
|
184
|
+
@cached_children ||= []
|
185
|
+
@cached_children.push(*nodes)
|
186
|
+
end
|
187
|
+
|
188
|
+
def left
|
189
|
+
read_attribute(self.class.nested_set_options[:left])
|
190
|
+
end
|
191
|
+
|
192
|
+
def right
|
193
|
+
read_attribute(self.class.nested_set_options[:right])
|
194
|
+
end
|
195
|
+
|
196
|
+
def recalculate_nested_set(left)
|
197
|
+
child_left = left + 1
|
198
|
+
children.each do |child|
|
199
|
+
child_left = child.recalculate_nested_set(child_left)
|
200
|
+
end
|
201
|
+
set_boundaries(left, child_left)
|
202
|
+
save_without_validation!
|
203
|
+
|
204
|
+
right + 1
|
205
|
+
end
|
206
|
+
|
207
|
+
protected
|
208
|
+
|
209
|
+
def illegal_nesting
|
210
|
+
if parent_id? and family_ids.include?(parent_id)
|
211
|
+
errors.add(:parent_id, 'cannot move node to its own descendant')
|
212
|
+
end
|
213
|
+
end
|
214
|
+
|
215
|
+
def remove_node
|
216
|
+
base_class.delete_all ["#{nested_set_column(:left)} > ? AND #{nested_set_column(:right)} < ?", left, right] # TODO: Figure out what to do with children's destroy callbacks
|
217
|
+
|
218
|
+
shift!(-node_width, right)
|
219
|
+
end
|
220
|
+
|
221
|
+
def append_node
|
222
|
+
boundary = 1
|
223
|
+
|
224
|
+
if parent_id?
|
225
|
+
transaction do
|
226
|
+
boundary = parent(true).right
|
227
|
+
shift! 2, boundary
|
228
|
+
end
|
229
|
+
elsif last_root = base_class.find_last_root
|
230
|
+
boundary = last_root.right + 1
|
231
|
+
end
|
232
|
+
|
233
|
+
set_boundaries(boundary, boundary + 1)
|
234
|
+
end
|
235
|
+
|
236
|
+
def move_node
|
237
|
+
if parent_id_changed?
|
238
|
+
transaction do
|
239
|
+
reload_boundaries
|
240
|
+
|
241
|
+
if parent_id.blank? # moved to root
|
242
|
+
shift_difference = base_class.find_last_root.right - left + 1
|
243
|
+
else # moved to non-root
|
244
|
+
new_parent = base_class.find_by_id(parent_id)
|
245
|
+
|
246
|
+
# open up a space
|
247
|
+
boundary = new_parent.right
|
248
|
+
shift! node_width, boundary
|
249
|
+
|
250
|
+
reload_boundaries
|
251
|
+
|
252
|
+
shift_difference = (new_parent.right - left)
|
253
|
+
end
|
254
|
+
# move itself and children into place
|
255
|
+
shift! shift_difference, left, right
|
256
|
+
|
257
|
+
# close up the space that was left behind after move
|
258
|
+
shift! -node_width, left
|
259
|
+
|
260
|
+
reload_boundaries
|
261
|
+
end
|
262
|
+
end
|
263
|
+
end
|
264
|
+
|
265
|
+
def shift!(positions, left_boundary, right_boundary=nil)
|
266
|
+
if right_boundary
|
267
|
+
base_class.update_all "#{nested_set_column(:left)} = (#{nested_set_column(:left)} + #{positions})", ["#{nested_set_column(:left)} >= ? AND #{nested_set_column(:left)} <= ?", left_boundary, right_boundary]
|
268
|
+
base_class.update_all "#{nested_set_column(:right)} = (#{nested_set_column(:right)} + #{positions})", ["#{nested_set_column(:right)} >= ? AND #{nested_set_column(:right)} <= ?", left_boundary, right_boundary]
|
269
|
+
else
|
270
|
+
base_class.update_all "#{nested_set_column(:left)} = (#{nested_set_column(:left)} + #{positions})", ["#{nested_set_column(:left)} >= ?", left_boundary]
|
271
|
+
base_class.update_all "#{nested_set_column(:right)} = (#{nested_set_column(:right)} + #{positions})", ["#{nested_set_column(:right)} >= ?", left_boundary]
|
272
|
+
end
|
273
|
+
end
|
274
|
+
|
275
|
+
def node_width
|
276
|
+
right - left + 1
|
277
|
+
end
|
278
|
+
|
279
|
+
def set_boundaries(left, right)
|
280
|
+
write_attribute(self.class.nested_set_options[:left], left)
|
281
|
+
write_attribute(self.class.nested_set_options[:right], right)
|
282
|
+
end
|
283
|
+
|
284
|
+
def reload_boundaries
|
285
|
+
set_boundaries(*base_class.find_boundaries(id))
|
286
|
+
end
|
287
|
+
|
288
|
+
def base_class
|
289
|
+
self.class.base_class
|
290
|
+
end
|
291
|
+
|
292
|
+
def validate_parent_is_within_scope
|
293
|
+
if self.class.nested_set_options[:scope] && parent_id
|
294
|
+
parent.reload # Make sure we are testing the record corresponding to the parent_id
|
295
|
+
if self.send(self.class.nested_set_options[:scope]) != parent.send(self.class.nested_set_options[:scope])
|
296
|
+
errors.add(:parent_id, "cannot be a record with a different #{self.class.nested_set_options[:scope]} to this record")
|
297
|
+
end
|
298
|
+
end
|
299
|
+
end
|
300
|
+
end
|
301
|
+
|
302
|
+
module ClassMethods
|
303
|
+
|
304
|
+
def acts_as_nested_set(options = {})
|
305
|
+
options = { :left => :left, :right => :right }.merge!(options)
|
306
|
+
options[:scope] = "#{options[:scope]}_id" if options[:scope]
|
307
|
+
|
308
|
+
include NestedSet
|
309
|
+
|
310
|
+
self.nested_set_options = options
|
311
|
+
|
312
|
+
class_eval <<-RUBY, __FILE__, __LINE__+1
|
313
|
+
def #{options[:left]}=(left)
|
314
|
+
raise EvenBetterNestedSet::IllegalAssignmentError, "#{options[:left]} is an internal attribute used by EvenBetterNestedSet, do not assign it directly as is may corrupt the data in your database"
|
315
|
+
end
|
316
|
+
|
317
|
+
def #{options[:right]}=(right)
|
318
|
+
raise EvenBetterNestedSet::IllegalAssignmentError, "#{options[:right]} is an internal attribute used by EvenBetterNestedSet, do not assign it directly as is may corrupt the data in your database"
|
319
|
+
end
|
320
|
+
RUBY
|
321
|
+
|
322
|
+
named_scope :roots, :conditions => { :parent_id => nil }, :order => "#{nested_set_column(:left)} asc"
|
323
|
+
|
324
|
+
has_many :uncached_children, :class_name => self.name, :foreign_key => :parent_id, :order => "#{nested_set_column(:left)} asc"
|
325
|
+
protected :uncached_children, :uncached_children=
|
326
|
+
|
327
|
+
belongs_to :parent, :class_name => self.name, :foreign_key => :parent_id
|
328
|
+
|
329
|
+
named_scope :descendants, lambda { |node|
|
330
|
+
left, right = find_boundaries(node.id)
|
331
|
+
{ :conditions => ["#{nested_set_column(:left)} > ? and #{nested_set_column(:right)} < ?", left, right],
|
332
|
+
:order => "#{nested_set_column(:left)} asc" }
|
333
|
+
}
|
334
|
+
|
335
|
+
before_create :append_node
|
336
|
+
before_update :move_node
|
337
|
+
before_destroy :reload
|
338
|
+
after_destroy :remove_node
|
339
|
+
validate_on_update :illegal_nesting
|
340
|
+
validate :validate_parent_is_within_scope
|
341
|
+
|
342
|
+
delegate :nested_set_column, :to => "self.class"
|
343
|
+
end
|
344
|
+
|
345
|
+
end
|
346
|
+
|
347
|
+
end
|
348
|
+
|
349
|
+
ActiveRecord::Base.send(:include, EvenBetterNestedSet) if defined?(ActiveRecord)
|
Binary file
|
@@ -0,0 +1,54 @@
|
|
1
|
+
require File.dirname(__FILE__) + '/spec_helper'
|
2
|
+
require File.dirname(__FILE__) + '/nested_set_behavior'
|
3
|
+
|
4
|
+
class Directory < ActiveRecord::Base
|
5
|
+
acts_as_nested_set :left => :lft, :right => :rgt
|
6
|
+
|
7
|
+
validates_presence_of :name
|
8
|
+
end
|
9
|
+
|
10
|
+
describe Directory do
|
11
|
+
|
12
|
+
def invalid_attributes(options = {})
|
13
|
+
return { }.merge(options)
|
14
|
+
end
|
15
|
+
|
16
|
+
def valid_attributes(options = {})
|
17
|
+
$directory_no = $directory_no ? $directory_no + 1 : 0
|
18
|
+
return { :name => "directory#{$directory_no}" }.merge(options)
|
19
|
+
end
|
20
|
+
|
21
|
+
before do
|
22
|
+
@model = Directory
|
23
|
+
@instance = @model.new
|
24
|
+
end
|
25
|
+
|
26
|
+
it_should_behave_like "all nested set models"
|
27
|
+
|
28
|
+
it "should throw an error when attempting to assign lft directly" do
|
29
|
+
lambda {
|
30
|
+
@instance.lft = 42
|
31
|
+
}.should raise_error(EvenBetterNestedSet::IllegalAssignmentError)
|
32
|
+
@instance.lft.should_not == 42
|
33
|
+
end
|
34
|
+
|
35
|
+
it "should throw an error when attempting to assign rgt directly" do
|
36
|
+
lambda {
|
37
|
+
@instance.rgt = 42
|
38
|
+
}.should raise_error(EvenBetterNestedSet::IllegalAssignmentError)
|
39
|
+
@instance.rgt.should_not == 42
|
40
|
+
end
|
41
|
+
|
42
|
+
it "should throw an error when mass assigning to lft" do
|
43
|
+
lambda {
|
44
|
+
@model.new(valid_attributes(:lft => 1))
|
45
|
+
}.should raise_error(EvenBetterNestedSet::IllegalAssignmentError)
|
46
|
+
end
|
47
|
+
|
48
|
+
it "should throw an error when mass assigning to rgt" do
|
49
|
+
lambda {
|
50
|
+
@model.new(valid_attributes(:rgt => 1))
|
51
|
+
}.should raise_error(EvenBetterNestedSet::IllegalAssignmentError)
|
52
|
+
end
|
53
|
+
|
54
|
+
end
|
@@ -0,0 +1,74 @@
|
|
1
|
+
require File.dirname(__FILE__) + '/spec_helper'
|
2
|
+
require File.dirname(__FILE__) + '/nested_set_behavior'
|
3
|
+
|
4
|
+
class Employee < ActiveRecord::Base
|
5
|
+
acts_as_nested_set :scope => :company
|
6
|
+
|
7
|
+
validates_presence_of :name
|
8
|
+
end
|
9
|
+
|
10
|
+
describe Employee, "with nested sets for two different companies" do
|
11
|
+
before do
|
12
|
+
# Company 1...
|
13
|
+
Employee.with_options :company_id => 1 do |c1|
|
14
|
+
@c1_1 = c1.create!(:name => "Company 1 - 1")
|
15
|
+
@c1_2 = c1.create!(:name => "Company 1 - 2")
|
16
|
+
|
17
|
+
@c1_11 = c1.create!(:name => "Company 1 - 11", :parent => @c1_1)
|
18
|
+
@c1_12 = c1.create!(:name => "Company 1 - 12", :parent => @c1_1)
|
19
|
+
|
20
|
+
@c1_111 = c1.create!(:name => "Company 1 - 111", :parent => @c1_11)
|
21
|
+
end
|
22
|
+
|
23
|
+
# Company 2...
|
24
|
+
Employee.with_options :company_id => 2 do |c2|
|
25
|
+
@c2_1 = c2.create!(:name => "Company 2 - 1")
|
26
|
+
@c2_11 = c2.create!(:name => "Company 2 - 11", :parent => @c2_1)
|
27
|
+
end
|
28
|
+
end
|
29
|
+
|
30
|
+
after do
|
31
|
+
Employee.delete_all
|
32
|
+
end
|
33
|
+
|
34
|
+
it "should not allow a new employee in one company to be a child of an employee in the other company, when parent is assigned to" do
|
35
|
+
@employee = Employee.create(:company_id => 1, :parent => @c2_11)
|
36
|
+
@employee.errors[:parent_id].should_not be_nil
|
37
|
+
end
|
38
|
+
|
39
|
+
it "should not allow a new employee in one company to be a child of an employee in the other company, when parent_id is assigned to" do
|
40
|
+
@employee = Employee.create(:company_id => 1, :parent_id => @c2_11.id)
|
41
|
+
@employee.errors[:parent_id].should_not be_nil
|
42
|
+
end
|
43
|
+
|
44
|
+
it "should not allow an existing employee in one company to become a child of an employee in the other company, when parent is assigned to" do
|
45
|
+
@c1_11.parent = @c2_11
|
46
|
+
@c1_11.save
|
47
|
+
@c1_11.errors[:parent_id].should_not be_nil
|
48
|
+
end
|
49
|
+
|
50
|
+
it "should not allow an existing employee in one company to become a child of an employee in the other company, when parent_id is assigned to" do
|
51
|
+
@c1_11.parent_id = @c2_11.id
|
52
|
+
@c1_11.save
|
53
|
+
@c1_11.errors[:parent_id].should_not be_nil
|
54
|
+
end
|
55
|
+
|
56
|
+
it "should keep the tree for company 1 and for company 2 entirely disjoint" do
|
57
|
+
c1_tree = (@c1_1.family + @c1_2.family).flatten
|
58
|
+
c2_tree = @c2_1.family
|
59
|
+
|
60
|
+
(c1_tree & c2_tree).should be_empty
|
61
|
+
end
|
62
|
+
|
63
|
+
it "should return the correct descendants when retrieving via a database query" do
|
64
|
+
@c1_1.descendants.should == [@c1_11, @c1_111, @c1_12]
|
65
|
+
@c1_2.descendants.should == []
|
66
|
+
@c2_1.descendants.should == [@c2_11]
|
67
|
+
end
|
68
|
+
|
69
|
+
it "should return the correct levels when retrieving via a database query" do
|
70
|
+
@c1_1.family.map { |d| d.level }.should == [0, 1, 2, 1]
|
71
|
+
@c1_2.family.map { |d| d.level }.should == [0]
|
72
|
+
@c2_1.family.map { |d| d.level }.should == [0, 1]
|
73
|
+
end
|
74
|
+
end
|
@@ -0,0 +1,586 @@
|
|
1
|
+
describe "all nested set models", :shared => true do
|
2
|
+
|
3
|
+
describe @model, 'model with acts_as_nested_set' do
|
4
|
+
|
5
|
+
before do
|
6
|
+
@instance = @model.new(valid_attributes)
|
7
|
+
end
|
8
|
+
|
9
|
+
it "should change the parent_id in the database when a parent is assigned" do
|
10
|
+
without_changing_the_database do
|
11
|
+
@parent = @model.create!(valid_attributes)
|
12
|
+
|
13
|
+
@instance.parent = @parent
|
14
|
+
@instance.save!
|
15
|
+
@instance = @model.find(@instance.id)
|
16
|
+
|
17
|
+
@instance.parent_id.should == @parent.id
|
18
|
+
end
|
19
|
+
end
|
20
|
+
|
21
|
+
it "should change the parent_id in the database when a parent_id is assigned" do
|
22
|
+
without_changing_the_database do
|
23
|
+
@parent = @model.create!(valid_attributes)
|
24
|
+
|
25
|
+
@instance.parent_id = @parent.id
|
26
|
+
@instance.save!
|
27
|
+
@instance = @model.find(@instance.id)
|
28
|
+
|
29
|
+
@instance.parent_id.should == @parent.id
|
30
|
+
end
|
31
|
+
end
|
32
|
+
|
33
|
+
describe '#bounds' do
|
34
|
+
|
35
|
+
it "should return a range, from left to right" do
|
36
|
+
without_changing_the_database do
|
37
|
+
@instance.save!
|
38
|
+
@instance.left.should == 1
|
39
|
+
@instance.right.should == 2
|
40
|
+
@instance.bounds.should == (1..2)
|
41
|
+
end
|
42
|
+
end
|
43
|
+
|
44
|
+
end
|
45
|
+
|
46
|
+
end
|
47
|
+
|
48
|
+
describe @model, "with many descendants" do
|
49
|
+
before do
|
50
|
+
@r1 = @model.create!(valid_attributes)
|
51
|
+
@r2 = @model.create!(valid_attributes)
|
52
|
+
@r3 = @model.create!(valid_attributes)
|
53
|
+
|
54
|
+
@r1c1 = @model.create!(valid_attributes(:parent => @r1))
|
55
|
+
@r1c2 = @model.create!(valid_attributes(:parent => @r1))
|
56
|
+
@r1c3 = @model.create!(valid_attributes(:parent => @r1))
|
57
|
+
@r2c1 = @model.create!(valid_attributes(:parent => @r2))
|
58
|
+
|
59
|
+
@r1c1s1 = @model.create!(valid_attributes(:parent => @r1c1))
|
60
|
+
@r1c2s1 = @model.create!(valid_attributes(:parent => @r1c2))
|
61
|
+
@r1c2s2 = @model.create!(valid_attributes(:parent => @r1c2))
|
62
|
+
@r1c2s3 = @model.create!(valid_attributes(:parent => @r1c2))
|
63
|
+
|
64
|
+
@r1c2s2m1 = @model.create!(valid_attributes(:parent => @r1c2s2))
|
65
|
+
end
|
66
|
+
|
67
|
+
after do
|
68
|
+
@model.delete_all
|
69
|
+
end
|
70
|
+
|
71
|
+
it "should find all root nodes" do
|
72
|
+
@model.roots.all.should == [@r1, @r2, @r3]
|
73
|
+
end
|
74
|
+
|
75
|
+
it "should find a root nodes" do
|
76
|
+
@model.roots.first.should == @r1
|
77
|
+
end
|
78
|
+
|
79
|
+
it "should maintain the integrity of the tree if a node is deleted" do
|
80
|
+
@r1c2.destroy
|
81
|
+
|
82
|
+
@r1.reload
|
83
|
+
@r1c3.reload
|
84
|
+
|
85
|
+
@r1.bounds.should == (1..8)
|
86
|
+
@r1c3.bounds.should == (6..7)
|
87
|
+
end
|
88
|
+
|
89
|
+
it "should maintain the integrity of the tree if a node is moved" do
|
90
|
+
@r1c2.parent_id = @r2.id
|
91
|
+
@r1c2.save!
|
92
|
+
|
93
|
+
reload_models(@r1, @r1c3, @r2, @r1c2, @r1c2s1)
|
94
|
+
|
95
|
+
@r1.bounds.should == (1..8)
|
96
|
+
@r1c3.bounds.should == (6..7)
|
97
|
+
@r2.bounds.should == (9..22)
|
98
|
+
@r1c2.bounds.should == (12..21)
|
99
|
+
@r1c2s1.bounds.should == (13..14)
|
100
|
+
end
|
101
|
+
|
102
|
+
it "should change the parent, left and right in the database when a node is moved" do
|
103
|
+
|
104
|
+
@r1c2.parent_id = @r2.id
|
105
|
+
|
106
|
+
@r1c2.save!
|
107
|
+
@r1c2 = @model.find(@r1c2.id)
|
108
|
+
|
109
|
+
@r1c2.bounds.should == (12..21)
|
110
|
+
@r1c2.parent_id.should == @r2.id
|
111
|
+
end
|
112
|
+
|
113
|
+
it "should maintain the integrity of the tree if a node is moved to a root position" do
|
114
|
+
@r1c2.parent_id = nil
|
115
|
+
@r1c2.save!
|
116
|
+
|
117
|
+
reload_models(@r1, @r1c3, @r2, @r1c2, @r1c2s1)
|
118
|
+
|
119
|
+
@r1.bounds.should == (1..8)
|
120
|
+
@r1c3.bounds.should == (6..7)
|
121
|
+
@r1c2.bounds.should == (15..24)
|
122
|
+
@r1c2s1.bounds.should == (16..17)
|
123
|
+
end
|
124
|
+
|
125
|
+
it "should maintain the integrity of the tree if a node is moved to a root position by assigning a blank string (mass assignment)" do
|
126
|
+
@r1c2.parent_id = ""
|
127
|
+
@r1c2.save!
|
128
|
+
|
129
|
+
reload_models(@r1, @r1c3, @r2, @r1c2, @r1c2s1)
|
130
|
+
|
131
|
+
@r1.bounds.should == (1..8)
|
132
|
+
@r1c3.bounds.should == (6..7)
|
133
|
+
@r1c2.bounds.should == (15..24)
|
134
|
+
@r1c2s1.bounds.should == (16..17)
|
135
|
+
end
|
136
|
+
|
137
|
+
it "should maintain the integrity of the tree if a root is to a non-root position" do
|
138
|
+
@r1c2.reload
|
139
|
+
@r2.parent_id = @r1c2.id
|
140
|
+
@r2.save!
|
141
|
+
|
142
|
+
reload_models(@r1, @r2, @r2c1, @r1c3, @r3, @r1c2)
|
143
|
+
|
144
|
+
@r1.bounds.should == (1..22)
|
145
|
+
@r1c2.bounds.should == (6..19)
|
146
|
+
@r1c3.bounds.should == (20..21)
|
147
|
+
@r3.bounds.should == (23..24)
|
148
|
+
@r2.bounds.should == (15..18)
|
149
|
+
@r2c1.bounds.should == (16..17)
|
150
|
+
end
|
151
|
+
|
152
|
+
it "should maintain the integrity of the tree if a node is moved through the parent association" do
|
153
|
+
@r1c2.parent = @r2
|
154
|
+
@r1c2.save!
|
155
|
+
|
156
|
+
reload_models(@r1, @r1c3, @r2, @r1c2, @r1c2s1)
|
157
|
+
|
158
|
+
@r1.bounds.should == (1..8)
|
159
|
+
@r1c3.bounds.should == (6..7)
|
160
|
+
@r2.bounds.should == (9..22)
|
161
|
+
@r1c2.bounds.should == (12..21)
|
162
|
+
@r1c2s1.bounds.should == (13..14)
|
163
|
+
end
|
164
|
+
|
165
|
+
it "should maintain the integrity of the tree if a node is moved to a root position through the parent association" do
|
166
|
+
@r1c2.parent = nil
|
167
|
+
@r1c2.save!
|
168
|
+
|
169
|
+
reload_models(@r1, @r1c3, @r2, @r1c2, @r1c2s1)
|
170
|
+
|
171
|
+
@r1.bounds.should == (1..8)
|
172
|
+
@r1c3.bounds.should == (6..7)
|
173
|
+
@r1c2.bounds.should == (15..24)
|
174
|
+
@r1c2s1.bounds.should == (16..17)
|
175
|
+
end
|
176
|
+
|
177
|
+
it "should maintain the integrity of the tree if a root is to a non-root position through the parent association" do
|
178
|
+
@r1c2.reload
|
179
|
+
@r2.parent = @r1c2
|
180
|
+
@r2.save!
|
181
|
+
|
182
|
+
reload_models(@r1, @r2, @r2c1, @r1c3, @r3, @r1c2)
|
183
|
+
|
184
|
+
@r1.bounds.should == (1..22)
|
185
|
+
@r1c2.bounds.should == (6..19)
|
186
|
+
@r1c3.bounds.should == (20..21)
|
187
|
+
@r3.bounds.should == (23..24)
|
188
|
+
@r2.bounds.should == (15..18)
|
189
|
+
@r2c1.bounds.should == (16..17)
|
190
|
+
end
|
191
|
+
|
192
|
+
it "should be invalid if parent is a descendant" do
|
193
|
+
@r2.parent = @r2c1
|
194
|
+
@r2.should_not be_valid
|
195
|
+
end
|
196
|
+
|
197
|
+
it "should be invalid if parent is self" do
|
198
|
+
@r2.parent = @r2
|
199
|
+
@r2.should_not be_valid
|
200
|
+
end
|
201
|
+
|
202
|
+
describe ".nested_set" do
|
203
|
+
it "should find all nodes as a nested set and cache that data" do
|
204
|
+
roots = @model.nested_set
|
205
|
+
|
206
|
+
@model.delete_all
|
207
|
+
|
208
|
+
roots[0].should == @r1
|
209
|
+
roots[0].children[0].should == @r1c1
|
210
|
+
roots[0].children[0].children[0].should == @r1c1s1
|
211
|
+
roots[0].children[1].should == @r1c2
|
212
|
+
roots[0].children[1].children[0].should == @r1c2s1
|
213
|
+
roots[0].children[1].children[1].should == @r1c2s2
|
214
|
+
roots[0].children[1].children[1].children[0].should == @r1c2s2m1
|
215
|
+
roots[0].children[1].children[2].should == @r1c2s3
|
216
|
+
roots[0].children[2].should == @r1c3
|
217
|
+
roots[1].should == @r2
|
218
|
+
roots[1].children[0].should == @r2c1
|
219
|
+
roots[2].should == @r3
|
220
|
+
|
221
|
+
roots[1].children[0].parent.should == @r2
|
222
|
+
end
|
223
|
+
end
|
224
|
+
|
225
|
+
describe ".find_with_nested_set" do
|
226
|
+
it "should find a single node and cache it's descendants" do
|
227
|
+
node = @model.find_with_nested_set(@r1c2.id)
|
228
|
+
|
229
|
+
@model.delete_all
|
230
|
+
|
231
|
+
node.should == @r1c2
|
232
|
+
node.children[0].should == @r1c2s1
|
233
|
+
node.children[1].should == @r1c2s2
|
234
|
+
node.children[1].children[0].should == @r1c2s2m1
|
235
|
+
node.children[2].should == @r1c2s3
|
236
|
+
end
|
237
|
+
|
238
|
+
it "should allow find with conditions" do
|
239
|
+
node = @model.find_with_nested_set(:first, :conditions => { :id => @r1c2.id })
|
240
|
+
|
241
|
+
@model.delete_all
|
242
|
+
|
243
|
+
node.should == @r1c2
|
244
|
+
node.children[0].should == @r1c2s1
|
245
|
+
node.children[1].should == @r1c2s2
|
246
|
+
node.children[1].children[0].should == @r1c2s2m1
|
247
|
+
node.children[2].should == @r1c2s3
|
248
|
+
end
|
249
|
+
|
250
|
+
it "should allow find all with conditions" do
|
251
|
+
nodes = @model.find_with_nested_set(:all, :conditions => { :parent_id => @r1.id })
|
252
|
+
|
253
|
+
@model.delete_all
|
254
|
+
|
255
|
+
nodes[0].should == @r1c1
|
256
|
+
nodes[0].children[0].should == @r1c1s1
|
257
|
+
nodes[1].should == @r1c2
|
258
|
+
nodes[1].children[0].should == @r1c2s1
|
259
|
+
nodes[1].children[1].should == @r1c2s2
|
260
|
+
nodes[1].children[1].children[0].should == @r1c2s2m1
|
261
|
+
nodes[1].children[2].should == @r1c2s3
|
262
|
+
nodes[2].should == @r1c3
|
263
|
+
end
|
264
|
+
end
|
265
|
+
|
266
|
+
describe ".sort_nodes_to_nested_set" do
|
267
|
+
it "should accept a list of nodes and sort them to a nested set" do
|
268
|
+
roots = @model.sort_nodes_to_nested_set(@model.find(:all))
|
269
|
+
roots[0].should == @r1
|
270
|
+
roots[0].children[0].should == @r1c1
|
271
|
+
roots[0].children[0].children[0].should == @r1c1s1
|
272
|
+
roots[0].children[1].should == @r1c2
|
273
|
+
roots[0].children[1].children[0].should == @r1c2s1
|
274
|
+
roots[0].children[1].children[1].should == @r1c2s2
|
275
|
+
roots[0].children[1].children[1].children[0].should == @r1c2s2m1
|
276
|
+
roots[0].children[1].children[2].should == @r1c2s3
|
277
|
+
roots[0].children[2].should == @r1c3
|
278
|
+
roots[1].should == @r2
|
279
|
+
roots[1].children[0].should == @r2c1
|
280
|
+
roots[2].should == @r3
|
281
|
+
end
|
282
|
+
end
|
283
|
+
|
284
|
+
describe ".recalculate_nested_set" do
|
285
|
+
def values
|
286
|
+
@model.find(:all, :order => :id).map { |node| [node.left, node.right] }
|
287
|
+
end
|
288
|
+
|
289
|
+
before do
|
290
|
+
@model.find(:all, :order => :id).each do |i|
|
291
|
+
i.send(:set_boundaries, rand(1000), rand(1000))
|
292
|
+
i.save_without_validation!
|
293
|
+
end
|
294
|
+
end
|
295
|
+
|
296
|
+
it "should correctly restore the left and right values for a messed up nested set" do
|
297
|
+
@model.recalculate_nested_set
|
298
|
+
[@r1, @r2, @r3].each(&:reload)
|
299
|
+
|
300
|
+
expected = [
|
301
|
+
[@r1c1, @r1c2, @r1c3, @r1c1s1, @r1c2s1, @r1c2s2, @r1c2s3, @r1c2s2m1],
|
302
|
+
[@r2c1],
|
303
|
+
[],
|
304
|
+
[@r1c1s1],
|
305
|
+
[@r1c2s1, @r1c2s2, @r1c2s3, @r1c2s2m1],
|
306
|
+
[],
|
307
|
+
[],
|
308
|
+
[],
|
309
|
+
[],
|
310
|
+
[@r1c2s2m1],
|
311
|
+
[],
|
312
|
+
[]
|
313
|
+
]
|
314
|
+
|
315
|
+
[@r1, @r2, @r3, @r1c1, @r1c2, @r1c3, @r2c1, @r1c1s1, @r1c2s1, @r1c2s2, @r1c2s3, @r1c2s2m1].each_with_index do |node, i|
|
316
|
+
node.descendants.find(:all, :order => :id).should == expected[i]
|
317
|
+
end
|
318
|
+
end
|
319
|
+
|
320
|
+
it "should leave all records valid after running" do
|
321
|
+
@model.recalculate_nested_set
|
322
|
+
@model.find(:all).each do |node|
|
323
|
+
node.should be_valid
|
324
|
+
end
|
325
|
+
end
|
326
|
+
end
|
327
|
+
|
328
|
+
describe "#cache_nested_set" do
|
329
|
+
it "should cache all descendant nodes so that calls to #children or #parent don't hit the database" do
|
330
|
+
@r1c2.cache_nested_set
|
331
|
+
|
332
|
+
@model.delete_all
|
333
|
+
|
334
|
+
@r1c2.children[0].should == @r1c2s1
|
335
|
+
@r1c2.children[1].should == @r1c2s2
|
336
|
+
@r1c2.children[1].children[0].should == @r1c2s2m1
|
337
|
+
@r1c2.children[2].should == @r1c2s3
|
338
|
+
|
339
|
+
@r1c2.children[1].children[0].parent.should == @r1c2s2
|
340
|
+
end
|
341
|
+
end
|
342
|
+
|
343
|
+
describe "#parent" do
|
344
|
+
it "should find the parent node" do
|
345
|
+
@r1c1.parent.should == @r1
|
346
|
+
@r1c2s2.parent.should == @r1c2
|
347
|
+
@r1c2s2m1.parent.should == @r1c2s2
|
348
|
+
end
|
349
|
+
end
|
350
|
+
|
351
|
+
describe "#children" do
|
352
|
+
it "should find all nodes that are direct descendants of this one" do
|
353
|
+
@r1.children.should == [@r1c1, @r1c2, @r1c3]
|
354
|
+
@r1c2s2.children.should == [@r1c2s2m1]
|
355
|
+
end
|
356
|
+
|
357
|
+
it "should allow creation of children" do
|
358
|
+
child = @r1c2.children.create!(valid_attributes)
|
359
|
+
|
360
|
+
child.parent_id.should == @r1c2.id
|
361
|
+
child.bounds.should == (15..16)
|
362
|
+
end
|
363
|
+
|
364
|
+
it "should allow addition of children" do
|
365
|
+
@r2.children << @r1c2
|
366
|
+
|
367
|
+
reload_models(@r1, @r1c3, @r2, @r1c2, @r1c2s1)
|
368
|
+
|
369
|
+
@r1.bounds.should == (1..8)
|
370
|
+
@r1c3.bounds.should == (6..7)
|
371
|
+
@r2.bounds.should == (9..22)
|
372
|
+
@r1c2.bounds.should == (12..21)
|
373
|
+
@r1c2s1.bounds.should == (13..14)
|
374
|
+
end
|
375
|
+
end
|
376
|
+
|
377
|
+
describe "#patriarch" do
|
378
|
+
it "should find the root node that this node descended from" do
|
379
|
+
@r1c1.patriarch.should == @r1
|
380
|
+
@r1c2s2.patriarch.should == @r1
|
381
|
+
@r1c2s2m1.patriarch.should == @r1
|
382
|
+
@r2c1.patriarch.should == @r2
|
383
|
+
@r1.patriarch.should == @r1
|
384
|
+
end
|
385
|
+
end
|
386
|
+
|
387
|
+
describe "#root" do
|
388
|
+
it "should find the root node that this node descended from" do
|
389
|
+
@r1c1.root.should == @r1
|
390
|
+
@r1c2s2.root.should == @r1
|
391
|
+
@r1c2s2m1.root.should == @r1
|
392
|
+
@r2c1.root.should == @r2
|
393
|
+
@r1.root.should == @r1
|
394
|
+
end
|
395
|
+
end
|
396
|
+
|
397
|
+
describe "#root?" do
|
398
|
+
it "should be true if node doesn't have a parent" do
|
399
|
+
@r1.should be_root
|
400
|
+
@model.roots.should include(@r1)
|
401
|
+
@r1.parent.should be_nil
|
402
|
+
end
|
403
|
+
end
|
404
|
+
|
405
|
+
describe "#descendant_of(other_node)" do
|
406
|
+
it "should be true if other_node is an ancestor of node" do
|
407
|
+
reload_models @r1, @r1c2s2
|
408
|
+
|
409
|
+
@r1c2s2.should be_descendant_of(@r1)
|
410
|
+
@r1c2s2.ancestors.should include(@r1)
|
411
|
+
@r1.descendants.should include(@r1c2s2)
|
412
|
+
end
|
413
|
+
end
|
414
|
+
|
415
|
+
describe "#generation" do
|
416
|
+
it "should find all nodes in the same generation as this one for a root node" do
|
417
|
+
@r1.generation.should == [@r1, @r2, @r3]
|
418
|
+
end
|
419
|
+
|
420
|
+
it "should find all nodes in the same generation as this one" do
|
421
|
+
@r1c1.generation.should == [@r1c1, @r1c2, @r1c3]
|
422
|
+
end
|
423
|
+
end
|
424
|
+
|
425
|
+
describe "#siblings" do
|
426
|
+
it "should find all sibling nodes for a root node" do
|
427
|
+
@r1.siblings.should == [@r2, @r3]
|
428
|
+
end
|
429
|
+
|
430
|
+
it "should find all sibling nodes for a child node" do
|
431
|
+
@r1c1.siblings.should == [@r1c2, @r1c3]
|
432
|
+
end
|
433
|
+
end
|
434
|
+
|
435
|
+
describe "#descendants" do
|
436
|
+
it "should find all descendants of this node" do
|
437
|
+
@r1.descendants.should == [@r1c1, @r1c1s1, @r1c2, @r1c2s1, @r1c2s2, @r1c2s2m1, @r1c2s3, @r1c3]
|
438
|
+
end
|
439
|
+
end
|
440
|
+
|
441
|
+
describe "#family" do
|
442
|
+
it "should combine self and descendants" do
|
443
|
+
@r1.family.should == [@r1, @r1c1, @r1c1s1, @r1c2, @r1c2s1, @r1c2s2, @r1c2s2m1, @r1c2s3, @r1c3]
|
444
|
+
end
|
445
|
+
end
|
446
|
+
|
447
|
+
describe "#family_ids" do
|
448
|
+
it "should find all ids of the node's nested set" do
|
449
|
+
@r1c1.family_ids.should == [@r1c1.id, @r1c1s1.id]
|
450
|
+
@r1c2.family_ids.should == [@r1c2.id, @r1c2s1.id, @r1c2s2.id, @r1c2s2m1.id, @r1c2s3.id]
|
451
|
+
end
|
452
|
+
end
|
453
|
+
|
454
|
+
describe "#ancestors" do
|
455
|
+
it "should return a node's parent and its parent's parents" do
|
456
|
+
@r1c2s2m1.ancestors.should == [@r1c2s2, @r1c2, @r1]
|
457
|
+
end
|
458
|
+
end
|
459
|
+
|
460
|
+
describe "#lineage" do
|
461
|
+
it "should return a node, it's parent and its parent's parents" do
|
462
|
+
@r1c2s2m1.lineage.should == [@r1c2s2m1, @r1c2s2, @r1c2, @r1]
|
463
|
+
end
|
464
|
+
end
|
465
|
+
|
466
|
+
describe "#level" do
|
467
|
+
it "should give the depth from the node to its root" do
|
468
|
+
@r1.level.should == 0
|
469
|
+
@r1c2.level.should == 1
|
470
|
+
@r1c2s2.level.should == 2
|
471
|
+
@r1c2s2m1.level.should == 3
|
472
|
+
end
|
473
|
+
end
|
474
|
+
|
475
|
+
describe "#kin" do
|
476
|
+
it "should find the patriarch and all its descendants" do
|
477
|
+
@r1c2s2.kin.should == [@r1, @r1c1, @r1c1s1, @r1c2, @r1c2s1, @r1c2s2, @r1c2s2m1, @r1c2s3, @r1c3]
|
478
|
+
end
|
479
|
+
end
|
480
|
+
|
481
|
+
end
|
482
|
+
|
483
|
+
describe @model, "with acts_as_nested_set" do
|
484
|
+
|
485
|
+
it "should add a new root node if the parent is not set" do
|
486
|
+
without_changing_the_database do
|
487
|
+
@instance = @model.create!(valid_attributes)
|
488
|
+
@instance.parent_id.should be_nil
|
489
|
+
|
490
|
+
@instance.bounds.should == (1..2)
|
491
|
+
end
|
492
|
+
end
|
493
|
+
|
494
|
+
it "should add a new root node if the parent is not set and there already are some root nodes" do
|
495
|
+
without_changing_the_database do
|
496
|
+
@model.create!(valid_attributes)
|
497
|
+
@model.create!(valid_attributes)
|
498
|
+
@instance = @model.create!(valid_attributes)
|
499
|
+
@instance.reload
|
500
|
+
|
501
|
+
@instance.parent_id.should be_nil
|
502
|
+
@instance.bounds.should == (5..6)
|
503
|
+
end
|
504
|
+
end
|
505
|
+
|
506
|
+
it "should append a child node to a parent" do
|
507
|
+
without_changing_the_database do
|
508
|
+
@parent = @model.create!(valid_attributes)
|
509
|
+
@parent.bounds.should == (1..2)
|
510
|
+
|
511
|
+
@instance = @model.create!(valid_attributes(:parent => @parent))
|
512
|
+
|
513
|
+
@parent.reload
|
514
|
+
|
515
|
+
@instance.parent.should == @parent
|
516
|
+
|
517
|
+
@instance.bounds.should == (2..3)
|
518
|
+
@parent.bounds.should == (1..4)
|
519
|
+
end
|
520
|
+
end
|
521
|
+
|
522
|
+
it "should rollback changes if the save is not successfull for some reason" do
|
523
|
+
without_changing_the_database do
|
524
|
+
@parent = @model.create!(valid_attributes)
|
525
|
+
@parent.bounds.should == (1..2)
|
526
|
+
|
527
|
+
@instance = @model.create(invalid_attributes(:parent => @parent))
|
528
|
+
@instance.should be_a_new_record
|
529
|
+
|
530
|
+
@parent.reload
|
531
|
+
|
532
|
+
@parent.bounds.should == (1..2)
|
533
|
+
end
|
534
|
+
end
|
535
|
+
|
536
|
+
it "should append a child node to a parent and shift other nodes out of the way" do
|
537
|
+
without_changing_the_database do
|
538
|
+
@root1 = @model.create!(valid_attributes)
|
539
|
+
@root2 = @model.create!(valid_attributes)
|
540
|
+
|
541
|
+
@root1.bounds.should == (1..2)
|
542
|
+
@root2.bounds.should == (3..4)
|
543
|
+
|
544
|
+
@child1 = @model.create!(valid_attributes(:parent => @root1))
|
545
|
+
reload_models(@root1, @root2)
|
546
|
+
|
547
|
+
@root1.bounds.should == (1..4)
|
548
|
+
@root2.bounds.should == (5..6)
|
549
|
+
@child1.bounds.should == (2..3)
|
550
|
+
|
551
|
+
@child2 = @model.create!(valid_attributes(:parent => @root1))
|
552
|
+
reload_models(@root1, @root2, @child1)
|
553
|
+
|
554
|
+
@root1.bounds.should == (1..6)
|
555
|
+
@root2.bounds.should == (7..8)
|
556
|
+
@child1.bounds.should == (2..3)
|
557
|
+
@child2.bounds.should == (4..5)
|
558
|
+
|
559
|
+
@subchild1 = @model.create!(valid_attributes(:parent => @child2))
|
560
|
+
reload_models(@root1, @root2, @child1, @child2)
|
561
|
+
|
562
|
+
@root1.bounds.should == (1..8)
|
563
|
+
@root2.bounds.should == (9..10)
|
564
|
+
@child1.bounds.should == (2..3)
|
565
|
+
@child2.bounds.should == (4..7)
|
566
|
+
@subchild1.bounds.should == (5..6)
|
567
|
+
|
568
|
+
@subchild2 = @model.create!(valid_attributes(:parent => @child1))
|
569
|
+
reload_models(@root1, @root2, @child1, @child2, @subchild1)
|
570
|
+
|
571
|
+
@root1.bounds.should == (1..10)
|
572
|
+
@root2.bounds.should == (11..12)
|
573
|
+
@child1.bounds.should == (2..5)
|
574
|
+
@child2.bounds.should == (6..9)
|
575
|
+
@subchild1.bounds.should == (7..8)
|
576
|
+
@subchild2.bounds.should == (3..4)
|
577
|
+
end
|
578
|
+
end
|
579
|
+
|
580
|
+
end
|
581
|
+
|
582
|
+
def reload_models(*attrs)
|
583
|
+
attrs.each {|m| m.reload }
|
584
|
+
end
|
585
|
+
|
586
|
+
end
|
data/spec/spec_helper.rb
ADDED
@@ -0,0 +1,83 @@
|
|
1
|
+
$TESTING=true
|
2
|
+
$:.push File.join(File.dirname(__FILE__), '..', 'lib')
|
3
|
+
|
4
|
+
require 'rubygems'
|
5
|
+
require 'active_record'
|
6
|
+
# require 'ruby-debug' # Slows down the tests massively
|
7
|
+
|
8
|
+
require 'eb_nested_set'
|
9
|
+
|
10
|
+
require 'spec'
|
11
|
+
|
12
|
+
# change this if sqlite is unavailable
|
13
|
+
dbconfig = case ENV["DB"]
|
14
|
+
when "postgresql"
|
15
|
+
{
|
16
|
+
:adapter => 'postgresql',
|
17
|
+
:database => 'even_better_nested_set_test',
|
18
|
+
:host => '127.0.0.1'
|
19
|
+
}
|
20
|
+
when "mysql"
|
21
|
+
{
|
22
|
+
:adapter => 'mysql',
|
23
|
+
:database => 'even_better_nested_set_test',
|
24
|
+
:host => '127.0.0.1'
|
25
|
+
}
|
26
|
+
else
|
27
|
+
{
|
28
|
+
:adapter => 'sqlite3',
|
29
|
+
:database => File.join(File.dirname(__FILE__), 'db', 'test.sqlite3')
|
30
|
+
}
|
31
|
+
end
|
32
|
+
|
33
|
+
ActiveRecord::Base.establish_connection(dbconfig)
|
34
|
+
ActiveRecord::Migration.verbose = false
|
35
|
+
|
36
|
+
def show_model_variables_for(context, model)
|
37
|
+
context.instance_variables.sort.each do |i|
|
38
|
+
m = eval(i)
|
39
|
+
if m.is_a?(model)
|
40
|
+
m.reload
|
41
|
+
puts "#{i.ljust(8)}\t#{m.left}\t#{m.right}\t#{m.name}"
|
42
|
+
end
|
43
|
+
end
|
44
|
+
end
|
45
|
+
|
46
|
+
#ActiveRecord::Base.logger = Logger.new(STDOUT)
|
47
|
+
|
48
|
+
|
49
|
+
class TestMigration < ActiveRecord::Migration
|
50
|
+
def self.up
|
51
|
+
create_table :directories, :force => true do |t|
|
52
|
+
t.column :lft, :integer
|
53
|
+
t.column :rgt, :integer
|
54
|
+
t.column :parent_id, :integer
|
55
|
+
t.column :name, :string
|
56
|
+
end
|
57
|
+
|
58
|
+
create_table :employees, :force => true do |t|
|
59
|
+
t.column :left, :integer
|
60
|
+
t.column :right, :integer
|
61
|
+
t.column :parent_id, :integer
|
62
|
+
t.column :name, :string
|
63
|
+
t.column :company_id, :integer
|
64
|
+
end
|
65
|
+
end
|
66
|
+
|
67
|
+
def self.down
|
68
|
+
drop_table :directories
|
69
|
+
drop_table :employees
|
70
|
+
rescue
|
71
|
+
nil
|
72
|
+
end
|
73
|
+
end
|
74
|
+
|
75
|
+
def without_changing_the_database
|
76
|
+
ActiveRecord::Base.transaction do
|
77
|
+
yield
|
78
|
+
raise ActiveRecord::Rollback
|
79
|
+
end
|
80
|
+
end
|
81
|
+
|
82
|
+
TestMigration.down
|
83
|
+
TestMigration.up
|
metadata
ADDED
@@ -0,0 +1,64 @@
|
|
1
|
+
--- !ruby/object:Gem::Specification
|
2
|
+
name: eb_nested_set
|
3
|
+
version: !ruby/object:Gem::Version
|
4
|
+
version: 0.3.3
|
5
|
+
platform: ruby
|
6
|
+
authors:
|
7
|
+
- Jonas Nicklas
|
8
|
+
autorequire: eb_nested_set
|
9
|
+
bindir: bin
|
10
|
+
cert_chain: []
|
11
|
+
|
12
|
+
date: 2009-02-15 00:00:00 +01:00
|
13
|
+
default_executable:
|
14
|
+
dependencies: []
|
15
|
+
|
16
|
+
description: A cool acts_as_nested_set alternative
|
17
|
+
email: jonas.nicklas@gmail.com
|
18
|
+
executables: []
|
19
|
+
|
20
|
+
extensions: []
|
21
|
+
|
22
|
+
extra_rdoc_files:
|
23
|
+
- README.md
|
24
|
+
- LICENSE
|
25
|
+
files:
|
26
|
+
- LICENSE
|
27
|
+
- README.md
|
28
|
+
- Rakefile
|
29
|
+
- init.rb
|
30
|
+
- lib/eb_nested_set.rb
|
31
|
+
- spec/db
|
32
|
+
- spec/db/test.sqlite3
|
33
|
+
- spec/directory_spec.rb
|
34
|
+
- spec/employee_spec.rb
|
35
|
+
- spec/nested_set_behavior.rb
|
36
|
+
- spec/spec_helper.rb
|
37
|
+
has_rdoc: true
|
38
|
+
homepage: http://github.com/jnicklas/even_better_nested_set/tree/master
|
39
|
+
post_install_message:
|
40
|
+
rdoc_options: []
|
41
|
+
|
42
|
+
require_paths:
|
43
|
+
- lib
|
44
|
+
required_ruby_version: !ruby/object:Gem::Requirement
|
45
|
+
requirements:
|
46
|
+
- - ">="
|
47
|
+
- !ruby/object:Gem::Version
|
48
|
+
version: "0"
|
49
|
+
version:
|
50
|
+
required_rubygems_version: !ruby/object:Gem::Requirement
|
51
|
+
requirements:
|
52
|
+
- - ">="
|
53
|
+
- !ruby/object:Gem::Version
|
54
|
+
version: "0"
|
55
|
+
version:
|
56
|
+
requirements: []
|
57
|
+
|
58
|
+
rubyforge_project:
|
59
|
+
rubygems_version: 1.3.1
|
60
|
+
signing_key:
|
61
|
+
specification_version: 2
|
62
|
+
summary: A cool acts_as_nested_set alternative
|
63
|
+
test_files: []
|
64
|
+
|