aboutsummaryrefslogtreecommitdiffstats
diff options
context:
space:
mode:
authorJeremy Kemper <jeremy@bitsweat.net>2007-09-28 14:56:07 +0000
committerJeremy Kemper <jeremy@bitsweat.net>2007-09-28 14:56:07 +0000
commitb095ce63f2dbc88c1cb6da018d02e3707b8b48b9 (patch)
tree72fe005aaf84d6a4bc91047a41b0b22dc88116ae
parente3b49c052b497827c9f058feaa066bbfe184f4da (diff)
downloadrails-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-xactiverecord/lib/active_record/fixtures.rb188
-rw-r--r--activerecord/test/connection_test_mysql.rb1
-rwxr-xr-xactiverecord/test/fixtures_test.rb54
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')