diff options
author | Jeremy Kemper <jeremy@bitsweat.net> | 2007-09-28 14:56:07 +0000 |
---|---|---|
committer | Jeremy Kemper <jeremy@bitsweat.net> | 2007-09-28 14:56:07 +0000 |
commit | b095ce63f2dbc88c1cb6da018d02e3707b8b48b9 (patch) | |
tree | 72fe005aaf84d6a4bc91047a41b0b22dc88116ae | |
parent | e3b49c052b497827c9f058feaa066bbfe184f4da (diff) | |
download | rails-b095ce63f2dbc88c1cb6da018d02e3707b8b48b9.tar.gz rails-b095ce63f2dbc88c1cb6da018d02e3707b8b48b9.tar.bz2 rails-b095ce63f2dbc88c1cb6da018d02e3707b8b48b9.zip |
Fixtures test fixes and general cleanup. Closes #9682 [norbert]
git-svn-id: http://svn-commit.rubyonrails.org/rails/trunk@7667 5ecf4fe2-1ee6-0310-87b1-e25e094e27de
-rwxr-xr-x | activerecord/lib/active_record/fixtures.rb | 188 | ||||
-rw-r--r-- | activerecord/test/connection_test_mysql.rb | 1 | ||||
-rwxr-xr-x | activerecord/test/fixtures_test.rb | 54 |
3 files changed, 113 insertions, 130 deletions
diff --git a/activerecord/lib/active_record/fixtures.rb b/activerecord/lib/active_record/fixtures.rb index 1eded9fd39..6013d6e156 100755 --- a/activerecord/lib/active_record/fixtures.rb +++ b/activerecord/lib/active_record/fixtures.rb @@ -125,7 +125,7 @@ end # ... # # By adding a "fixtures" method to the test case and passing it a list of symbols (only one is shown here tho), we trigger -# the testing environment to automatically load the appropriate fixtures into the database before each test. +# the testing environment to automatically load the appropriate fixtures into the database before each test. # To ensure consistent data, the environment deletes the fixtures before running the load. # # In addition to being available in the database, the fixtures are also loaded into a hash stored in an instance variable @@ -151,7 +151,7 @@ end # self.use_instantiated_fixtures = false # # - to keep the fixture instance (@web_sites) available, but do not automatically 'find' each instance: -# self.use_instantiated_fixtures = :no_instances +# self.use_instantiated_fixtures = :no_instances # # Even if auto-instantiated fixtures are disabled, you can still access them # by name via special dynamic methods. Each method has the same name as the @@ -183,7 +183,7 @@ end # # = Transactional fixtures # -# TestCases can use begin+rollback to isolate their changes to the database instead of having to delete+insert for every test case. +# TestCases can use begin+rollback to isolate their changes to the database instead of having to delete+insert for every test case. # They can also turn off auto-instantiation of fixture data since the feature is costly and often unused. # # class FooTest < Test::Unit::TestCase @@ -203,32 +203,33 @@ end # end # end # -# If you preload your test database with all fixture data (probably in the Rakefile task) and use transactional fixtures, +# If you preload your test database with all fixture data (probably in the Rakefile task) and use transactional fixtures, # then you may omit all fixtures declarations in your test cases since all the data's already there and every case rolls back its changes. # -# In order to use instantiated fixtures with preloaded data, set +self.pre_loaded_fixtures+ to true. This will provide +# In order to use instantiated fixtures with preloaded data, set +self.pre_loaded_fixtures+ to true. This will provide # access to fixture data for every table that has been loaded through fixtures (depending on the value of +use_instantiated_fixtures+) # -# When *not* to use transactional fixtures: -# 1. You're testing whether a transaction works correctly. Nested transactions don't commit until all parent transactions commit, -# particularly, the fixtures transaction which is begun in setup and rolled back in teardown. Thus, you won't be able to verify -# the results of your transaction until Active Record supports nested transactions or savepoints (in progress.) -# 2. Your database does not support transactions. Every Active Record database supports transactions except MySQL MyISAM. +# When *not* to use transactional fixtures: +# 1. You're testing whether a transaction works correctly. Nested transactions don't commit until all parent transactions commit, +# particularly, the fixtures transaction which is begun in setup and rolled back in teardown. Thus, you won't be able to verify +# the results of your transaction until Active Record supports nested transactions or savepoints (in progress.) +# 2. Your database does not support transactions. Every Active Record database supports transactions except MySQL MyISAM. # Use InnoDB, MaxDB, or NDB instead. class Fixtures < YAML::Omap DEFAULT_FILTER_RE = /\.ya?ml$/ @@all_cached_fixtures = {} - - def self.reset_cache(connection=ActiveRecord::Base.connection) + + def self.reset_cache(connection = nil) + connection ||= ActiveRecord::Base.connection @@all_cached_fixtures[connection.object_id] = {} end - + def self.cache_for_connection(connection) @@all_cached_fixtures[connection.object_id] ||= {} @@all_cached_fixtures[connection.object_id] end - + def self.fixture_is_cached?(connection, table_name) cache_for_connection(connection)[table_name] end @@ -243,10 +244,10 @@ class Fixtures < YAML::Omap end def self.cache_fixtures(connection, fixtures) - cache_for_connection(connection).merge! fixtures.index_by(&:table_name) + cache_for_connection(connection).update(fixtures.index_by(&:table_name)) end - - def self.instantiate_fixtures(object, table_name, fixtures, load_instances=true) + + def self.instantiate_fixtures(object, table_name, fixtures, load_instances = true) object.instance_variable_set "@#{table_name.to_s.gsub('.','_')}", fixtures if load_instances ActiveRecord::Base.silence do @@ -261,12 +262,12 @@ class Fixtures < YAML::Omap end end - def self.instantiate_all_loaded_fixtures(object, load_instances=true) + def self.instantiate_all_loaded_fixtures(object, load_instances = true) all_loaded_fixtures.each do |table_name, fixtures| Fixtures.instantiate_fixtures(object, table_name, fixtures, load_instances) end end - + cattr_accessor :all_loaded_fixtures self.all_loaded_fixtures = {} @@ -274,17 +275,17 @@ class Fixtures < YAML::Omap table_names = [table_names].flatten.map { |n| n.to_s } connection = block_given? ? yield : ActiveRecord::Base.connection - table_names_to_fetch = table_names.reject {|table_name| fixture_is_cached?(connection, table_name)} - + table_names_to_fetch = table_names.reject { |table_name| fixture_is_cached?(connection, table_name) } + unless table_names_to_fetch.empty? ActiveRecord::Base.silence do fixtures_map = {} fixtures = table_names_to_fetch.map do |table_name| fixtures_map[table_name] = Fixtures.new(connection, File.split(table_name.to_s).last, class_names[table_name.to_sym], File.join(fixtures_directory, table_name.to_s)) - end + end - all_loaded_fixtures.merge! fixtures_map + all_loaded_fixtures.update(fixtures_map) connection.transaction(Thread.current['open_transactions'].to_i == 0) do fixtures.reverse.each { |fixture| fixture.delete_existing_fixtures } @@ -297,19 +298,18 @@ class Fixtures < YAML::Omap end end end - + cache_fixtures(connection, fixtures) end end - return cached_fixtures(connection, table_names) + cached_fixtures(connection, table_names) end - attr_reader :table_name def initialize(connection, table_name, class_name, fixture_path, file_filter = DEFAULT_FILTER_RE) @connection, @table_name, @fixture_path, @file_filter = connection, table_name, fixture_path, file_filter - @class_name = class_name || + @class_name = class_name || (ActiveRecord::Base.pluralize_table_names ? @table_name.singularize.camelize : @table_name.camelize) @table_name = ActiveRecord::Base.table_name_prefix + @table_name + ActiveRecord::Base.table_name_suffix @table_name = class_name.table_name if class_name.respond_to?(:table_name) @@ -323,58 +323,16 @@ class Fixtures < YAML::Omap def insert_fixtures values.each do |fixture| - @connection.insert_fixture fixture, @table_name + @connection.insert_fixture(fixture, @table_name) end end - private def read_fixture_files if File.file?(yaml_file_path) - # YAML fixtures - yaml_string = "" - Dir["#{@fixture_path}/**/*.yml"].select {|f| test(?f,f) }.each do |subfixture_path| - yaml_string << IO.read(subfixture_path) - end - yaml_string << IO.read(yaml_file_path) - - begin - yaml = YAML::load(erb_render(yaml_string)) - rescue Exception=>boom - raise Fixture::FormatError, "a YAML error occurred parsing #{yaml_file_path}. Please note that YAML must be consistently indented using spaces. Tabs are not allowed. Please have a look at http://www.yaml.org/faq.html\nThe exact error was:\n #{boom.class}: #{boom}" - end - - if yaml - # If the file is an ordered map, extract its children. - yaml_value = - if yaml.respond_to?(:type_id) && yaml.respond_to?(:value) - yaml.value - else - [yaml] - end - - yaml_value.each do |fixture| - fixture.each do |name, data| - unless data - raise Fixture::FormatError, "Bad data for #{@class_name} fixture named #{name} (nil)" - end - - self[name] = Fixture.new(data, @class_name) - end - end - end + read_yaml_fixture_files elsif File.file?(csv_file_path) - # CSV fixtures - reader = CSV::Reader.create(erb_render(IO.read(csv_file_path))) - header = reader.shift - i = 0 - reader.each do |row| - data = {} - row.each_with_index { |cell, j| data[header[j].to_s.strip] = cell.to_s.strip } - self["#{Inflector::underscore(@class_name)}_#{i+=1}"]= Fixture.new(data, @class_name) - end - elsif File.file?(deprecated_yaml_file_path) - raise Fixture::FormatError, ".yml extension required: rename #{deprecated_yaml_file_path} to #{yaml_file_path}" + read_csv_fixture_files else # Standard fixtures Dir.entries(@fixture_path).each do |file| @@ -386,12 +344,47 @@ class Fixtures < YAML::Omap end end - def yaml_file_path - "#{@fixture_path}.yml" + def read_yaml_fixture_files + yaml_string = "" + Dir["#{@fixture_path}/**/*.yml"].select { |f| test(?f, f) }.each do |subfixture_path| + yaml_string << IO.read(subfixture_path) + end + yaml_string << IO.read(yaml_file_path) + + if yaml = parse_yaml_string(yaml_string) + # If the file is an ordered map, extract its children. + yaml_value = + if yaml.respond_to?(:type_id) && yaml.respond_to?(:value) + yaml.value + else + [yaml] + end + + yaml_value.each do |fixture| + fixture.each do |name, data| + unless data + raise Fixture::FormatError, "Bad data for #{@class_name} fixture named #{name} (nil)" + end + + self[name] = Fixture.new(data, @class_name) + end + end + end end - def deprecated_yaml_file_path - "#{@fixture_path}.yaml" + def read_csv_fixture_files + reader = CSV::Reader.create(erb_render(IO.read(csv_file_path))) + header = reader.shift + i = 0 + reader.each do |row| + data = {} + row.each_with_index { |cell, j| data[header[j].to_s.strip] = cell.to_s.strip } + self["#{Inflector::underscore(@class_name)}_#{i+=1}"]= Fixture.new(data, @class_name) + end + end + + def yaml_file_path + "#{@fixture_path}.yml" end def csv_file_path @@ -402,6 +395,12 @@ class Fixtures < YAML::Omap File.basename(@fixture_path).split(".").first end + def parse_yaml_string(fixture_content) + YAML::load(erb_render(fixture_content)) + rescue => error + raise Fixture::FormatError, "a YAML error occurred parsing #{yaml_file_path}. Please note that YAML must be consistently indented using spaces. Tabs are not allowed. Please have a look at http://www.yaml.org/faq.html\nThe exact error was:\n #{error.class}: #{error}" + end + def erb_render(fixture_content) ERB.new(fixture_content).result end @@ -409,9 +408,11 @@ end class Fixture #:nodoc: include Enumerable - class FixtureError < StandardError#:nodoc: + + class FixtureError < StandardError #:nodoc: end - class FormatError < FixtureError#:nodoc: + + class FormatError < FixtureError #:nodoc: end attr_reader :class_name @@ -499,14 +500,14 @@ module Test #:nodoc: self.use_transactional_fixtures = false self.use_instantiated_fixtures = true self.pre_loaded_fixtures = false - + @@already_loaded_fixtures = {} self.fixture_class_names = {} - + def self.set_fixture_class(class_names = {}) self.fixture_class_names = self.fixture_class_names.merge(class_names) end - + def self.fixtures(*table_names) if table_names.first == :all table_names = Dir["#{fixture_path}/*.yml"] + Dir["#{fixture_path}/*.csv"] @@ -520,8 +521,8 @@ module Test #:nodoc: setup_fixture_accessors(table_names) end - def self.require_fixture_classes(table_names=nil) - (table_names || fixture_table_names).each do |table_name| + def self.require_fixture_classes(table_names = nil) + (table_names || fixture_table_names).each do |table_name| file_name = table_name.to_s file_name = file_name.singularize if ActiveRecord::Base.pluralize_table_names begin @@ -539,7 +540,7 @@ module Test #:nodoc: define_method(table_name) do |*fixtures| force_reload = fixtures.pop if fixtures.last == true || fixtures.last == :reload - @fixture_cache[table_name] ||= Hash.new + @fixture_cache[table_name] ||= {} instances = fixtures.map do |fixture| @fixture_cache[table_name].delete(fixture) if force_reload @@ -575,15 +576,11 @@ module Test #:nodoc: return unless defined?(ActiveRecord::Base) && !ActiveRecord::Base.configurations.blank? if pre_loaded_fixtures && !use_transactional_fixtures - raise RuntimeError, 'pre_loaded_fixtures requires use_transactional_fixtures' + raise RuntimeError, 'pre_loaded_fixtures requires use_transactional_fixtures' end - @fixture_cache = Hash.new + @fixture_cache = {} - if !use_transactional_fixtures? - Fixtures.reset_cache #we don't want the test to use any of our cached data - end - # Load fixtures once and begin transaction. if use_transactional_fixtures? if @@already_loaded_fixtures[self.class] @@ -594,9 +591,9 @@ module Test #:nodoc: end ActiveRecord::Base.send :increment_open_transactions ActiveRecord::Base.connection.begin_db_transaction - # Load fixtures for every test. else + Fixtures.reset_cache @@already_loaded_fixtures[self.class] = nil load_fixtures end @@ -604,16 +601,15 @@ module Test #:nodoc: # Instantiate fixtures for every test if requested. instantiate_fixtures if use_instantiated_fixtures end - alias_method :setup, :setup_with_fixtures def teardown_with_fixtures return unless defined?(ActiveRecord::Base) && !ActiveRecord::Base.configurations.blank? - if !use_transactional_fixtures? - Fixtures.reset_cache #the non transactional test may have fiddled with the data - dump caches + unless use_transactional_fixtures? + Fixtures.reset_cache end - + # Rollback changes if a transaction is active. if use_transactional_fixtures? && Thread.current['open_transactions'] != 0 ActiveRecord::Base.connection.rollback_db_transaction @@ -621,7 +617,6 @@ module Test #:nodoc: end ActiveRecord::Base.verify_active_connections! end - alias_method :teardown, :teardown_with_fixtures def self.method_added(method) @@ -681,6 +676,5 @@ module Test #:nodoc: use_instantiated_fixtures != :no_instances end end - end end diff --git a/activerecord/test/connection_test_mysql.rb b/activerecord/test/connection_test_mysql.rb index fe132cb5ed..e3f589c4af 100644 --- a/activerecord/test/connection_test_mysql.rb +++ b/activerecord/test/connection_test_mysql.rb @@ -1,7 +1,6 @@ require "#{File.dirname(__FILE__)}/abstract_unit" class MysqlConnectionTest < Test::Unit::TestCase - self.use_transactional_fixtures = false def setup @connection = ActiveRecord::Base.connection end diff --git a/activerecord/test/fixtures_test.rb b/activerecord/test/fixtures_test.rb index c666005b69..66fc4ed65d 100755 --- a/activerecord/test/fixtures_test.rb +++ b/activerecord/test/fixtures_test.rb @@ -49,16 +49,18 @@ class FixturesTest < Test::Unit::TestCase def test_inserts topics = create_fixtures("topics") - firstRow = ActiveRecord::Base.connection.select_one("SELECT * FROM topics WHERE author_name = 'David'") - assert_equal("The First Topic", firstRow["title"]) + first_row = ActiveRecord::Base.connection.select_one("SELECT * FROM topics WHERE author_name = 'David'") + assert_equal("The First Topic", first_row["title"]) - secondRow = ActiveRecord::Base.connection.select_one("SELECT * FROM topics WHERE author_name = 'Mary'") - assert_nil(secondRow["author_email_address"]) + second_row = ActiveRecord::Base.connection.select_one("SELECT * FROM topics WHERE author_name = 'Mary'") + assert_nil(second_row["author_email_address"]) end if ActiveRecord::Base.connection.supports_migrations? def test_inserts_with_pre_and_suffix + # Reset cache to make finds on the new table work Fixtures.reset_cache + ActiveRecord::Base.connection.create_table :prefix_topics_suffix do |t| t.column :title, :string t.column :author_name, :string @@ -83,15 +85,15 @@ class FixturesTest < Test::Unit::TestCase topics = create_fixtures("topics") - firstRow = ActiveRecord::Base.connection.select_one("SELECT * FROM prefix_topics_suffix WHERE author_name = 'David'") - assert_equal("The First Topic", firstRow["title"]) + first_row = ActiveRecord::Base.connection.select_one("SELECT * FROM prefix_topics_suffix WHERE author_name = 'David'") + assert_equal("The First Topic", first_row["title"]) - secondRow = ActiveRecord::Base.connection.select_one("SELECT * FROM prefix_topics_suffix WHERE author_name = 'Mary'") - assert_nil(secondRow["author_email_address"]) + second_row = ActiveRecord::Base.connection.select_one("SELECT * FROM prefix_topics_suffix WHERE author_name = 'Mary'") + assert_nil(second_row["author_email_address"]) ensure # Restore prefix/suffix to its previous values - ActiveRecord::Base.table_name_prefix = old_prefix - ActiveRecord::Base.table_name_suffix = old_suffix + ActiveRecord::Base.table_name_prefix = old_prefix + ActiveRecord::Base.table_name_suffix = old_suffix ActiveRecord::Base.connection.drop_table :prefix_topics_suffix rescue nil end @@ -200,6 +202,7 @@ if Account.connection.respond_to?(:reset_pk_sequence!) def setup @instances = [Account.new(:credit_limit => 50), Company.new(:name => 'RoR Consulting')] + Fixtures.reset_cache # make sure tables get reinitialized end def test_resets_to_min_pk_with_specified_pk_and_sequence @@ -224,7 +227,7 @@ if Account.connection.respond_to?(:reset_pk_sequence!) end end - def test_create_fixtures_resets_sequences + def test_create_fixtures_resets_sequences_when_not_cached @instances.each do |instance| max_id = create_fixtures(instance.class.table_name).inject(0) do |max_id, (name, fixture)| fixture_id = fixture['id'].to_i @@ -239,7 +242,6 @@ if Account.connection.respond_to?(:reset_pk_sequence!) end end - class FixturesWithoutInstantiationTest < Test::Unit::TestCase self.use_instantiated_fixtures = false fixtures :topics, :developers, :accounts @@ -275,7 +277,6 @@ class FixturesWithoutInstantiationTest < Test::Unit::TestCase end end - class FixturesWithoutInstanceInstantiationTest < Test::Unit::TestCase self.use_instantiated_fixtures = true self.use_instantiated_fixtures = :no_instances @@ -290,7 +291,6 @@ class FixturesWithoutInstanceInstantiationTest < Test::Unit::TestCase end end - class TransactionalFixturesTest < Test::Unit::TestCase self.use_instantiated_fixtures = true self.use_transactional_fixtures = true @@ -307,7 +307,6 @@ class TransactionalFixturesTest < Test::Unit::TestCase end end - class MultipleFixturesTest < Test::Unit::TestCase fixtures :topics fixtures :developers, :accounts @@ -317,7 +316,6 @@ class MultipleFixturesTest < Test::Unit::TestCase end end - class OverlappingFixturesTest < Test::Unit::TestCase fixtures :topics, :developers fixtures :developers, :accounts @@ -327,7 +325,6 @@ class OverlappingFixturesTest < Test::Unit::TestCase end end - class ForeignKeyFixturesTest < Test::Unit::TestCase fixtures :fk_test_has_pk, :fk_test_has_fk @@ -347,7 +344,7 @@ end class SetTableNameFixturesTest < Test::Unit::TestCase set_fixture_class :funny_jokes => 'Joke' fixtures :funny_jokes - + def test_table_method assert_kind_of Joke, funny_jokes(:a_joke) end @@ -356,7 +353,7 @@ end class CustomConnectionFixturesTest < Test::Unit::TestCase set_fixture_class :courses => Course fixtures :courses - + def test_connection assert_kind_of Course, courses(:ruby) assert_equal Course.connection, courses(:ruby).connection @@ -382,17 +379,15 @@ class CheckEscapedYamlFixturesTest < Test::Unit::TestCase end end -class DevelopersProject; end; - +class DevelopersProject; end class ManyToManyFixturesWithClassDefined < Test::Unit::TestCase fixtures :developers_projects - + def test_this_should_run_cleanly assert true end end - class FixturesBrokenRollbackTest < Test::Unit::TestCase def blank_setup; end alias_method :ar_setup_with_fixtures, :setup_with_fixtures @@ -429,27 +424,22 @@ end class FasterFixturesTest < Test::Unit::TestCase fixtures :categories, :authors - - def run(*args, &block) - Fixtures.reset_cache - super(*args, &block) - end - + def load_extra_fixture(name) fixture = create_fixtures(name) assert fixture.is_a?(Fixtures) @loaded_fixtures[fixture.table_name] = fixture end - + def test_cache assert Fixtures.fixture_is_cached?(ActiveRecord::Base.connection, 'categories') assert Fixtures.fixture_is_cached?(ActiveRecord::Base.connection, 'authors') - + assert_no_queries do create_fixtures('categories') create_fixtures('authors') end - + load_extra_fixture('posts') assert Fixtures.fixture_is_cached?(ActiveRecord::Base.connection, 'posts') self.class.setup_fixture_accessors('posts') |