2004-11-23 20:04:44 -05:00
require 'erb'
require 'yaml'
2004-12-07 05:24:13 -05:00
require 'csv'
2008-01-05 08:34:15 -05:00
require 'active_support/test_case'
2004-11-23 20:04:44 -05:00
2008-01-05 08:34:15 -05:00
if RUBY_VERSION < '1.9'
module YAML #:nodoc:
class Omap #:nodoc:
def keys ; map { | k , v | k } end
def values ; map { | k , v | v } end
end
2005-10-16 12:00:16 -04:00
end
2005-10-13 15:06:15 -04:00
end
2007-12-08 23:37:46 -05:00
if defined? ActiveRecord
class FixtureClassNotFound < ActiveRecord :: ActiveRecordError #:nodoc:
end
else
class FixtureClassNotFound < StandardError #:nodoc:
end
2006-03-06 18:03:35 -05:00
end
2007-11-07 22:37:16 -05:00
# Fixtures are a way of organizing data that you want to test against; in short, sample data. They come in 3 flavors:
2004-11-23 20:04:44 -05:00
#
2004-12-07 05:24:13 -05:00
# 1. YAML fixtures
# 2. CSV fixtures
# 3. Single-file fixtures
2004-11-23 20:04:44 -05:00
#
2004-12-07 05:24:13 -05:00
# = YAML fixtures
2004-11-23 20:04:44 -05:00
#
2004-12-07 05:24:13 -05:00
# This type of fixture is in YAML format and the preferred default. YAML is a file format which describes data structures
2007-11-07 22:37:16 -05:00
# in a non-verbose, human-readable format. It ships with Ruby 1.8.1+.
2004-11-23 20:04:44 -05:00
#
2005-10-26 09:05:48 -04:00
# Unlike single-file fixtures, YAML fixtures are stored in a single file per model, which are placed in the directory appointed
2008-01-05 08:34:15 -05:00
# by <tt>ActiveSupport::TestCase.fixture_path=(path)</tt> (this is automatically configured for Rails, so you can just
2008-05-25 07:29:00 -04:00
# put your files in <tt><your-rails-app>/test/fixtures/</tt>). The fixture file ends with the <tt>.yml</tt> file extension (Rails example:
# <tt><your-rails-app>/test/fixtures/web_sites.yml</tt>). The format of a YAML fixture file looks like this:
2004-11-23 20:04:44 -05:00
#
2004-12-07 05:24:13 -05:00
# rubyonrails:
# id: 1
# name: Ruby on Rails
# url: http://www.rubyonrails.org
2004-11-23 20:04:44 -05:00
#
2004-12-07 05:24:13 -05:00
# google:
# id: 2
# name: Google
# url: http://www.google.com
#
2004-12-09 10:52:54 -05:00
# This YAML fixture file includes two fixtures. Each YAML fixture (ie. record) is given a name and is followed by an
2004-12-07 05:24:13 -05:00
# indented list of key/value pairs in the "key: value" format. Records are separated by a blank line for your viewing
2004-12-09 10:52:54 -05:00
# pleasure.
2004-12-07 05:24:13 -05:00
#
2005-10-14 22:01:38 -04:00
# Note that YAML fixtures are unordered. If you want ordered fixtures, use the omap YAML type. See http://yaml.org/type/omap.html
# for the specification. You will need ordered fixtures when you have foreign key constraints on keys in the same table.
# This is commonly needed for tree structures. Example:
#
# --- !omap
# - parent:
# id: 1
# parent_id: NULL
# title: Parent
# - child:
# id: 2
# parent_id: 1
# title: Child
#
2004-12-07 05:24:13 -05:00
# = CSV fixtures
#
2004-12-07 06:12:05 -05:00
# Fixtures can also be kept in the Comma Separated Value format. Akin to YAML fixtures, CSV fixtures are stored
2008-05-25 07:29:00 -04:00
# in a single file, but instead end with the <tt>.csv</tt> file extension
# (Rails example: <tt><your-rails-app>/test/fixtures/web_sites.csv</tt>).
2004-12-07 05:24:13 -05:00
#
2005-02-07 09:15:53 -05:00
# The format of this type of fixture file is much more compact than the others, but also a little harder to read by us
2004-12-07 05:24:13 -05:00
# humans. The first line of the CSV file is a comma-separated list of field names. The rest of the file is then comprised
# of the actual data (1 per line). Here's an example:
#
# id, name, url
# 1, Ruby On Rails, http://www.rubyonrails.org
# 2, Google, http://www.google.com
#
# Should you have a piece of data with a comma character in it, you can place double quotes around that value. If you
# need to use a double quote character, you must escape it with another double quote.
#
# Another unique attribute of the CSV fixture is that it has *no* fixture name like the other two formats. Instead, the
2004-12-09 10:52:54 -05:00
# fixture names are automatically generated by deriving the class name of the fixture file and adding an incrementing
2004-12-07 05:24:13 -05:00
# number to the end. In our example, the 1st fixture would be called "web_site_1" and the 2nd one would be called
# "web_site_2".
#
2004-12-09 10:52:54 -05:00
# Most databases and spreadsheets support exporting to CSV format, so this is a great format for you to choose if you
2004-12-07 05:24:13 -05:00
# have existing data somewhere already.
2004-12-09 10:52:54 -05:00
#
# = Single-file fixtures
2004-12-07 05:24:13 -05:00
#
2007-11-07 22:37:16 -05:00
# This type of fixture was the original format for Active Record that has since been deprecated in favor of the YAML and CSV formats.
2004-12-09 10:52:54 -05:00
# Fixtures for this format are created by placing text files in a sub-directory (with the name of the model) to the directory
2008-01-05 08:34:15 -05:00
# appointed by <tt>ActiveSupport::TestCase.fixture_path=(path)</tt> (this is automatically configured for Rails, so you can just
2008-05-25 07:29:00 -04:00
# put your files in <tt><your-rails-app>/test/fixtures/<your-model-name>/</tt> --
# like <tt><your-rails-app>/test/fixtures/web_sites/</tt> for the WebSite model).
2004-12-07 05:24:13 -05:00
#
# Each text file placed in this directory represents a "record". Usually these types of fixtures are named without
2008-05-25 07:29:00 -04:00
# extensions, but if you are on a Windows machine, you might consider adding <tt>.txt</tt> as the extension. Here's what the
2004-12-07 05:24:13 -05:00
# above example might look like:
#
# web_sites/google
# web_sites/yahoo.txt
# web_sites/ruby-on-rails
#
# The file format of a standard fixture is simple. Each line is a property (or column in db speak) and has the syntax
# of "name => value". Here's an example of the ruby-on-rails fixture above:
#
# id => 1
# name => Ruby on Rails
# url => http://www.rubyonrails.org
#
# = Using Fixtures
#
# Since fixtures are a testing construct, we use them in our unit and functional tests. There are two ways to use the
2007-11-07 22:37:16 -05:00
# fixtures, but first let's take a look at a sample unit test:
2004-12-07 05:24:13 -05:00
#
# require 'web_site'
#
2008-01-05 08:34:15 -05:00
# class WebSiteTest < ActiveSupport::TestCase
2004-12-07 05:24:13 -05:00
# def test_web_site_count
# assert_equal 2, WebSite.count
2004-12-09 10:52:54 -05:00
# end
2004-12-07 05:24:13 -05:00
# end
2004-12-09 10:52:54 -05:00
#
# As it stands, unless we pre-load the web_site table in our database with two records, this test will fail. Here's the
2004-12-07 05:24:13 -05:00
# easiest way to add fixtures to the database:
#
# ...
2008-01-05 08:34:15 -05:00
# class WebSiteTest < ActiveSupport::TestCase
2004-12-07 05:24:13 -05:00
# fixtures :web_sites # add more by separating the symbols with commas
# ...
2004-12-09 10:52:54 -05:00
#
2007-11-07 22:37:16 -05:00
# By adding a "fixtures" method to the test case and passing it a list of symbols (only one is shown here though), we trigger
2007-09-28 10:56:07 -04:00
# the testing environment to automatically load the appropriate fixtures into the database before each test.
2005-06-18 01:18:59 -04:00
# To ensure consistent data, the environment deletes the fixtures before running the load.
2004-12-07 05:24:13 -05:00
#
# In addition to being available in the database, the fixtures are also loaded into a hash stored in an instance variable
# of the test case. It is named after the symbol... so, in our example, there would be a hash available called
2008-05-25 07:29:00 -04:00
# <tt>@web_sites</tt>. This is where the "fixture name" comes into play.
2004-12-07 05:24:13 -05:00
#
2008-05-25 07:29:00 -04:00
# On top of that, each record is automatically "found" (using <tt>Model.find(id)</tt>) and placed in the instance variable of its name.
# So for the YAML fixtures, we'd get <tt>@rubyonrails</tt> and <tt>@google</tt>, which could be interrogated using regular Active Record semantics:
2004-12-07 05:24:13 -05:00
#
# # test if the object created from the fixture data has the same attributes as the data itself
2004-11-23 20:04:44 -05:00
# def test_find
2004-12-07 05:24:13 -05:00
# assert_equal @web_sites["rubyonrails"]["name"], @rubyonrails.name
2004-11-23 20:04:44 -05:00
# end
#
2008-05-25 07:29:00 -04:00
# As seen above, the data hash created from the YAML fixtures would have <tt>@web_sites["rubyonrails"]["url"]</tt> return
# "http://www.rubyonrails.org" and <tt>@web_sites["google"]["name"]</tt> would return "Google". The same fixtures, but loaded
# from a CSV fixture file, would be accessible via <tt>@web_sites["web_site_1"]["name"] == "Ruby on Rails"</tt> and have the individual
# fixtures available as instance variables <tt>@web_site_1</tt> and <tt>@web_site_2</tt>.
2004-11-23 20:04:44 -05:00
#
2005-03-20 08:42:35 -05:00
# If you do not wish to use instantiated fixtures (usually for performance reasons) there are two options.
#
# - to completely disable instantiated fixtures:
# self.use_instantiated_fixtures = false
#
# - to keep the fixture instance (@web_sites) available, but do not automatically 'find' each instance:
2007-09-28 10:56:07 -04:00
# self.use_instantiated_fixtures = :no_instances
2005-03-20 08:42:35 -05:00
#
2005-06-03 07:49:34 -04:00
# 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
# model, and accepts the name of the fixture to instantiate:
#
# fixtures :web_sites
#
# def test_find
# assert_equal "Ruby on Rails", web_sites(:rubyonrails).name
# end
#
2004-12-07 05:24:13 -05:00
# = Dynamic fixtures with ERb
2004-11-23 20:04:44 -05:00
#
2004-12-07 05:24:13 -05:00
# Some times you don't care about the content of the fixtures as much as you care about the volume. In these cases, you can
# mix ERb in with your YAML or CSV fixtures to create a bunch of fixtures for load testing, like:
2004-11-23 20:04:44 -05:00
#
2008-03-26 08:27:52 -04:00
# <% for i in 1..1000 %>
# fix_<%= i %>:
# id: <%= i %>
# name: guy_<%= 1 %>
# <% end %>
2004-11-23 20:04:44 -05:00
#
2004-12-07 06:12:05 -05:00
# This will create 1000 very simple YAML fixtures.
2004-12-07 05:24:13 -05:00
#
2008-05-25 07:29:00 -04:00
# Using ERb, you can also inject dynamic values into your fixtures with inserts like <tt><%= Date.today.strftime("%Y-%m-%d") %></tt>.
2004-12-09 10:52:54 -05:00
# This is however a feature to be used with some caution. The point of fixtures are that they're stable units of predictable
2004-12-07 05:24:13 -05:00
# sample data. If you feel that you need to inject dynamic values, then perhaps you should reexamine whether your application
# is properly testable. Hence, dynamic values in fixtures are to be considered a code smell.
2005-03-06 08:51:55 -05:00
#
# = Transactional fixtures
#
2007-09-28 10:56:07 -04:00
# TestCases can use begin+rollback to isolate their changes to the database instead of having to delete+insert for every test case.
2005-03-06 08:51:55 -05:00
# They can also turn off auto-instantiation of fixture data since the feature is costly and often unused.
#
2008-01-05 08:34:15 -05:00
# class FooTest < ActiveSupport::TestCase
2005-03-06 08:51:55 -05:00
# self.use_transactional_fixtures = true
# self.use_instantiated_fixtures = false
2008-05-14 15:09:49 -04:00
#
2005-03-06 08:51:55 -05:00
# fixtures :foos
2008-05-14 15:09:49 -04:00
#
2005-03-06 08:51:55 -05:00
# def test_godzilla
2005-06-26 07:25:32 -04:00
# assert !Foo.find(:all).empty?
2005-03-06 08:51:55 -05:00
# Foo.destroy_all
2005-06-26 07:25:32 -04:00
# assert Foo.find(:all).empty?
2005-03-06 08:51:55 -05:00
# end
2008-05-14 15:09:49 -04:00
#
2005-03-06 08:51:55 -05:00
# def test_godzilla_aftermath
2005-06-26 07:25:32 -04:00
# assert !Foo.find(:all).empty?
2005-03-06 08:51:55 -05:00
# end
# end
2008-05-14 15:09:49 -04:00
#
2007-09-28 10:56:07 -04:00
# If you preload your test database with all fixture data (probably in the Rakefile task) and use transactional fixtures,
2005-03-06 08:51:55 -05:00
# 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.
#
2007-09-28 10:56:07 -04:00
# In order to use instantiated fixtures with preloaded data, set +self.pre_loaded_fixtures+ to true. This will provide
2005-03-20 08:42:35 -05:00
# access to fixture data for every table that has been loaded through fixtures (depending on the value of +use_instantiated_fixtures+)
#
2007-09-28 10:56:07 -04:00
# 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
2007-11-07 22:37:16 -05:00
# the results of your transaction until Active Record supports nested transactions or savepoints (in progress).
2007-09-28 10:56:07 -04:00
# 2. Your database does not support transactions. Every Active Record database supports transactions except MySQL MyISAM.
2005-03-06 08:51:55 -05:00
# Use InnoDB, MaxDB, or NDB instead.
2007-10-26 01:56:46 -04:00
#
# = Advanced YAML Fixtures
#
# YAML fixtures that don't specify an ID get some extra features:
#
# * Stable, autogenerated ID's
# * Label references for associations (belongs_to, has_one, has_many)
# * HABTM associations as inline lists
# * Autofilled timestamp columns
# * Fixture label interpolation
# * Support for YAML defaults
#
# == Stable, autogenerated ID's
#
# Here, have a monkey fixture:
#
# george:
# id: 1
# name: George the Monkey
#
# reginald:
# id: 2
# name: Reginald the Pirate
#
# Each of these fixtures has two unique identifiers: one for the database
# and one for the humans. Why don't we generate the primary key instead?
# Hashing each fixture's label yields a consistent ID:
#
# george: # generated id: 503576764
# name: George the Monkey
#
# reginald: # generated id: 324201669
# name: Reginald the Pirate
#
2008-05-25 07:29:00 -04:00
# Active Record looks at the fixture's model class, discovers the correct
2007-10-26 01:56:46 -04:00
# primary key, and generates it right before inserting the fixture
# into the database.
#
# The generated ID for a given label is constant, so we can discover
# any fixture's ID without loading anything, as long as we know the label.
#
# == Label references for associations (belongs_to, has_one, has_many)
#
# Specifying foreign keys in fixtures can be very fragile, not to
2008-05-25 07:29:00 -04:00
# mention difficult to read. Since Active Record can figure out the ID of
2007-11-07 22:37:16 -05:00
# any fixture from its label, you can specify FK's by label instead of ID.
2007-10-26 01:56:46 -04:00
#
# === belongs_to
#
# Let's break out some more monkeys and pirates.
#
# ### in pirates.yml
#
# reginald:
# id: 1
# name: Reginald the Pirate
# monkey_id: 1
#
# ### in monkeys.yml
#
# george:
# id: 1
# name: George the Monkey
# pirate_id: 1
#
# Add a few more monkeys and pirates and break this into multiple files,
# and it gets pretty hard to keep track of what's going on. Let's
# use labels instead of ID's:
#
# ### in pirates.yml
#
# reginald:
# name: Reginald the Pirate
# monkey: george
#
# ### in monkeys.yml
#
# george:
# name: George the Monkey
# pirate: reginald
#
2008-05-25 07:29:00 -04:00
# Pow! All is made clear. Active Record reflects on the fixture's model class,
2007-10-26 01:56:46 -04:00
# finds all the +belongs_to+ associations, and allows you to specify
# a target *label* for the *association* (monkey: george) rather than
2008-05-25 07:29:00 -04:00
# a target *id* for the *FK* (<tt>monkey_id: 1</tt>).
2007-10-26 01:56:46 -04:00
#
2007-11-20 16:53:22 -05:00
# ==== Polymorphic belongs_to
#
# Supporting polymorphic relationships is a little bit more complicated, since
2008-05-25 07:29:00 -04:00
# Active Record needs to know what type your association is pointing at. Something
2007-11-20 16:53:22 -05:00
# like this should look familiar:
#
# ### in fruit.rb
#
# belongs_to :eater, :polymorphic => true
#
# ### in fruits.yml
#
# apple:
# id: 1
# name: apple
# eater_id: 1
# eater_type: Monkey
#
# Can we do better? You bet!
#
# apple:
# eater: george (Monkey)
#
2008-05-25 07:29:00 -04:00
# Just provide the polymorphic target type and Active Record will take care of the rest.
2007-11-20 16:53:22 -05:00
#
2007-10-26 01:56:46 -04:00
# === has_and_belongs_to_many
#
# Time to give our monkey some fruit.
#
# ### in monkeys.yml
#
# george:
# id: 1
# name: George the Monkey
# pirate_id: 1
#
# ### in fruits.yml
#
# apple:
# id: 1
# name: apple
#
# orange:
# id: 2
# name: orange
#
# grape:
# id: 3
# name: grape
#
# ### in fruits_monkeys.yml
#
# apple_george:
# fruit_id: 1
# monkey_id: 1
#
# orange_george:
# fruit_id: 2
# monkey_id: 1
#
# grape_george:
# fruit_id: 3
# monkey_id: 1
#
# Let's make the HABTM fixture go away.
#
# ### in monkeys.yml
#
# george:
# name: George the Monkey
# pirate: reginald
# fruits: apple, orange, grape
#
# ### in fruits.yml
#
# apple:
# name: apple
#
# orange:
# name: orange
#
# grape:
# name: grape
#
# Zap! No more fruits_monkeys.yml file. We've specified the list of fruits
# on George's fixture, but we could've just as easily specified a list
2008-05-25 07:29:00 -04:00
# of monkeys on each fruit. As with +belongs_to+, Active Record reflects on
2007-10-26 01:56:46 -04:00
# the fixture's model class and discovers the +has_and_belongs_to_many+
# associations.
#
# == Autofilled timestamp columns
#
2008-05-25 07:29:00 -04:00
# If your table/model specifies any of Active Record's
# standard timestamp columns (+created_at+, +created_on+, +updated_at+, +updated_on+),
# they will automatically be set to <tt>Time.now</tt>.
2007-10-26 01:56:46 -04:00
#
# If you've set specific values, they'll be left alone.
#
# == Fixture label interpolation
#
# The label of the current fixture is always available as a column value:
#
# geeksomnia:
# name: Geeksomnia's Account
# subdomain: $LABEL
#
# Also, sometimes (like when porting older join table fixtures) you'll need
# to be able to get ahold of the identifier for a given label. ERB
# to the rescue:
#
# george_reginald:
# monkey_id: <%= Fixtures.identify(:reginald) %>
# pirate_id: <%= Fixtures.identify(:george) %>
#
# == Support for YAML defaults
#
# You probably already know how to use YAML to set and reuse defaults in
2008-05-09 05:38:02 -04:00
# your <tt>database.yml</tt> file. You can use the same technique in your fixtures:
2007-10-26 01:56:46 -04:00
#
# DEFAULTS: &DEFAULTS
# created_on: <%= 3.weeks.ago.to_s(:db) %>
#
# first:
# name: Smurf
# <<: *DEFAULTS
#
# second:
# name: Fraggle
# <<: *DEFAULTS
#
# Any fixture labeled "DEFAULTS" is safely ignored.
2007-12-22 03:28:57 -05:00
class Fixtures < ( RUBY_VERSION < '1.9' ? YAML :: Omap : Hash )
2004-12-09 10:52:54 -05:00
DEFAULT_FILTER_RE = / \ .ya?ml$ /
2007-09-28 10:05:03 -04:00
@@all_cached_fixtures = { }
2007-09-28 10:56:07 -04:00
def self . reset_cache ( connection = nil )
connection || = ActiveRecord :: Base . connection
2007-09-28 10:05:03 -04:00
@@all_cached_fixtures [ connection . object_id ] = { }
end
2007-09-28 10:56:07 -04:00
2007-09-28 10:05:03 -04:00
def self . cache_for_connection ( connection )
@@all_cached_fixtures [ connection . object_id ] || = { }
@@all_cached_fixtures [ connection . object_id ]
end
2007-09-28 10:56:07 -04:00
2007-09-28 10:05:03 -04:00
def self . fixture_is_cached? ( connection , table_name )
cache_for_connection ( connection ) [ table_name ]
end
def self . cached_fixtures ( connection , keys_to_fetch = nil )
if keys_to_fetch
fixtures = cache_for_connection ( connection ) . values_at ( * keys_to_fetch )
else
fixtures = cache_for_connection ( connection ) . values
end
fixtures . size > 1 ? fixtures : fixtures . first
end
2008-04-30 17:59:22 -04:00
def self . cache_fixtures ( connection , fixtures_map )
cache_for_connection ( connection ) . update ( fixtures_map )
2007-09-28 10:05:03 -04:00
end
2007-09-28 10:56:07 -04:00
def self . instantiate_fixtures ( object , table_name , fixtures , load_instances = true )
2005-04-18 04:11:15 -04:00
object . instance_variable_set " @ #{ table_name . to_s . gsub ( '.' , '_' ) } " , fixtures
2005-03-20 08:42:35 -05:00
if load_instances
2005-10-28 00:53:35 -04:00
ActiveRecord :: Base . silence do
2005-10-24 12:45:33 -04:00
fixtures . each do | name , fixture |
2006-03-14 10:15:06 -05:00
begin
object . instance_variable_set " @ #{ name } " , fixture . find
rescue FixtureClassNotFound
nil
2005-10-24 12:45:33 -04:00
end
2005-03-20 08:42:35 -05:00
end
2005-02-07 09:10:44 -05:00
end
2004-11-23 20:04:44 -05:00
end
end
2005-10-23 15:02:38 -04:00
2007-09-28 10:56:07 -04:00
def self . instantiate_all_loaded_fixtures ( object , load_instances = true )
2005-10-25 14:14:09 -04:00
all_loaded_fixtures . each do | table_name , fixtures |
Fixtures . instantiate_fixtures ( object , table_name , fixtures , load_instances )
2005-04-03 13:50:11 -04:00
end
2005-03-20 08:42:35 -05:00
end
2007-09-28 10:56:07 -04:00
2005-03-20 08:42:35 -05:00
cattr_accessor :all_loaded_fixtures
2005-10-25 14:14:09 -04:00
self . all_loaded_fixtures = { }
2004-12-09 10:52:54 -05:00
2006-02-27 19:40:33 -05:00
def self . create_fixtures ( fixtures_directory , table_names , class_names = { } )
2006-02-28 15:18:02 -05:00
table_names = [ table_names ] . flatten . map { | n | n . to_s }
2007-02-25 12:31:43 -05:00
connection = block_given? ? yield : ActiveRecord :: Base . connection
2007-09-28 10:56:07 -04:00
table_names_to_fetch = table_names . reject { | table_name | fixture_is_cached? ( connection , table_name ) }
2007-09-28 10:05:03 -04:00
unless table_names_to_fetch . empty?
ActiveRecord :: Base . silence do
2007-10-26 01:56:46 -04:00
connection . disable_referential_integrity do
fixtures_map = { }
2007-02-25 12:31:43 -05:00
2007-10-26 01:56:46 -04:00
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
2007-02-25 12:31:43 -05:00
2007-10-26 01:56:46 -04:00
all_loaded_fixtures . update ( fixtures_map )
2004-12-09 10:52:54 -05:00
2008-07-02 00:01:26 -04:00
connection . transaction ( connection . open_transactions . zero? ) do
2007-10-26 01:56:46 -04:00
fixtures . reverse . each { | fixture | fixture . delete_existing_fixtures }
fixtures . each { | fixture | fixture . insert_fixtures }
2005-10-23 15:02:38 -04:00
2007-10-26 01:56:46 -04:00
# Cap primary key sequences to max(pk).
if connection . respond_to? ( :reset_pk_sequence! )
table_names . each do | table_name |
connection . reset_pk_sequence! ( table_name )
end
2007-09-28 10:05:03 -04:00
end
2005-10-23 15:02:38 -04:00
end
2007-09-28 10:56:07 -04:00
2008-04-30 17:59:22 -04:00
cache_fixtures ( connection , fixtures_map )
2007-10-26 01:56:46 -04:00
end
2005-10-15 23:45:39 -04:00
end
2005-10-23 15:02:38 -04:00
end
2007-09-28 10:56:07 -04:00
cached_fixtures ( connection , table_names )
2005-10-23 15:02:38 -04:00
end
2007-10-26 01:56:46 -04:00
# Returns a consistent identifier for +label+. This will always
# be a positive integer, and will always be the same for a given
# label, assuming the same OS, platform, and version of Ruby.
def self . identify ( label )
label . to_s . hash . abs
end
2008-07-13 21:01:52 -04:00
attr_reader :table_name , :name
2005-04-18 03:52:58 -04:00
2006-02-27 15:29:28 -05:00
def initialize ( connection , table_name , class_name , fixture_path , file_filter = DEFAULT_FILTER_RE )
2004-11-23 20:04:44 -05:00
@connection , @table_name , @fixture_path , @file_filter = connection , table_name , fixture_path , file_filter
2008-07-13 21:01:52 -04:00
@name = table_name # preserve fixture base name
2007-09-28 10:56:07 -04:00
@class_name = class_name ||
2006-02-27 15:29:28 -05:00
( ActiveRecord :: Base . pluralize_table_names ? @table_name . singularize . camelize : @table_name . camelize )
2008-06-09 01:08:59 -04:00
@table_name = " #{ ActiveRecord :: Base . table_name_prefix } #{ @table_name } #{ ActiveRecord :: Base . table_name_suffix } "
2007-01-23 20:29:49 -05:00
@table_name = class_name . table_name if class_name . respond_to? ( :table_name )
@connection = class_name . connection if class_name . respond_to? ( :connection )
2004-11-23 20:04:44 -05:00
read_fixture_files
2004-12-01 07:25:04 -05:00
end
def delete_existing_fixtures
2007-10-16 01:06:33 -04:00
@connection . delete " DELETE FROM #{ @connection . quote_table_name ( table_name ) } " , 'Fixture Delete'
2004-12-01 07:25:04 -05:00
end
def insert_fixtures
2007-10-26 01:56:46 -04:00
now = ActiveRecord :: Base . default_timezone == :utc ? Time . now . utc : Time . now
now = now . to_s ( :db )
# allow a standard key to be used for doing defaults in YAML
2008-04-01 02:33:16 -04:00
if is_a? ( Hash )
delete ( 'DEFAULTS' )
else
delete ( assoc ( 'DEFAULTS' ) )
end
2007-10-26 01:56:46 -04:00
# track any join tables we need to insert later
habtm_fixtures = Hash . new do | h , habtm |
h [ habtm ] = HabtmFixtures . new ( @connection , habtm . options [ :join_table ] , nil , nil )
end
each do | label , fixture |
row = fixture . to_hash
2007-11-26 17:45:43 -05:00
if model_class && model_class < ActiveRecord :: Base
# fill in timestamp columns if they aren't specified and the model is set to record_timestamps
if model_class . record_timestamps
timestamp_column_names . each do | name |
row [ name ] = now unless row . key? ( name )
end
2007-10-26 01:56:46 -04:00
end
# interpolate the fixture label
row . each do | key , value |
row [ key ] = label if value == " $LABEL "
end
2007-11-25 17:08:38 -05:00
# generate a primary key if necessary
2007-11-26 17:45:43 -05:00
if has_primary_key_column? && ! row . include? ( primary_key_name )
row [ primary_key_name ] = Fixtures . identify ( label )
end
2007-10-26 01:56:46 -04:00
2007-11-26 17:46:11 -05:00
# If STI is used, find the correct subclass for association reflection
reflection_class =
if row . include? ( inheritance_column_name )
row [ inheritance_column_name ] . constantize rescue model_class
else
model_class
end
reflection_class . reflect_on_all_associations . each do | association |
2007-10-26 01:56:46 -04:00
case association . macro
when :belongs_to
2007-11-26 17:45:43 -05:00
# Do not replace association name with association foreign key if they are named the same
fk_name = ( association . options [ :foreign_key ] || " #{ association . name } _id " ) . to_s
if association . name . to_s != fk_name && value = row . delete ( association . name . to_s )
2007-11-20 16:53:22 -05:00
if association . options [ :polymorphic ]
if value . sub! ( / \ s* \ (([^ \ )]*) \ ) \ s*$ / , " " )
target_type = $1
target_type_name = ( association . options [ :foreign_type ] || " #{ association . name } _type " ) . to_s
# support polymorphic belongs_to as "label (Type)"
row [ target_type_name ] = target_type
end
end
2007-10-26 01:56:46 -04:00
row [ fk_name ] = Fixtures . identify ( value )
end
when :has_and_belongs_to_many
if ( targets = row . delete ( association . name . to_s ) )
targets = targets . is_a? ( Array ) ? targets : targets . split ( / \ s*, \ s* / )
join_fixtures = habtm_fixtures [ association ]
targets . each do | target |
join_fixtures [ " #{ label } _ #{ target } " ] = Fixture . new (
2007-11-26 17:45:43 -05:00
{ association . primary_key_name = > row [ primary_key_name ] ,
2007-10-26 01:56:46 -04:00
association . association_foreign_key = > Fixtures . identify ( target ) } , nil )
end
end
end
end
end
2007-09-28 10:56:07 -04:00
@connection . insert_fixture ( fixture , @table_name )
2004-12-01 07:25:04 -05:00
end
2007-10-26 01:56:46 -04:00
# insert any HABTM join tables we discovered
habtm_fixtures . values . each do | fixture |
fixture . delete_existing_fixtures
fixture . insert_fixtures
end
2004-11-23 20:04:44 -05:00
end
2007-02-25 12:31:43 -05:00
private
2007-10-26 01:56:46 -04:00
class HabtmFixtures < :: Fixtures #:nodoc:
def read_fixture_files ; end
end
def model_class
2008-01-04 21:20:26 -05:00
unless defined? ( @model_class )
@model_class =
if @class_name . nil? || @class_name . is_a? ( Class )
@class_name
else
@class_name . constantize rescue nil
end
end
@model_class
2007-10-26 01:56:46 -04:00
end
def primary_key_name
@primary_key_name || = model_class && model_class . primary_key
end
2007-11-26 17:45:43 -05:00
def has_primary_key_column?
@has_primary_key_column || = model_class && primary_key_name &&
model_class . columns . find { | c | c . name == primary_key_name }
2007-11-25 17:08:38 -05:00
end
2007-11-26 17:45:43 -05:00
2007-10-26 01:56:46 -04:00
def timestamp_column_names
@timestamp_column_names || = %w( created_at created_on updated_at updated_on ) . select do | name |
column_names . include? ( name )
end
end
2007-11-26 17:46:11 -05:00
def inheritance_column_name
@inheritance_column_name || = model_class && model_class . inheritance_column
end
2007-10-26 01:56:46 -04:00
def column_names
@column_names || = @connection . columns ( @table_name ) . collect ( & :name )
end
2004-11-23 20:04:44 -05:00
def read_fixture_files
2004-12-09 10:52:54 -05:00
if File . file? ( yaml_file_path )
2007-09-28 10:56:07 -04:00
read_yaml_fixture_files
2004-12-09 10:52:54 -05:00
elsif File . file? ( csv_file_path )
2007-09-28 10:56:07 -04:00
read_csv_fixture_files
2004-11-23 20:04:44 -05:00
end
end
2007-09-28 10:56:07 -04:00
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 |
2008-01-02 04:11:39 -05:00
raise Fixture :: FormatError , " Bad data for #{ @class_name } fixture named #{ fixture } " unless fixture . respond_to? ( :each )
fixture . each do | name , data |
2007-09-28 10:56:07 -04:00
unless data
raise Fixture :: FormatError , " Bad data for #{ @class_name } fixture named #{ name } (nil) "
end
2008-01-04 21:20:57 -05:00
self [ name ] = Fixture . new ( data , model_class )
2007-09-28 10:56:07 -04:00
end
end
end
2004-12-09 10:52:54 -05:00
end
2007-09-28 10:56:07 -04:00
def read_csv_fixture_files
2008-01-03 14:14:38 -05:00
reader = CSV . parse ( erb_render ( IO . read ( csv_file_path ) ) )
2007-09-28 10:56:07 -04:00
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 }
2008-05-14 15:09:49 -04:00
self [ " #{ @class_name . to_s . underscore } _ #{ i += 1 } " ] = Fixture . new ( data , model_class )
2007-09-28 10:56:07 -04:00
end
end
def yaml_file_path
" #{ @fixture_path } .yml "
2004-11-23 20:04:44 -05:00
end
2004-12-07 05:24:13 -05:00
def csv_file_path
@fixture_path + " .csv "
end
2004-12-09 10:52:54 -05:00
2004-11-23 20:04:44 -05:00
def yaml_fixtures_key ( path )
File . basename ( @fixture_path ) . split ( " . " ) . first
end
2007-09-28 10:56:07 -04:00
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 \n The exact error was: \n #{ error . class } : #{ error } "
end
2004-11-23 20:04:44 -05:00
def erb_render ( fixture_content )
ERB . new ( fixture_content ) . result
end
end
class Fixture #:nodoc:
include Enumerable
2007-09-28 10:56:07 -04:00
class FixtureError < StandardError #:nodoc:
2004-12-16 12:14:41 -05:00
end
2007-09-28 10:56:07 -04:00
class FormatError < FixtureError #:nodoc:
2004-12-16 12:14:41 -05:00
end
2004-11-23 20:04:44 -05:00
2008-01-04 21:20:57 -05:00
attr_reader :model_class
2007-05-26 02:26:50 -04:00
2008-01-04 21:20:57 -05:00
def initialize ( fixture , model_class )
2008-01-07 22:54:40 -05:00
@fixture = fixture
2008-01-04 21:20:57 -05:00
@model_class = model_class . is_a? ( Class ) ? model_class : model_class . constantize rescue nil
end
def class_name
@model_class . name if @model_class
2004-11-23 20:04:44 -05:00
end
def each
@fixture . each { | item | yield item }
end
def [] ( key )
@fixture [ key ]
end
def to_hash
@fixture
end
def key_list
2005-04-07 02:54:25 -04:00
columns = @fixture . keys . collect { | column_name | ActiveRecord :: Base . connection . quote_column_name ( column_name ) }
columns . join ( " , " )
2004-11-23 20:04:44 -05:00
end
def value_list
2006-04-27 18:39:45 -04:00
list = @fixture . inject ( [ ] ) do | fixtures , ( key , value ) |
2008-01-04 21:20:57 -05:00
col = model_class . columns_hash [ key ] if model_class . respond_to? ( :ancestors ) && model_class . ancestors . include? ( ActiveRecord :: Base )
2006-08-23 21:50:24 -04:00
fixtures << ActiveRecord :: Base . connection . quote ( value , col ) . gsub ( '[^\]\\n' , " \n " ) . gsub ( '[^\]\\r' , " \r " )
2006-04-27 18:39:45 -04:00
end
list * ', '
2004-11-23 20:04:44 -05:00
end
2004-12-09 10:52:54 -05:00
2004-11-23 20:04:44 -05:00
def find
2008-01-04 21:20:57 -05:00
if model_class
model_class . find ( self [ model_class . primary_key ] )
2006-03-06 18:03:35 -05:00
else
2008-01-04 21:20:57 -05:00
raise FixtureClassNotFound , " No class attached to find. "
2005-02-07 09:10:44 -05:00
end
2004-11-23 20:04:44 -05:00
end
end
2005-03-06 08:51:55 -05:00
module Test #:nodoc:
module Unit #:nodoc:
2004-12-16 12:14:41 -05:00
class TestCase #:nodoc:
2008-01-05 08:34:15 -05:00
setup :setup_fixtures
teardown :teardown_fixtures
2007-10-31 01:43:52 -04:00
superclass_delegating_accessor :fixture_path
superclass_delegating_accessor :fixture_table_names
superclass_delegating_accessor :fixture_class_names
superclass_delegating_accessor :use_transactional_fixtures
superclass_delegating_accessor :use_instantiated_fixtures # true, false, or :no_instances
superclass_delegating_accessor :pre_loaded_fixtures
2005-03-06 08:51:55 -05:00
self . fixture_table_names = [ ]
self . use_transactional_fixtures = false
self . use_instantiated_fixtures = true
2005-03-20 08:42:35 -05:00
self . pre_loaded_fixtures = false
2007-09-28 10:56:07 -04:00
2005-10-25 14:14:09 -04:00
@@already_loaded_fixtures = { }
2006-02-27 15:29:28 -05:00
self . fixture_class_names = { }
2007-09-28 10:56:07 -04:00
2008-01-05 08:34:15 -05:00
class << self
def set_fixture_class ( class_names = { } )
self . fixture_class_names = self . fixture_class_names . merge ( class_names )
2007-02-25 12:31:43 -05:00
end
2008-01-05 08:34:15 -05:00
def fixtures ( * table_names )
if table_names . first == :all
table_names = Dir [ " #{ fixture_path } /*.yml " ] + Dir [ " #{ fixture_path } /*.csv " ]
table_names . map! { | f | File . basename ( f ) . split ( '.' ) [ 0 .. - 2 ] . join ( '.' ) }
else
table_names = table_names . flatten . map { | n | n . to_s }
end
2004-12-14 13:01:28 -05:00
2008-01-05 08:34:15 -05:00
self . fixture_table_names |= table_names
require_fixture_classes ( table_names )
setup_fixture_accessors ( table_names )
end
2008-03-17 17:48:28 -04:00
def try_to_load_dependency ( file_name )
require_dependency file_name
rescue LoadError = > e
# Let's hope the developer has included it himself
2008-05-14 15:09:49 -04:00
2008-03-17 17:48:28 -04:00
# Let's warn in case this is a subdependency, otherwise
# subdependency error messages are totally cryptic
2008-03-17 18:07:54 -04:00
if ActiveRecord :: Base . logger
ActiveRecord :: Base . logger . warn ( " Unable to load #{ file_name } , underlying cause #{ e . message } \n \n #{ e . backtrace . join ( " \n " ) } " )
end
2008-03-17 17:48:28 -04:00
end
2008-05-14 15:09:49 -04:00
2008-01-05 08:34:15 -05:00
def 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
2008-03-17 17:48:28 -04:00
try_to_load_dependency ( file_name )
2004-12-16 12:14:41 -05:00
end
end
2004-12-09 10:52:54 -05:00
2008-01-05 08:34:15 -05:00
def setup_fixture_accessors ( table_names = nil )
table_names = [ table_names ] if table_names && ! table_names . respond_to? ( :each )
( table_names || fixture_table_names ) . each do | table_name |
table_name = table_name . to_s . tr ( '.' , '_' )
2007-06-04 23:47:02 -04:00
2008-01-05 08:34:15 -05:00
define_method ( table_name ) do | * fixtures |
force_reload = fixtures . pop if fixtures . last == true || fixtures . last == :reload
2007-02-25 14:27:56 -05:00
2008-01-05 08:34:15 -05:00
@fixture_cache [ table_name ] || = { }
2007-02-25 14:27:56 -05:00
2008-01-05 08:34:15 -05:00
instances = fixtures . map do | fixture |
@fixture_cache [ table_name ] . delete ( fixture ) if force_reload
2007-06-04 23:47:02 -04:00
2008-01-05 08:34:15 -05:00
if @loaded_fixtures [ table_name ] [ fixture . to_s ]
@fixture_cache [ table_name ] [ fixture ] || = @loaded_fixtures [ table_name ] [ fixture . to_s ] . find
else
raise StandardError , " No fixture with name ' #{ fixture } ' found for table ' #{ table_name } ' "
end
2007-06-04 23:47:02 -04:00
end
2008-01-05 08:34:15 -05:00
instances . size == 1 ? instances . first : instances
end
2005-06-03 07:49:34 -04:00
end
end
2008-01-05 08:34:15 -05:00
def uses_transaction ( * methods )
@uses_transaction = [ ] unless defined? ( @uses_transaction )
@uses_transaction . concat methods . map ( & :to_s )
end
2005-06-10 10:58:02 -04:00
2008-01-05 08:34:15 -05:00
def uses_transaction? ( method )
@uses_transaction = [ ] unless defined? ( @uses_transaction )
@uses_transaction . include? ( method . to_s )
end
2005-06-10 10:58:02 -04:00
end
def use_transactional_fixtures?
2005-10-25 14:14:09 -04:00
use_transactional_fixtures &&
! self . class . uses_transaction? ( method_name )
2005-06-10 10:58:02 -04:00
end
2008-01-05 08:34:15 -05:00
def setup_fixtures
return unless defined? ( ActiveRecord ) && ! ActiveRecord :: Base . configurations . blank?
2006-06-28 21:13:55 -04:00
2005-03-20 08:42:35 -05:00
if pre_loaded_fixtures && ! use_transactional_fixtures
2007-09-28 10:56:07 -04:00
raise RuntimeError , 'pre_loaded_fixtures requires use_transactional_fixtures'
2005-03-20 08:42:35 -05:00
end
2007-09-28 10:56:07 -04:00
@fixture_cache = { }
2005-10-25 14:14:09 -04:00
2005-03-06 08:51:55 -05:00
# Load fixtures once and begin transaction.
2005-06-10 10:58:02 -04:00
if use_transactional_fixtures?
2005-10-25 14:14:09 -04:00
if @@already_loaded_fixtures [ self . class ]
@loaded_fixtures = @@already_loaded_fixtures [ self . class ]
else
load_fixtures
@@already_loaded_fixtures [ self . class ] = @loaded_fixtures
end
2008-07-02 00:01:26 -04:00
ActiveRecord :: Base . connection . increment_open_transactions
2005-03-06 08:51:55 -05:00
ActiveRecord :: Base . connection . begin_db_transaction
# Load fixtures for every test.
else
2007-09-28 10:56:07 -04:00
Fixtures . reset_cache
2005-10-25 14:14:09 -04:00
@@already_loaded_fixtures [ self . class ] = nil
load_fixtures
2005-03-06 08:51:55 -05:00
end
# Instantiate fixtures for every test if requested.
instantiate_fixtures if use_instantiated_fixtures
end
2008-01-05 08:34:15 -05:00
def teardown_fixtures
return unless defined? ( ActiveRecord ) && ! ActiveRecord :: Base . configurations . blank?
2006-06-28 21:13:55 -04:00
2007-09-28 10:56:07 -04:00
unless use_transactional_fixtures?
Fixtures . reset_cache
2007-09-28 10:05:03 -04:00
end
2007-09-28 10:56:07 -04:00
2006-10-09 03:48:27 -04:00
# Rollback changes if a transaction is active.
2008-07-02 00:01:26 -04:00
if use_transactional_fixtures? && ActiveRecord :: Base . connection . open_transactions != 0
2005-03-06 08:51:55 -05:00
ActiveRecord :: Base . connection . rollback_db_transaction
2008-07-02 00:01:26 -04:00
ActiveRecord :: Base . connection . decrement_open_transactions
2005-03-06 08:51:55 -05:00
end
2006-03-16 00:21:40 -05:00
ActiveRecord :: Base . verify_active_connections!
2004-11-23 20:04:44 -05:00
end
2004-12-09 10:52:54 -05:00
2004-12-16 12:14:41 -05:00
private
2005-10-25 14:14:09 -04:00
def load_fixtures
@loaded_fixtures = { }
2006-02-27 15:29:28 -05:00
fixtures = Fixtures . create_fixtures ( fixture_path , fixture_table_names , fixture_class_names )
2005-10-25 14:14:09 -04:00
unless fixtures . nil?
if fixtures . instance_of? ( Fixtures )
2008-07-13 21:01:52 -04:00
@loaded_fixtures [ fixtures . name ] = fixtures
2005-10-24 12:45:33 -04:00
else
2008-07-13 21:01:52 -04:00
fixtures . each { | f | @loaded_fixtures [ f . name ] = f }
2005-10-25 14:14:09 -04:00
end
2005-03-06 08:51:55 -05:00
end
2004-12-16 12:14:41 -05:00
end
2005-04-18 03:52:58 -04:00
2005-03-20 08:42:35 -05:00
# for pre_loaded_fixtures, only require the classes once. huge speed improvement
@@required_fixture_classes = false
2005-04-18 03:52:58 -04:00
2005-03-06 08:51:55 -05:00
def instantiate_fixtures
2005-03-20 08:42:35 -05:00
if pre_loaded_fixtures
raise RuntimeError , 'Load fixtures before instantiating them.' if Fixtures . all_loaded_fixtures . empty?
unless @@required_fixture_classes
2005-10-25 14:14:09 -04:00
self . class . require_fixture_classes Fixtures . all_loaded_fixtures . keys
2005-03-20 08:42:35 -05:00
@@required_fixture_classes = true
end
Fixtures . instantiate_all_loaded_fixtures ( self , load_instances? )
2005-04-18 03:52:58 -04:00
else
2005-03-20 08:42:35 -05:00
raise RuntimeError , 'Load fixtures before instantiating them.' if @loaded_fixtures . nil?
@loaded_fixtures . each do | table_name , fixtures |
Fixtures . instantiate_fixtures ( self , table_name , fixtures , load_instances? )
end
2005-03-06 08:51:55 -05:00
end
2004-12-16 12:14:41 -05:00
end
2005-04-18 03:52:58 -04:00
2005-03-20 08:42:35 -05:00
def load_instances?
use_instantiated_fixtures != :no_instances
end
2004-11-23 20:04:44 -05:00
end
2004-12-16 12:14:41 -05:00
end
2005-01-01 14:22:16 -05:00
end