Updated README.rdoc again
[feedcatcher.git] / vendor / rails / activerecord / test / cases / fixtures_test.rb
1 require "cases/helper"
2 require 'models/post'
3 require 'models/binary'
4 require 'models/topic'
5 require 'models/computer'
6 require 'models/developer'
7 require 'models/company'
8 require 'models/task'
9 require 'models/reply'
10 require 'models/joke'
11 require 'models/course'
12 require 'models/category'
13 require 'models/parrot'
14 require 'models/pirate'
15 require 'models/treasure'
16 require 'models/matey'
17 require 'models/ship'
18 require 'models/book'
19
20 class FixturesTest < ActiveRecord::TestCase
21 self.use_instantiated_fixtures = true
22 self.use_transactional_fixtures = false
23
24 fixtures :topics, :developers, :accounts, :tasks, :categories, :funny_jokes, :binaries
25
26 FIXTURES = %w( accounts binaries companies customers
27 developers developers_projects entrants
28 movies projects subscribers topics tasks )
29 MATCH_ATTRIBUTE_NAME = /[a-zA-Z][-_\w]*/
30
31 def test_clean_fixtures
32 FIXTURES.each do |name|
33 fixtures = nil
34 assert_nothing_raised { fixtures = create_fixtures(name) }
35 assert_kind_of(Fixtures, fixtures)
36 fixtures.each { |name, fixture|
37 fixture.each { |key, value|
38 assert_match(MATCH_ATTRIBUTE_NAME, key)
39 }
40 }
41 end
42 end
43
44 def test_multiple_clean_fixtures
45 fixtures_array = nil
46 assert_nothing_raised { fixtures_array = create_fixtures(*FIXTURES) }
47 assert_kind_of(Array, fixtures_array)
48 fixtures_array.each { |fixtures| assert_kind_of(Fixtures, fixtures) }
49 end
50
51 def test_attributes
52 topics = create_fixtures("topics")
53 assert_equal("The First Topic", topics["first"]["title"])
54 assert_nil(topics["second"]["author_email_address"])
55 end
56
57 def test_inserts
58 topics = create_fixtures("topics")
59 first_row = ActiveRecord::Base.connection.select_one("SELECT * FROM topics WHERE author_name = 'David'")
60 assert_equal("The First Topic", first_row["title"])
61
62 second_row = ActiveRecord::Base.connection.select_one("SELECT * FROM topics WHERE author_name = 'Mary'")
63 assert_nil(second_row["author_email_address"])
64 end
65
66 if ActiveRecord::Base.connection.supports_migrations?
67 def test_inserts_with_pre_and_suffix
68 # Reset cache to make finds on the new table work
69 Fixtures.reset_cache
70
71 ActiveRecord::Base.connection.create_table :prefix_topics_suffix do |t|
72 t.column :title, :string
73 t.column :author_name, :string
74 t.column :author_email_address, :string
75 t.column :written_on, :datetime
76 t.column :bonus_time, :time
77 t.column :last_read, :date
78 t.column :content, :string
79 t.column :approved, :boolean, :default => true
80 t.column :replies_count, :integer, :default => 0
81 t.column :parent_id, :integer
82 t.column :type, :string, :limit => 50
83 end
84
85 # Store existing prefix/suffix
86 old_prefix = ActiveRecord::Base.table_name_prefix
87 old_suffix = ActiveRecord::Base.table_name_suffix
88
89 # Set a prefix/suffix we can test against
90 ActiveRecord::Base.table_name_prefix = 'prefix_'
91 ActiveRecord::Base.table_name_suffix = '_suffix'
92
93 topics = create_fixtures("topics")
94
95 first_row = ActiveRecord::Base.connection.select_one("SELECT * FROM prefix_topics_suffix WHERE author_name = 'David'")
96 assert_equal("The First Topic", first_row["title"])
97
98 second_row = ActiveRecord::Base.connection.select_one("SELECT * FROM prefix_topics_suffix WHERE author_name = 'Mary'")
99 assert_nil(second_row["author_email_address"])
100
101 # This checks for a caching problem which causes a bug in the fixtures
102 # class-level configuration helper.
103 assert_not_nil topics, "Fixture data inserted, but fixture objects not returned from create"
104 ensure
105 # Restore prefix/suffix to its previous values
106 ActiveRecord::Base.table_name_prefix = old_prefix
107 ActiveRecord::Base.table_name_suffix = old_suffix
108
109 ActiveRecord::Base.connection.drop_table :prefix_topics_suffix rescue nil
110 end
111 end
112
113 def test_insert_with_datetime
114 topics = create_fixtures("tasks")
115 first = Task.find(1)
116 assert first
117 end
118
119 def test_logger_level_invariant
120 level = ActiveRecord::Base.logger.level
121 create_fixtures('topics')
122 assert_equal level, ActiveRecord::Base.logger.level
123 end
124
125 def test_instantiation
126 topics = create_fixtures("topics")
127 assert_kind_of Topic, topics["first"].find
128 end
129
130 def test_complete_instantiation
131 assert_equal 4, @topics.size
132 assert_equal "The First Topic", @first.title
133 end
134
135 def test_fixtures_from_root_yml_with_instantiation
136 # assert_equal 2, @accounts.size
137 assert_equal 50, @unknown.credit_limit
138 end
139
140 def test_erb_in_fixtures
141 assert_equal 11, @developers.size
142 assert_equal "fixture_5", @dev_5.name
143 end
144
145 def test_empty_yaml_fixture
146 assert_not_nil Fixtures.new( Account.connection, "accounts", 'Account', FIXTURES_ROOT + "/naked/yml/accounts")
147 end
148
149 def test_empty_yaml_fixture_with_a_comment_in_it
150 assert_not_nil Fixtures.new( Account.connection, "companies", 'Company', FIXTURES_ROOT + "/naked/yml/companies")
151 end
152
153 def test_dirty_dirty_yaml_file
154 assert_raise(Fixture::FormatError) do
155 Fixtures.new( Account.connection, "courses", 'Course', FIXTURES_ROOT + "/naked/yml/courses")
156 end
157 end
158
159 def test_empty_csv_fixtures
160 assert_not_nil Fixtures.new( Account.connection, "accounts", 'Account', FIXTURES_ROOT + "/naked/csv/accounts")
161 end
162
163 def test_omap_fixtures
164 assert_nothing_raised do
165 fixtures = Fixtures.new(Account.connection, 'categories', 'Category', FIXTURES_ROOT + "/categories_ordered")
166
167 i = 0
168 fixtures.each do |name, fixture|
169 assert_equal "fixture_no_#{i}", name
170 assert_equal "Category #{i}", fixture['name']
171 i += 1
172 end
173 end
174 end
175
176 def test_yml_file_in_subdirectory
177 assert_equal(categories(:sub_special_1).name, "A special category in a subdir file")
178 assert_equal(categories(:sub_special_1).class, SpecialCategory)
179 end
180
181 def test_subsubdir_file_with_arbitrary_name
182 assert_equal(categories(:sub_special_3).name, "A special category in an arbitrarily named subsubdir file")
183 assert_equal(categories(:sub_special_3).class, SpecialCategory)
184 end
185
186 def test_binary_in_fixtures
187 assert_equal 1, @binaries.size
188 data = File.read(ASSETS_ROOT + "/flowers.jpg")
189 data.force_encoding('ASCII-8BIT') if data.respond_to?(:force_encoding)
190 data.freeze
191 assert_equal data, @flowers.data
192 end
193 end
194
195 if Account.connection.respond_to?(:reset_pk_sequence!)
196 class FixturesResetPkSequenceTest < ActiveRecord::TestCase
197 fixtures :accounts
198 fixtures :companies
199
200 def setup
201 @instances = [Account.new(:credit_limit => 50), Company.new(:name => 'RoR Consulting')]
202 Fixtures.reset_cache # make sure tables get reinitialized
203 end
204
205 def test_resets_to_min_pk_with_specified_pk_and_sequence
206 @instances.each do |instance|
207 model = instance.class
208 model.delete_all
209 model.connection.reset_pk_sequence!(model.table_name, model.primary_key, model.sequence_name)
210
211 instance.save!
212 assert_equal 1, instance.id, "Sequence reset for #{model.table_name} failed."
213 end
214 end
215
216 def test_resets_to_min_pk_with_default_pk_and_sequence
217 @instances.each do |instance|
218 model = instance.class
219 model.delete_all
220 model.connection.reset_pk_sequence!(model.table_name)
221
222 instance.save!
223 assert_equal 1, instance.id, "Sequence reset for #{model.table_name} failed."
224 end
225 end
226
227 def test_create_fixtures_resets_sequences_when_not_cached
228 @instances.each do |instance|
229 max_id = create_fixtures(instance.class.table_name).inject(0) do |max_id, (name, fixture)|
230 fixture_id = fixture['id'].to_i
231 fixture_id > max_id ? fixture_id : max_id
232 end
233
234 # Clone the last fixture to check that it gets the next greatest id.
235 instance.save!
236 assert_equal max_id + 1, instance.id, "Sequence reset for #{instance.class.table_name} failed."
237 end
238 end
239 end
240 end
241
242 class FixturesWithoutInstantiationTest < ActiveRecord::TestCase
243 self.use_instantiated_fixtures = false
244 fixtures :topics, :developers, :accounts
245
246 def test_without_complete_instantiation
247 assert_nil @first
248 assert_nil @topics
249 assert_nil @developers
250 assert_nil @accounts
251 end
252
253 def test_fixtures_from_root_yml_without_instantiation
254 assert_nil @unknown
255 end
256
257 def test_accessor_methods
258 assert_equal "The First Topic", topics(:first).title
259 assert_equal "Jamis", developers(:jamis).name
260 assert_equal 50, accounts(:signals37).credit_limit
261 end
262
263 def test_accessor_methods_with_multiple_args
264 assert_equal 2, topics(:first, :second).size
265 assert_raise(StandardError) { topics([:first, :second]) }
266 end
267
268 def test_reloading_fixtures_through_accessor_methods
269 assert_equal "The First Topic", topics(:first).title
270 @loaded_fixtures['topics']['first'].expects(:find).returns(stub(:title => "Fresh Topic!"))
271 assert_equal "Fresh Topic!", topics(:first, true).title
272 end
273 end
274
275 class FixturesWithoutInstanceInstantiationTest < ActiveRecord::TestCase
276 self.use_instantiated_fixtures = true
277 self.use_instantiated_fixtures = :no_instances
278
279 fixtures :topics, :developers, :accounts
280
281 def test_without_instance_instantiation
282 assert_nil @first
283 assert_not_nil @topics
284 assert_not_nil @developers
285 assert_not_nil @accounts
286 end
287 end
288
289 class TransactionalFixturesTest < ActiveRecord::TestCase
290 self.use_instantiated_fixtures = true
291 self.use_transactional_fixtures = true
292
293 fixtures :topics
294
295 def test_destroy
296 assert_not_nil @first
297 @first.destroy
298 end
299
300 def test_destroy_just_kidding
301 assert_not_nil @first
302 end
303 end
304
305 class MultipleFixturesTest < ActiveRecord::TestCase
306 fixtures :topics
307 fixtures :developers, :accounts
308
309 def test_fixture_table_names
310 assert_equal %w(topics developers accounts), fixture_table_names
311 end
312 end
313
314 class SetupTest < ActiveRecord::TestCase
315 # fixtures :topics
316
317 def setup
318 @first = true
319 end
320
321 def test_nothing
322 end
323 end
324
325 class SetupSubclassTest < SetupTest
326 def setup
327 super
328 @second = true
329 end
330
331 def test_subclassing_should_preserve_setups
332 assert @first
333 assert @second
334 end
335 end
336
337
338 class OverlappingFixturesTest < ActiveRecord::TestCase
339 fixtures :topics, :developers
340 fixtures :developers, :accounts
341
342 def test_fixture_table_names
343 assert_equal %w(topics developers accounts), fixture_table_names
344 end
345 end
346
347 class ForeignKeyFixturesTest < ActiveRecord::TestCase
348 fixtures :fk_test_has_pk, :fk_test_has_fk
349
350 # if foreign keys are implemented and fixtures
351 # are not deleted in reverse order then this test
352 # case will raise StatementInvalid
353
354 def test_number1
355 assert true
356 end
357
358 def test_number2
359 assert true
360 end
361 end
362
363 class CheckSetTableNameFixturesTest < ActiveRecord::TestCase
364 set_fixture_class :funny_jokes => 'Joke'
365 fixtures :funny_jokes
366 # Set to false to blow away fixtures cache and ensure our fixtures are loaded
367 # and thus takes into account our set_fixture_class
368 self.use_transactional_fixtures = false
369
370 def test_table_method
371 assert_kind_of Joke, funny_jokes(:a_joke)
372 end
373 end
374
375 class FixtureNameIsNotTableNameFixturesTest < ActiveRecord::TestCase
376 set_fixture_class :items => Book
377 fixtures :items
378 # Set to false to blow away fixtures cache and ensure our fixtures are loaded
379 # and thus takes into account our set_fixture_class
380 self.use_transactional_fixtures = false
381
382 def test_named_accessor
383 assert_kind_of Book, items(:dvd)
384 end
385 end
386
387 class FixtureNameIsNotTableNameMultipleFixturesTest < ActiveRecord::TestCase
388 set_fixture_class :items => Book, :funny_jokes => Joke
389 fixtures :items, :funny_jokes
390 # Set to false to blow away fixtures cache and ensure our fixtures are loaded
391 # and thus takes into account our set_fixture_class
392 self.use_transactional_fixtures = false
393
394 def test_named_accessor_of_differently_named_fixture
395 assert_kind_of Book, items(:dvd)
396 end
397
398 def test_named_accessor_of_same_named_fixture
399 assert_kind_of Joke, funny_jokes(:a_joke)
400 end
401 end
402
403 class CustomConnectionFixturesTest < ActiveRecord::TestCase
404 set_fixture_class :courses => Course
405 fixtures :courses
406 # Set to false to blow away fixtures cache and ensure our fixtures are loaded
407 # and thus takes into account our set_fixture_class
408 self.use_transactional_fixtures = false
409
410 def test_connection
411 assert_kind_of Course, courses(:ruby)
412 assert_equal Course.connection, courses(:ruby).connection
413 end
414 end
415
416 class InvalidTableNameFixturesTest < ActiveRecord::TestCase
417 fixtures :funny_jokes
418 # Set to false to blow away fixtures cache and ensure our fixtures are loaded
419 # and thus takes into account our lack of set_fixture_class
420 self.use_transactional_fixtures = false
421
422 def test_raises_error
423 assert_raise FixtureClassNotFound do
424 funny_jokes(:a_joke)
425 end
426 end
427 end
428
429 class CheckEscapedYamlFixturesTest < ActiveRecord::TestCase
430 set_fixture_class :funny_jokes => 'Joke'
431 fixtures :funny_jokes
432 # Set to false to blow away fixtures cache and ensure our fixtures are loaded
433 # and thus takes into account our set_fixture_class
434 self.use_transactional_fixtures = false
435
436 def test_proper_escaped_fixture
437 assert_equal "The \\n Aristocrats\nAte the candy\n", funny_jokes(:another_joke).name
438 end
439 end
440
441 class DevelopersProject; end
442 class ManyToManyFixturesWithClassDefined < ActiveRecord::TestCase
443 fixtures :developers_projects
444
445 def test_this_should_run_cleanly
446 assert true
447 end
448 end
449
450 class FixturesBrokenRollbackTest < ActiveRecord::TestCase
451 def blank_setup; end
452 alias_method :ar_setup_fixtures, :setup_fixtures
453 alias_method :setup_fixtures, :blank_setup
454 alias_method :setup, :blank_setup
455
456 def blank_teardown; end
457 alias_method :ar_teardown_fixtures, :teardown_fixtures
458 alias_method :teardown_fixtures, :blank_teardown
459 alias_method :teardown, :blank_teardown
460
461 def test_no_rollback_in_teardown_unless_transaction_active
462 assert_equal 0, ActiveRecord::Base.connection.open_transactions
463 assert_raise(RuntimeError) { ar_setup_fixtures }
464 assert_equal 0, ActiveRecord::Base.connection.open_transactions
465 assert_nothing_raised { ar_teardown_fixtures }
466 assert_equal 0, ActiveRecord::Base.connection.open_transactions
467 end
468
469 private
470 def load_fixtures
471 raise 'argh'
472 end
473 end
474
475 class LoadAllFixturesTest < ActiveRecord::TestCase
476 self.fixture_path = FIXTURES_ROOT + "/all"
477 fixtures :all
478
479 def test_all_there
480 assert_equal %w(developers people tasks), fixture_table_names.sort
481 end
482 end
483
484 class FasterFixturesTest < ActiveRecord::TestCase
485 fixtures :categories, :authors
486
487 def load_extra_fixture(name)
488 fixture = create_fixtures(name)
489 assert fixture.is_a?(Fixtures)
490 @loaded_fixtures[fixture.table_name] = fixture
491 end
492
493 def test_cache
494 assert Fixtures.fixture_is_cached?(ActiveRecord::Base.connection, 'categories')
495 assert Fixtures.fixture_is_cached?(ActiveRecord::Base.connection, 'authors')
496
497 assert_no_queries do
498 create_fixtures('categories')
499 create_fixtures('authors')
500 end
501
502 load_extra_fixture('posts')
503 assert Fixtures.fixture_is_cached?(ActiveRecord::Base.connection, 'posts')
504 self.class.setup_fixture_accessors('posts')
505 assert_equal 'Welcome to the weblog', posts(:welcome).title
506 end
507 end
508
509 class FoxyFixturesTest < ActiveRecord::TestCase
510 fixtures :parrots, :parrots_pirates, :pirates, :treasures, :mateys, :ships, :computers, :developers
511
512 def test_identifies_strings
513 assert_equal(Fixtures.identify("foo"), Fixtures.identify("foo"))
514 assert_not_equal(Fixtures.identify("foo"), Fixtures.identify("FOO"))
515 end
516
517 def test_identifies_symbols
518 assert_equal(Fixtures.identify(:foo), Fixtures.identify(:foo))
519 end
520
521 TIMESTAMP_COLUMNS = %w(created_at created_on updated_at updated_on)
522
523 def test_populates_timestamp_columns
524 TIMESTAMP_COLUMNS.each do |property|
525 assert_not_nil(parrots(:george).send(property), "should set #{property}")
526 end
527 end
528
529 def test_does_not_populate_timestamp_columns_if_model_has_set_record_timestamps_to_false
530 TIMESTAMP_COLUMNS.each do |property|
531 assert_nil(ships(:black_pearl).send(property), "should not set #{property}")
532 end
533 end
534
535 def test_populates_all_columns_with_the_same_time
536 last = nil
537
538 TIMESTAMP_COLUMNS.each do |property|
539 current = parrots(:george).send(property)
540 last ||= current
541
542 assert_equal(last, current)
543 last = current
544 end
545 end
546
547 def test_only_populates_columns_that_exist
548 assert_not_nil(pirates(:blackbeard).created_on)
549 assert_not_nil(pirates(:blackbeard).updated_on)
550 end
551
552 def test_preserves_existing_fixture_data
553 assert_equal(2.weeks.ago.to_date, pirates(:redbeard).created_on.to_date)
554 assert_equal(2.weeks.ago.to_date, pirates(:redbeard).updated_on.to_date)
555 end
556
557 def test_generates_unique_ids
558 assert_not_nil(parrots(:george).id)
559 assert_not_equal(parrots(:george).id, parrots(:louis).id)
560 end
561
562 def test_automatically_sets_primary_key
563 assert_not_nil(ships(:black_pearl))
564 end
565
566 def test_preserves_existing_primary_key
567 assert_equal(2, ships(:interceptor).id)
568 end
569
570 def test_resolves_belongs_to_symbols
571 assert_equal(parrots(:george), pirates(:blackbeard).parrot)
572 end
573
574 def test_ignores_belongs_to_symbols_if_association_and_foreign_key_are_named_the_same
575 assert_equal(developers(:david), computers(:workstation).developer)
576 end
577
578 def test_supports_join_tables
579 assert(pirates(:blackbeard).parrots.include?(parrots(:george)))
580 assert(pirates(:blackbeard).parrots.include?(parrots(:louis)))
581 assert(parrots(:george).pirates.include?(pirates(:blackbeard)))
582 end
583
584 def test_supports_inline_habtm
585 assert(parrots(:george).treasures.include?(treasures(:diamond)))
586 assert(parrots(:george).treasures.include?(treasures(:sapphire)))
587 assert(!parrots(:george).treasures.include?(treasures(:ruby)))
588 end
589
590 def test_supports_inline_habtm_with_specified_id
591 assert(parrots(:polly).treasures.include?(treasures(:ruby)))
592 assert(parrots(:polly).treasures.include?(treasures(:sapphire)))
593 assert(!parrots(:polly).treasures.include?(treasures(:diamond)))
594 end
595
596 def test_supports_yaml_arrays
597 assert(parrots(:louis).treasures.include?(treasures(:diamond)))
598 assert(parrots(:louis).treasures.include?(treasures(:sapphire)))
599 end
600
601 def test_strips_DEFAULTS_key
602 assert_raise(StandardError) { parrots(:DEFAULTS) }
603
604 # this lets us do YAML defaults and not have an extra fixture entry
605 %w(sapphire ruby).each { |t| assert(parrots(:davey).treasures.include?(treasures(t))) }
606 end
607
608 def test_supports_label_interpolation
609 assert_equal("frederick", parrots(:frederick).name)
610 end
611
612 def test_supports_polymorphic_belongs_to
613 assert_equal(pirates(:redbeard), treasures(:sapphire).looter)
614 assert_equal(parrots(:louis), treasures(:ruby).looter)
615 end
616
617 def test_only_generates_a_pk_if_necessary
618 m = Matey.find(:first)
619 m.pirate = pirates(:blackbeard)
620 m.target = pirates(:redbeard)
621 end
622
623 def test_supports_sti
624 assert_kind_of DeadParrot, parrots(:polly)
625 assert_equal pirates(:blackbeard), parrots(:polly).killer
626 end
627 end
628
629 class ActiveSupportSubclassWithFixturesTest < ActiveRecord::TestCase
630 fixtures :parrots
631
632 # This seemingly useless assertion catches a bug that caused the fixtures
633 # setup code call nil[]
634 def test_foo
635 assert_equal parrots(:louis), Parrot.find_by_name("King Louis")
636 end
637 end
638
639 class FixtureLoadingTest < ActiveRecord::TestCase
640 def test_logs_message_for_failed_dependency_load
641 ActiveRecord::TestCase.expects(:require_dependency).with(:does_not_exist).raises(LoadError)
642 ActiveRecord::Base.logger.expects(:warn)
643 ActiveRecord::TestCase.try_to_load_dependency(:does_not_exist)
644 end
645
646 def test_does_not_logs_message_for_successful_dependency_load
647 ActiveRecord::TestCase.expects(:require_dependency).with(:works_out_fine)
648 ActiveRecord::Base.logger.expects(:warn).never
649 ActiveRecord::TestCase.try_to_load_dependency(:works_out_fine)
650 end
651 end