sam-dm-core 0.9.6

Sign up to get free protection for your applications and to get access to all the features.
Files changed (126) hide show
  1. data/.autotest +26 -0
  2. data/CONTRIBUTING +51 -0
  3. data/FAQ +92 -0
  4. data/History.txt +145 -0
  5. data/MIT-LICENSE +22 -0
  6. data/Manifest.txt +125 -0
  7. data/QUICKLINKS +12 -0
  8. data/README.txt +143 -0
  9. data/Rakefile +30 -0
  10. data/SPECS +63 -0
  11. data/TODO +1 -0
  12. data/lib/dm-core.rb +224 -0
  13. data/lib/dm-core/adapters.rb +4 -0
  14. data/lib/dm-core/adapters/abstract_adapter.rb +202 -0
  15. data/lib/dm-core/adapters/data_objects_adapter.rb +707 -0
  16. data/lib/dm-core/adapters/mysql_adapter.rb +136 -0
  17. data/lib/dm-core/adapters/postgres_adapter.rb +188 -0
  18. data/lib/dm-core/adapters/sqlite3_adapter.rb +105 -0
  19. data/lib/dm-core/associations.rb +199 -0
  20. data/lib/dm-core/associations/many_to_many.rb +147 -0
  21. data/lib/dm-core/associations/many_to_one.rb +107 -0
  22. data/lib/dm-core/associations/one_to_many.rb +309 -0
  23. data/lib/dm-core/associations/one_to_one.rb +61 -0
  24. data/lib/dm-core/associations/relationship.rb +218 -0
  25. data/lib/dm-core/associations/relationship_chain.rb +81 -0
  26. data/lib/dm-core/auto_migrations.rb +113 -0
  27. data/lib/dm-core/collection.rb +638 -0
  28. data/lib/dm-core/dependency_queue.rb +31 -0
  29. data/lib/dm-core/hook.rb +11 -0
  30. data/lib/dm-core/identity_map.rb +45 -0
  31. data/lib/dm-core/is.rb +16 -0
  32. data/lib/dm-core/logger.rb +232 -0
  33. data/lib/dm-core/migrations/destructive_migrations.rb +17 -0
  34. data/lib/dm-core/migrator.rb +29 -0
  35. data/lib/dm-core/model.rb +471 -0
  36. data/lib/dm-core/naming_conventions.rb +84 -0
  37. data/lib/dm-core/property.rb +673 -0
  38. data/lib/dm-core/property_set.rb +162 -0
  39. data/lib/dm-core/query.rb +625 -0
  40. data/lib/dm-core/repository.rb +159 -0
  41. data/lib/dm-core/resource.rb +637 -0
  42. data/lib/dm-core/scope.rb +58 -0
  43. data/lib/dm-core/support.rb +7 -0
  44. data/lib/dm-core/support/array.rb +13 -0
  45. data/lib/dm-core/support/assertions.rb +8 -0
  46. data/lib/dm-core/support/errors.rb +23 -0
  47. data/lib/dm-core/support/kernel.rb +7 -0
  48. data/lib/dm-core/support/symbol.rb +41 -0
  49. data/lib/dm-core/transaction.rb +267 -0
  50. data/lib/dm-core/type.rb +160 -0
  51. data/lib/dm-core/type_map.rb +80 -0
  52. data/lib/dm-core/types.rb +19 -0
  53. data/lib/dm-core/types/boolean.rb +7 -0
  54. data/lib/dm-core/types/discriminator.rb +34 -0
  55. data/lib/dm-core/types/object.rb +24 -0
  56. data/lib/dm-core/types/paranoid_boolean.rb +34 -0
  57. data/lib/dm-core/types/paranoid_datetime.rb +33 -0
  58. data/lib/dm-core/types/serial.rb +9 -0
  59. data/lib/dm-core/types/text.rb +10 -0
  60. data/lib/dm-core/version.rb +3 -0
  61. data/script/all +5 -0
  62. data/script/performance.rb +203 -0
  63. data/script/profile.rb +87 -0
  64. data/spec/integration/association_spec.rb +1371 -0
  65. data/spec/integration/association_through_spec.rb +203 -0
  66. data/spec/integration/associations/many_to_many_spec.rb +449 -0
  67. data/spec/integration/associations/many_to_one_spec.rb +163 -0
  68. data/spec/integration/associations/one_to_many_spec.rb +151 -0
  69. data/spec/integration/auto_migrations_spec.rb +398 -0
  70. data/spec/integration/collection_spec.rb +1069 -0
  71. data/spec/integration/data_objects_adapter_spec.rb +32 -0
  72. data/spec/integration/dependency_queue_spec.rb +58 -0
  73. data/spec/integration/model_spec.rb +127 -0
  74. data/spec/integration/mysql_adapter_spec.rb +85 -0
  75. data/spec/integration/postgres_adapter_spec.rb +731 -0
  76. data/spec/integration/property_spec.rb +233 -0
  77. data/spec/integration/query_spec.rb +506 -0
  78. data/spec/integration/repository_spec.rb +57 -0
  79. data/spec/integration/resource_spec.rb +475 -0
  80. data/spec/integration/sqlite3_adapter_spec.rb +352 -0
  81. data/spec/integration/sti_spec.rb +208 -0
  82. data/spec/integration/strategic_eager_loading_spec.rb +138 -0
  83. data/spec/integration/transaction_spec.rb +75 -0
  84. data/spec/integration/type_spec.rb +271 -0
  85. data/spec/lib/logging_helper.rb +18 -0
  86. data/spec/lib/mock_adapter.rb +27 -0
  87. data/spec/lib/model_loader.rb +91 -0
  88. data/spec/lib/publicize_methods.rb +28 -0
  89. data/spec/models/vehicles.rb +34 -0
  90. data/spec/models/zoo.rb +47 -0
  91. data/spec/spec.opts +3 -0
  92. data/spec/spec_helper.rb +86 -0
  93. data/spec/unit/adapters/abstract_adapter_spec.rb +133 -0
  94. data/spec/unit/adapters/adapter_shared_spec.rb +15 -0
  95. data/spec/unit/adapters/data_objects_adapter_spec.rb +628 -0
  96. data/spec/unit/adapters/postgres_adapter_spec.rb +133 -0
  97. data/spec/unit/associations/many_to_many_spec.rb +17 -0
  98. data/spec/unit/associations/many_to_one_spec.rb +152 -0
  99. data/spec/unit/associations/one_to_many_spec.rb +393 -0
  100. data/spec/unit/associations/one_to_one_spec.rb +7 -0
  101. data/spec/unit/associations/relationship_spec.rb +71 -0
  102. data/spec/unit/associations_spec.rb +242 -0
  103. data/spec/unit/auto_migrations_spec.rb +111 -0
  104. data/spec/unit/collection_spec.rb +182 -0
  105. data/spec/unit/data_mapper_spec.rb +35 -0
  106. data/spec/unit/identity_map_spec.rb +126 -0
  107. data/spec/unit/is_spec.rb +80 -0
  108. data/spec/unit/migrator_spec.rb +33 -0
  109. data/spec/unit/model_spec.rb +339 -0
  110. data/spec/unit/naming_conventions_spec.rb +36 -0
  111. data/spec/unit/property_set_spec.rb +83 -0
  112. data/spec/unit/property_spec.rb +753 -0
  113. data/spec/unit/query_spec.rb +530 -0
  114. data/spec/unit/repository_spec.rb +93 -0
  115. data/spec/unit/resource_spec.rb +626 -0
  116. data/spec/unit/scope_spec.rb +142 -0
  117. data/spec/unit/transaction_spec.rb +493 -0
  118. data/spec/unit/type_map_spec.rb +114 -0
  119. data/spec/unit/type_spec.rb +119 -0
  120. data/tasks/ci.rb +68 -0
  121. data/tasks/dm.rb +63 -0
  122. data/tasks/doc.rb +20 -0
  123. data/tasks/gemspec.rb +23 -0
  124. data/tasks/hoe.rb +46 -0
  125. data/tasks/install.rb +20 -0
  126. metadata +216 -0
@@ -0,0 +1,203 @@
1
+ require File.expand_path(File.join(File.dirname(__FILE__), '..', 'spec_helper'))
2
+
3
+ if ADAPTER
4
+ describe 'through-associations' do
5
+ before :all do
6
+ repository(ADAPTER) do
7
+ class Tag
8
+ include DataMapper::Resource
9
+ def self.default_repository_name
10
+ ADAPTER
11
+ end
12
+
13
+ property :id, Serial
14
+ property :title, String
15
+ property :voided, Boolean, :default => false
16
+
17
+ has n, :taggings
18
+
19
+ has n, :relationships
20
+ has n, :related_posts, :through => :relationships, :class_name => 'Post', :child_key => [:post_id]
21
+
22
+ has n, :posts, :through => :taggings
23
+ end
24
+
25
+ class Tagging
26
+ include DataMapper::Resource
27
+ def self.default_repository_name
28
+ ADAPTER
29
+ end
30
+
31
+ property :id, Serial
32
+ property :title, String
33
+
34
+ belongs_to :post
35
+ belongs_to :tag
36
+ end
37
+
38
+ class Post
39
+ include DataMapper::Resource
40
+ def self.default_repository_name
41
+ ADAPTER
42
+ end
43
+
44
+ property :id, Serial
45
+ property :title, String
46
+
47
+ has n, :taggings
48
+ has n, :tags, :through => :taggings
49
+
50
+ has n, :relationships
51
+ has n, :related_posts,
52
+ :through => :relationships,
53
+ :child_key => [:post_id],
54
+ :class_name => "Post"
55
+
56
+ has n, :void_tags,
57
+ :through => :taggings,
58
+ :child_key => [:post_id],
59
+ :class_name => "Tag",
60
+ :remote_relationship_name => :tag,
61
+ Post.taggings.tag.voided => true
62
+ end
63
+
64
+ class Relationship
65
+ include DataMapper::Resource
66
+ def self.default_repository_name
67
+ ADAPTER
68
+ end
69
+
70
+ property :id, Serial
71
+ belongs_to :post
72
+ belongs_to :related_post, :class_name => "Post", :child_key => [:related_post_id]
73
+ end
74
+
75
+ [Post, Tag, Tagging, Relationship].each do |descendant|
76
+ descendant.auto_migrate!(ADAPTER)
77
+ end
78
+ end
79
+ end
80
+
81
+ describe '(sample data)' do
82
+ before(:each) do
83
+ post = Post.create(:title => "Entry")
84
+ another_post = Post.create(:title => "Another")
85
+
86
+ crappy = Tagging.new
87
+ post.taggings << crappy
88
+ post.save
89
+
90
+ crap = Tag.create(:title => "crap")
91
+ crap.taggings << crappy
92
+ crap.save
93
+
94
+ crappier = Tagging.new
95
+ post.taggings << crappier
96
+ post.save
97
+
98
+ crapz = Tag.create(:title => "crapz", :voided => true)
99
+ crapz.taggings << crappier
100
+ crapz.save
101
+
102
+ goody = Tagging.new
103
+ another_post.taggings << goody
104
+ another_post.save
105
+
106
+ good = Tag.create(:title => "good")
107
+ good.taggings << goody
108
+ good.save
109
+
110
+ relation = Relationship.new(:related_post_id => another_post.id)
111
+ post.relationships << relation
112
+ post.save
113
+ end
114
+
115
+ it 'should return the right children for has n => belongs_to relationships' do
116
+ Post.first.tags.select do |tag|
117
+ tag.title == 'crap'
118
+ end.size.should == 1
119
+ end
120
+
121
+ it 'should return the right children for has n => belongs_to self-referential relationships' do
122
+ Post.first.related_posts.select do |post|
123
+ post.title == 'Another'
124
+ end.size.should == 1
125
+ end
126
+
127
+ it 'should handle all()' do
128
+ related_posts = Post.first.related_posts
129
+ related_posts.all.object_id.should == related_posts.object_id
130
+ related_posts.all(:id => 2).first.should == Post.get!(2)
131
+ end
132
+
133
+ it 'should handle first()' do
134
+ post = Post.get!(2)
135
+ related_posts = Post.first.related_posts
136
+ related_posts.first.should == post
137
+ related_posts.first(10).should == [ post ]
138
+ related_posts.first(:id => 2).should == post
139
+ related_posts.first(10, :id => 2).map { |r| r.id }.should == [post.id]
140
+ end
141
+
142
+ it 'should handle get()' do
143
+ post = Post.get!(2)
144
+ related_posts = Post.first.related_posts
145
+ related_posts.get(2).should == post
146
+ end
147
+
148
+ it 'should proxy object should be frozen' do
149
+ Post.first.related_posts.should be_frozen
150
+ end
151
+
152
+ it "should respect tagging with conditions" do
153
+ post = Post.get(1)
154
+ post.tags.size
155
+ post.tags.select{ |t| t.voided == true }.size.should == 1
156
+ post.void_tags.size.should == 1
157
+ post.void_tags.all?{ |t| t.voided == true }.should be_true
158
+ end
159
+ end
160
+
161
+ describe "Saved Tag, Post, Tagging" do
162
+ before(:each) do
163
+ @tag = Tag.create
164
+ @post = Post.create
165
+ @tagging = Tagging.create(
166
+ :tag => @tag,
167
+ :post => @post
168
+ )
169
+ end
170
+
171
+ it "should get posts of a tag" do
172
+ @tag.posts.should == [@post]
173
+ end
174
+
175
+ it "should get tags of a post" do
176
+ @post.tags.should == [@tag]
177
+ end
178
+ end
179
+
180
+ describe "In-memory Tag, Post, Tagging" do
181
+ before(:each) do
182
+ @tag = Tag.new
183
+ @post = Post.new
184
+ @tagging = Tagging.new(
185
+ :tag => @tag,
186
+ :post => @post
187
+ )
188
+ end
189
+
190
+ it "should get posts of a tag" do
191
+ pending("DataMapper does not yet support in-memory associations") do
192
+ @tag.posts.should == [@post]
193
+ end
194
+ end
195
+
196
+ it "should get tags of a post" do
197
+ pending("DataMapper does not yet support in-memory associations") do
198
+ @post.tags.should == [@tag]
199
+ end
200
+ end
201
+ end
202
+ end
203
+ end
@@ -0,0 +1,449 @@
1
+ require File.expand_path(File.join(File.dirname(__FILE__), "..", "..", "spec_helper"))
2
+
3
+ describe DataMapper::Associations::ManyToMany::Proxy do
4
+ before :all do
5
+ class Editor
6
+ include DataMapper::Resource
7
+
8
+ def self.default_repository_name; ADAPTER end
9
+
10
+ property :id, Serial
11
+ property :name, String
12
+
13
+ has n, :books, :through => Resource
14
+ end
15
+
16
+ Object.send(:remove_const, :Book) if defined?(Book)
17
+
18
+ class Book
19
+ include DataMapper::Resource
20
+
21
+ def self.default_repository_name; ADAPTER end
22
+
23
+ property :id, Serial
24
+ property :title, String
25
+
26
+ has n, :editors, :through => Resource
27
+ end
28
+ end
29
+
30
+ before do
31
+ [ Book, Editor, BookEditor ].each { |k| k.auto_migrate! }
32
+
33
+ repository(ADAPTER) do
34
+ book_1 = Book.create(:title => "Dubliners")
35
+ book_2 = Book.create(:title => "Portrait of the Artist as a Young Man")
36
+ book_3 = Book.create(:title => "Ulysses")
37
+
38
+ editor_1 = Editor.create(:name => "Jon Doe")
39
+ editor_2 = Editor.create(:name => "Jane Doe")
40
+
41
+ BookEditor.create(:book => book_1, :editor => editor_1)
42
+ BookEditor.create(:book => book_2, :editor => editor_1)
43
+ BookEditor.create(:book => book_1, :editor => editor_2)
44
+
45
+ @parent = book_3
46
+ @association = @parent.editors
47
+ @other = [ editor_1 ]
48
+ end
49
+ end
50
+
51
+ it "should provide #replace" do
52
+ @association.should respond_to(:replace)
53
+ end
54
+
55
+ describe "#replace" do
56
+ it "should remove the resource from the collection" do
57
+ @association.should have(0).entries
58
+ @association.replace(@other)
59
+ @association.should == @other
60
+ end
61
+
62
+ it "should not automatically save that the resource was removed from the association" do
63
+ @association.replace(@other)
64
+ @parent.reload.should have(0).editors
65
+ end
66
+
67
+ it "should return the association" do
68
+ @association.replace(@other).object_id.should == @association.object_id
69
+ end
70
+
71
+ it "should add the new resources so they will be saved when saving the parent" do
72
+ @association.replace(@other)
73
+ @association.should == @other
74
+ @parent.save
75
+ @association.reload.should == @other
76
+ end
77
+
78
+ it "should instantiate the remote model if passed an array of hashes" do
79
+ @association.replace([ { :name => "Jim Smith" } ])
80
+ other = [ Editor.first(:name => "Jim Smith") ]
81
+ other.first.should_not be_nil
82
+ @association.should == other
83
+ @parent.save
84
+ @association.reload.should == other
85
+ end
86
+ end
87
+
88
+ it "should correctly link records" do
89
+ Book.get(1).should have(2).editors
90
+ Book.get(2).should have(1).editors
91
+ Book.get(3).should have(0).editors
92
+ Editor.get(1).should have(2).books
93
+ Editor.get(2).should have(1).books
94
+ end
95
+
96
+ it "should be able to have associated objects manually added" do
97
+ book = Book.get(3)
98
+ book.should have(0).editors
99
+
100
+ be = BookEditor.new(:book_id => book.id, :editor_id => 2)
101
+ book.book_editors << be
102
+ book.save
103
+
104
+ book.reload.should have(1).editors
105
+ end
106
+
107
+ it "should automatically added necessary through class" do
108
+ book = Book.get(3)
109
+ book.should have(0).editors
110
+
111
+ book.editors << Editor.get(1)
112
+ book.editors << Editor.new(:name => "Jimmy John")
113
+ book.save
114
+
115
+ book.reload.should have(2).editors
116
+ end
117
+
118
+ it "should react correctly to a new record" do
119
+ book = Book.new(:title => "Finnegan's Wake")
120
+ editor = Editor.get(2)
121
+ book.should have(0).editors
122
+ editor.should have(1).books
123
+
124
+ book.editors << editor
125
+ book.save
126
+
127
+ book.reload.should have(1).editors
128
+ editor.reload.should have(2).books
129
+ end
130
+
131
+ it "should be able to delete intermediate model" do
132
+ book = Book.get(1)
133
+ book.should have(2).book_editors
134
+ book.should have(2).editors
135
+
136
+ be = BookEditor.get(1,1)
137
+ book.book_editors.delete(be)
138
+ book.save
139
+
140
+ book.reload
141
+ book.should have(1).book_editors
142
+ book.should have(1).editors
143
+ end
144
+
145
+ it "should be clearable" do
146
+ repository(ADAPTER) do
147
+ book = Book.get(2)
148
+ book.should have(1).book_editors
149
+ book.should have(1).editors
150
+
151
+ book.editors.clear
152
+ book.save
153
+
154
+ book.reload
155
+ book.should have(0).book_editors
156
+ book.should have(0).editors
157
+ end
158
+ repository(ADAPTER) do
159
+ Book.get(2).should have(0).editors
160
+ end
161
+ end
162
+
163
+ it "should be able to delete one object" do
164
+ book = Book.get(1)
165
+ book.should have(2).book_editors
166
+ book.should have(2).editors
167
+
168
+ editor = book.editors.first
169
+ book.editors.delete(editor)
170
+ book.save
171
+
172
+ book.reload
173
+ book.should have(1).book_editors
174
+ book.should have(1).editors
175
+ editor.reload.books.should_not include(book)
176
+ end
177
+
178
+ it "should be destroyable" do
179
+ pending "cannot destroy a collection yet" do
180
+ book = Book.get(2)
181
+ book.should have(1).editors
182
+
183
+ book.editors.destroy
184
+ book.save
185
+
186
+ book.reload
187
+ book.should have(0).editors
188
+ end
189
+ end
190
+
191
+ describe "with natural keys" do
192
+ before :all do
193
+ class Author
194
+ include DataMapper::Resource
195
+
196
+ def self.default_repository_name; ADAPTER end
197
+
198
+ property :name, String, :key => true
199
+
200
+ has n, :books, :through => Resource
201
+ end
202
+
203
+ class Book
204
+ has n, :authors, :through => Resource
205
+ end
206
+ end
207
+
208
+ before do
209
+ [ Author, AuthorBook ].each { |k| k.auto_migrate! }
210
+
211
+ @author = Author.create(:name => "James Joyce")
212
+
213
+ @book_1 = Book.get!(1)
214
+ @book_2 = Book.get!(2)
215
+ @book_3 = Book.get!(3)
216
+
217
+ AuthorBook.create(:book => @book_1, :author => @author)
218
+ AuthorBook.create(:book => @book_2, :author => @author)
219
+ AuthorBook.create(:book => @book_3, :author => @author)
220
+ end
221
+
222
+ it "should have a join resource where the natural key is a property" do
223
+ AuthorBook.properties[:author_name].primitive.should == String
224
+ end
225
+
226
+ it "should have a join resource where every property is part of the key" do
227
+ AuthorBook.key.should == AuthorBook.properties.to_a
228
+ end
229
+
230
+ it "should correctly link records" do
231
+ @author.should have(3).books
232
+ @book_1.should have(1).authors
233
+ @book_2.should have(1).authors
234
+ @book_3.should have(1).authors
235
+ end
236
+ end
237
+
238
+ describe "When join model has non-serial (integer) natural keys." do
239
+ before :all do
240
+ class Tag
241
+ include DataMapper::Resource
242
+
243
+ def self.default_repository_name; ADAPTER end
244
+
245
+ property :id, Serial
246
+ property :name, String, :size => 128
247
+
248
+ has n, :book_taggings
249
+ has n, :books, :through => :book_taggings
250
+ end
251
+
252
+ class BookTagging
253
+ include DataMapper::Resource
254
+
255
+ def self.default_repository_name; ADAPTER end
256
+
257
+ property :book_id, Integer, :key => true
258
+ property :tag_id, Integer, :key => true
259
+
260
+ belongs_to :book
261
+ belongs_to :tag
262
+ end
263
+
264
+ class Book
265
+ has n, :book_taggings
266
+ has n, :tags, :through => :book_taggings
267
+ end
268
+ end
269
+
270
+ before do
271
+ [ Tag, BookTagging ].each { |k| k.auto_migrate! }
272
+
273
+ @tag_1 = Tag.create(:name => "good")
274
+ @tag_2 = Tag.create(:name => "long")
275
+
276
+ @book_1 = Book.get!(1)
277
+ @book_2 = Book.get!(2)
278
+ @book_3 = Book.get!(3)
279
+
280
+ BookTagging.create(:book => @book_2, :tag => @tag_1)
281
+ BookTagging.create(:book => @book_2, :tag => @tag_2)
282
+ BookTagging.create(:book => @book_3, :tag => @tag_2)
283
+ end
284
+
285
+ it "should fetch all tags for a book" do
286
+ @book_1.tags.should have(0).tags
287
+ @book_2.tags.should have(2).tags
288
+ @book_3.tags.should have(1).tags
289
+ end
290
+
291
+ it "should allow for adding an association using the << operator" do
292
+ @book_1.book_taggings << @tag_1
293
+ @book_1.tags.should have(0).tags
294
+ end
295
+ end
296
+
297
+ describe "with renamed associations" do
298
+ before :all do
299
+ class Singer
300
+ include DataMapper::Resource
301
+
302
+ def self.default_repository_name; ADAPTER end
303
+
304
+ property :id, Serial
305
+ property :name, String
306
+
307
+ has n, :tunes, :through => Resource, :class_name => 'Song'
308
+ end
309
+
310
+ class Song
311
+ include DataMapper::Resource
312
+
313
+ def self.default_repository_name; ADAPTER end
314
+
315
+ property :id, Serial
316
+ property :title, String
317
+
318
+ has n, :performers, :through => Resource, :class_name => 'Singer'
319
+ end
320
+ end
321
+
322
+ before do
323
+ [ Singer, Song, SingerSong ].each { |k| k.auto_migrate! }
324
+
325
+ song_1 = Song.create(:title => "Dubliners")
326
+ song_2 = Song.create(:title => "Portrait of the Artist as a Young Man")
327
+ song_3 = Song.create(:title => "Ulysses")
328
+
329
+ singer_1 = Singer.create(:name => "Jon Doe")
330
+ singer_2 = Singer.create(:name => "Jane Doe")
331
+
332
+ SingerSong.create(:song => song_1, :singer => singer_1)
333
+ SingerSong.create(:song => song_2, :singer => singer_1)
334
+ SingerSong.create(:song => song_1, :singer => singer_2)
335
+
336
+ @parent = song_3
337
+ @association = @parent.performers
338
+ @other = [ singer_1 ]
339
+ end
340
+
341
+ it "should provide #replace" do
342
+ @association.should respond_to(:replace)
343
+ end
344
+
345
+ it "should correctly link records" do
346
+ Song.get(1).should have(2).performers
347
+ Song.get(2).should have(1).performers
348
+ Song.get(3).should have(0).performers
349
+ Singer.get(1).should have(2).tunes
350
+ Singer.get(2).should have(1).tunes
351
+ end
352
+
353
+ it "should be able to have associated objects manually added" do
354
+ song = Song.get(3)
355
+ song.should have(0).performers
356
+
357
+ be = SingerSong.new(:song_id => song.id, :singer_id => 2)
358
+ song.singer_songs << be
359
+ song.save
360
+
361
+ song.reload.should have(1).performers
362
+ end
363
+
364
+ it "should automatically added necessary through class" do
365
+ song = Song.get(3)
366
+ song.should have(0).performers
367
+
368
+ song.performers << Singer.get(1)
369
+ song.performers << Singer.new(:name => "Jimmy John")
370
+ song.save
371
+
372
+ song.reload.should have(2).performers
373
+ end
374
+
375
+ it "should react correctly to a new record" do
376
+ song = Song.new(:title => "Finnegan's Wake")
377
+ singer = Singer.get(2)
378
+ song.should have(0).performers
379
+ singer.should have(1).tunes
380
+
381
+ song.performers << singer
382
+ song.save
383
+
384
+ song.reload.should have(1).performers
385
+ singer.reload.should have(2).tunes
386
+ end
387
+
388
+ it "should be able to delete intermediate model" do
389
+ song = Song.get(1)
390
+ song.should have(2).singer_songs
391
+ song.should have(2).performers
392
+
393
+ be = SingerSong.get(1,1)
394
+ song.singer_songs.delete(be)
395
+ song.save
396
+
397
+ song.reload
398
+ song.should have(1).singer_songs
399
+ song.should have(1).performers
400
+ end
401
+
402
+ it "should be clearable" do
403
+ repository(ADAPTER) do
404
+ song = Song.get(2)
405
+ song.should have(1).singer_songs
406
+ song.should have(1).performers
407
+
408
+ song.performers.clear
409
+ song.save
410
+
411
+ song.reload
412
+ song.should have(0).singer_songs
413
+ song.should have(0).performers
414
+ end
415
+ repository(ADAPTER) do
416
+ Song.get(2).should have(0).performers
417
+ end
418
+ end
419
+
420
+ it "should be able to delete one object" do
421
+ song = Song.get(1)
422
+ song.should have(2).singer_songs
423
+ song.should have(2).performers
424
+
425
+ editor = song.performers.first
426
+ song.performers.delete(editor)
427
+ song.save
428
+
429
+ song.reload
430
+ song.should have(1).singer_songs
431
+ song.should have(1).performers
432
+ editor.reload.tunes.should_not include(song)
433
+ end
434
+
435
+ it "should be destroyable" do
436
+ pending "cannot destroy a collection yet" do
437
+ song = Song.get(2)
438
+ song.should have(1).performers
439
+
440
+ song.performers.destroy
441
+ song.save
442
+
443
+ song.reload
444
+ song.should have(0).performers
445
+ end
446
+ end
447
+ end
448
+
449
+ end