2004-11-23 20:04:44 -05:00
require 'erb'
require 'yaml'
2004-12-07 05:24:13 -05:00
require 'csv'
2004-11-23 20:04:44 -05:00
2005-10-16 12:00:16 -04:00
module YAML #:nodoc:
class Omap #:nodoc:
def keys ; map { | k , v | k } end
def values ; map { | k , v | v } end
end
2005-10-13 15:06:15 -04:00
end
2006-03-06 18:03:35 -05:00
class FixtureClassNotFound < ActiveRecord :: ActiveRecordError #:nodoc:
end
2004-12-07 05:24:13 -05:00
# Fixtures are a way of organizing data that you want to test against; in short, sample data. They come in 3 flavours:
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
# in a non-verbose, humanly-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
2004-12-07 05:24:13 -05:00
# by <tt>Test::Unit::TestCase.fixture_path=(path)</tt> (this is automatically configured for Rails, so you can just
# put your files in <your-rails-app>/test/fixtures/). The fixture file ends with the .yml file extension (Rails example:
# "<your-rails-app>/test/fixtures/web_sites.yml"). 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
2005-10-26 09:05:48 -04:00
# in a single file, but instead end with the .csv file extension (Rails example: "<your-rails-app>/test/fixtures/web_sites.csv")
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
#
# This type of fixtures 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
2004-12-07 05:24:13 -05:00
# appointed by <tt>Test::Unit::TestCase.fixture_path=(path)</tt> (this is automatically configured for Rails, so you can just
# put your files in <your-rails-app>/test/fixtures/<your-model-name>/ -- like <your-rails-app>/test/fixtures/web_sites/ for the WebSite
# model).
#
# Each text file placed in this directory represents a "record". Usually these types of fixtures are named without
# extensions, but if you are on a Windows machine, you might consider adding .txt as the extension. Here's what the
# 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
2005-10-26 09:05:48 -04:00
# fixtures, but first let's take a look at a sample unit test found:
2004-12-07 05:24:13 -05:00
#
# require 'web_site'
#
# class WebSiteTest < Test::Unit::TestCase
# 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:
#
# ...
# class WebSiteTest < Test::Unit::TestCase
# fixtures :web_sites # add more by separating the symbols with commas
# ...
2004-12-09 10:52:54 -05:00
#
2004-12-07 05:24:13 -05:00
# By adding a "fixtures" method to the test case and passing it a list of symbols (only one is shown here tho), we trigger
2005-10-25 14:14:09 -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
# @web_sites. This is where the "fixture name" comes into play.
#
# On top of that, each record is automatically "found" (using Model.find(id)) and placed in the instance variable of its name.
# So for the YAML fixtures, we'd get @rubyonrails and @google, which could be interrogated using regular Active Record semantics:
#
# # 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
#
2004-12-07 05:24:13 -05:00
# As seen above, the data hash created from the YAML fixtures would have @web_sites["rubyonrails"]["url"] return
# "http://www.rubyonrails.org" and @web_sites["google"]["name"] would return "Google". The same fixtures, but loaded
2005-10-26 09:05:48 -04:00
# from a CSV fixture file, would be accessible via @web_sites["web_site_1"]["name"] == "Ruby on Rails" and have the individual
2004-12-07 05:24:13 -05:00
# fixtures available as instance variables @web_site_1 and @web_site_2.
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:
2005-10-25 14:14:09 -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
#
2004-12-07 05:24:13 -05: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
#
# Using ERb, you can also inject dynamic values into your fixtures with inserts like <%= Date.today.strftime("%Y-%m-%d") %>.
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
#
2005-10-25 14:14:09 -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.
#
# class FooTest < Test::Unit::TestCase
# self.use_transactional_fixtures = true
# self.use_instantiated_fixtures = false
2005-10-25 14:14:09 -04:00
#
2005-03-06 08:51:55 -05:00
# fixtures :foos
2005-10-25 14:14:09 -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
2005-10-25 14:14:09 -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
2005-10-25 14:14:09 -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.
#
2005-10-25 14:14:09 -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+)
#
2005-10-25 14:14:09 -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
# 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.
2005-03-06 08:51:55 -05:00
# Use InnoDB, MaxDB, or NDB instead.
2005-10-13 15:06:15 -04:00
class Fixtures < YAML :: Omap
2004-12-09 10:52:54 -05:00
DEFAULT_FILTER_RE = / \ .ya?ml$ /
2005-03-20 08:42:35 -05: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
2005-03-20 08:42:35 -05: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
2005-10-25 14:14:09 -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
2005-10-28 00:53:35 -04:00
ActiveRecord :: Base . silence do
2005-10-25 14:14:09 -04:00
fixtures_map = { }
2007-02-25 12:31:43 -05:00
2005-10-23 15:02:38 -04:00
fixtures = table_names . map do | table_name |
2006-02-27 15:29:28 -05:00
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 ) )
2005-10-25 14:14:09 -04:00
end
2007-02-25 12:31:43 -05:00
2005-10-25 14:14:09 -04:00
all_loaded_fixtures . merge! fixtures_map
2004-12-09 10:52:54 -05:00
2007-06-05 00:37:05 -04:00
connection . transaction ( Thread . current [ 'open_transactions' ] . to_i == 0 ) do
2005-10-25 14:14:09 -04:00
fixtures . reverse . each { | fixture | fixture . delete_existing_fixtures }
fixtures . each { | fixture | fixture . insert_fixtures }
2005-10-23 15:02:38 -04:00
2005-10-25 14:14:09 -04:00
# Cap primary key sequences to max(pk).
2005-10-23 15:02:38 -04:00
if connection . respond_to? ( :reset_pk_sequence! )
2005-10-25 14:14:09 -04:00
table_names . each do | table_name |
2005-10-23 15:02:38 -04:00
connection . reset_pk_sequence! ( table_name )
end
2005-10-15 23:45:39 -04:00
end
end
2004-12-22 18:09:30 -05:00
2004-11-23 20:04:44 -05:00
return fixtures . size > 1 ? fixtures : fixtures . first
2005-10-23 15:02:38 -04:00
end
end
2004-12-22 18:09:30 -05:00
2005-04-18 03:52:58 -04:00
attr_reader :table_name
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
2006-02-27 15:29:28 -05:00
@class_name = class_name ||
( ActiveRecord :: Base . pluralize_table_names ? @table_name . singularize . camelize : @table_name . camelize )
2005-09-02 06:51:23 -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
2004-12-09 10:52:54 -05:00
@connection . delete " DELETE FROM #{ @table_name } " , 'Fixture Delete'
2004-12-01 07:25:04 -05:00
end
def insert_fixtures
values . each do | fixture |
2007-05-26 02:26:50 -04:00
@connection . insert_fixture fixture , @table_name
2004-12-01 07:25:04 -05:00
end
2004-11-23 20:04:44 -05:00
end
2006-02-27 00:14:57 -05:00
2007-02-25 12:31:43 -05:00
private
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 )
2004-12-07 05:24:13 -05:00
# YAML fixtures
2006-09-13 19:31:54 -04:00
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 )
2007-03-08 21:06:50 -05:00
2004-12-14 19:46:26 -05:00
begin
2006-09-13 19:31:54 -04:00
yaml = YAML :: load ( erb_render ( yaml_string ) )
2004-12-14 19:46:26 -05:00
rescue Exception = > boom
2006-07-10 15:41:59 -04:00
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 #{ boom . class } : #{ boom } "
2007-03-08 21:06:50 -05:00
end
2006-09-13 19:31:54 -04:00
if yaml
2007-03-08 21:06:50 -05:00
# 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 )
2006-09-13 19:31:54 -04:00
end
end
2004-11-23 20:04:44 -05:00
end
2004-12-09 10:52:54 -05:00
elsif File . file? ( csv_file_path )
2004-12-07 05:24:13 -05:00
# 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
2004-12-09 10:52:54 -05:00
elsif File . file? ( deprecated_yaml_file_path )
raise Fixture :: FormatError , " .yml extension required: rename #{ deprecated_yaml_file_path } to #{ yaml_file_path } "
2004-11-23 20:04:44 -05:00
else
2004-12-07 05:24:13 -05:00
# Standard fixtures
2004-12-09 10:52:54 -05:00
Dir . entries ( @fixture_path ) . each do | file |
path = File . join ( @fixture_path , file )
if File . file? ( path ) and file !~ @file_filter
self [ file ] = Fixture . new ( path , @class_name )
end
2004-11-23 20:04:44 -05:00
end
end
end
def yaml_file_path
2004-12-09 10:52:54 -05:00
" #{ @fixture_path } .yml "
end
def deprecated_yaml_file_path
" #{ @fixture_path } .yaml "
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
def erb_render ( fixture_content )
ERB . new ( fixture_content ) . result
end
end
class Fixture #:nodoc:
include Enumerable
2004-12-16 12:14:41 -05:00
class FixtureError < StandardError #:nodoc:
end
class FormatError < FixtureError #:nodoc:
end
2004-11-23 20:04:44 -05:00
2007-05-26 02:26:50 -04:00
attr_reader :class_name
2004-11-23 20:04:44 -05:00
def initialize ( fixture , class_name )
2005-10-14 22:01:38 -04:00
case fixture
when Hash , YAML :: Omap
@fixture = fixture
when String
@fixture = read_fixture_file ( fixture )
else
2006-09-13 19:31:54 -04:00
raise ArgumentError , " Bad fixture argument #{ fixture . inspect } during creation of #{ class_name } fixture "
2005-10-14 22:01:38 -04:00
end
2004-11-23 20:04:44 -05:00
@class_name = class_name
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
klass = @class_name . constantize rescue nil
list = @fixture . inject ( [ ] ) do | fixtures , ( key , value ) |
2007-05-21 14:54:51 -04:00
col = klass . columns_hash [ key ] if klass . respond_to? ( :ancestors ) && klass . 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
2006-03-06 18:03:35 -05:00
klass = @class_name . is_a? ( Class ) ? @class_name : Object . const_get ( @class_name ) rescue nil
if klass
2005-02-07 09:10:44 -05:00
klass . find ( self [ klass . primary_key ] )
2006-03-06 18:03:35 -05:00
else
raise FixtureClassNotFound , " The class #{ @class_name . inspect } was not found. "
2005-02-07 09:10:44 -05:00
end
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
private
def read_fixture_file ( fixture_file_path )
IO . readlines ( fixture_file_path ) . inject ( { } ) do | fixture , line |
# Mercifully skip empty lines.
2004-12-09 10:52:54 -05:00
next if line =~ / ^ \ s*$ /
2004-11-23 20:04:44 -05:00
# Use the same regular expression for attributes as Active Record.
unless md = / ^ \ s*([a-zA-Z][-_ \ w]*) \ s*=> \ s*(.+) \ s*$ / . match ( line )
2004-12-09 10:52:54 -05:00
raise FormatError , " #{ fixture_file_path } : fixture format error at ' #{ line } '. Expecting 'key => value'. "
2004-11-23 20:04:44 -05:00
end
key , value = md . captures
# Disallow duplicate keys to catch typos.
2004-12-09 10:52:54 -05:00
raise FormatError , " #{ fixture_file_path } : duplicate ' #{ key } ' in fixture. " if fixture [ key ]
2004-11-23 20:04:44 -05:00
fixture [ key ] = value . strip
fixture
end
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:
2006-11-21 17:28:24 -05:00
class_inheritable_accessor :fixture_path
2005-03-06 08:51:55 -05:00
class_inheritable_accessor :fixture_table_names
2006-02-27 15:29:28 -05:00
class_inheritable_accessor :fixture_class_names
2005-03-06 08:51:55 -05:00
class_inheritable_accessor :use_transactional_fixtures
2005-03-20 08:42:35 -05:00
class_inheritable_accessor :use_instantiated_fixtures # true, false, or :no_instances
class_inheritable_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
2006-02-27 15:29:28 -05:00
self . fixture_class_names = { }
2005-10-25 14:14:09 -04:00
@@already_loaded_fixtures = { }
2006-02-27 15:29:28 -05:00
self . fixture_class_names = { }
def self . set_fixture_class ( class_names = { } )
self . fixture_class_names = self . fixture_class_names . merge ( class_names )
end
2005-10-25 14:14:09 -04:00
def self . fixtures ( * table_names )
2007-02-25 12:31:43 -05:00
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
2005-10-25 14:14:09 -04:00
self . fixture_table_names |= table_names
require_fixture_classes ( table_names )
setup_fixture_accessors ( table_names )
end
2004-12-14 13:01:28 -05:00
2005-10-25 14:14:09 -04:00
def self . require_fixture_classes ( table_names = nil )
( table_names || fixture_table_names ) . each do | table_name |
2005-11-03 10:43:48 -05:00
file_name = table_name . to_s
file_name = file_name . singularize if ActiveRecord :: Base . pluralize_table_names
2005-10-25 14:14:09 -04:00
begin
2006-08-12 15:29:46 -04:00
require_dependency file_name
2005-10-25 14:14:09 -04:00
rescue LoadError
# Let's hope the developer has included it himself
2004-12-16 12:14:41 -05:00
end
end
2005-10-25 14:14:09 -04:00
end
2004-12-09 10:52:54 -05:00
2007-06-04 23:47:02 -04:00
def self . setup_fixture_accessors ( table_names = nil )
2005-10-25 14:14:09 -04:00
( table_names || fixture_table_names ) . each do | table_name |
2007-06-04 23:47:02 -04:00
table_name = table_name . to_s . tr ( '.' , '_' )
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
2005-10-25 14:14:09 -04:00
@fixture_cache [ table_name ] || = Hash . new
2007-02-25 14:27:56 -05:00
2007-06-04 23:47:02 -04:00
instances = fixtures . map do | fixture |
@fixture_cache [ table_name ] . delete ( fixture ) if force_reload
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
2006-01-24 21:40:25 -05:00
end
2007-06-04 23:47:02 -04:00
instances . size == 1 ? instances . first : instances
2005-06-03 07:49:34 -04:00
end
end
2005-10-25 14:14:09 -04:00
end
2005-06-03 07:49:34 -04:00
2005-10-25 14:14:09 -04:00
def self . uses_transaction ( * methods )
2006-05-31 22:02:42 -04:00
@uses_transaction = [ ] unless defined? ( @uses_transaction )
@uses_transaction . concat methods . map ( & :to_s )
2005-10-25 14:14:09 -04:00
end
2005-06-10 10:58:02 -04:00
2005-10-25 14:14:09 -04:00
def self . uses_transaction? ( method )
2006-05-31 22:02:42 -04:00
@uses_transaction = [ ] unless defined? ( @uses_transaction )
@uses_transaction . include? ( method . to_s )
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
2005-03-06 08:51:55 -05:00
def setup_with_fixtures
2006-07-10 15:31:44 -04:00
return unless defined? ( ActiveRecord :: Base ) && ! 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
2005-10-25 14:14:09 -04:00
raise RuntimeError , 'pre_loaded_fixtures requires use_transactional_fixtures'
2005-03-20 08:42:35 -05:00
end
2005-10-25 14:14:09 -04:00
@fixture_cache = Hash . new
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
2006-06-19 18:48:51 -04:00
ActiveRecord :: Base . send :increment_open_transactions
2005-03-06 08:51:55 -05:00
ActiveRecord :: Base . connection . begin_db_transaction
# Load fixtures for every test.
else
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
alias_method :setup , :setup_with_fixtures
def teardown_with_fixtures
2006-07-10 15:31:44 -04:00
return unless defined? ( ActiveRecord :: Base ) && ! ActiveRecord :: Base . configurations . blank?
2006-06-28 21:13:55 -04:00
2006-10-09 03:48:27 -04:00
# Rollback changes if a transaction is active.
2006-10-10 20:29:59 -04:00
if use_transactional_fixtures? && Thread . current [ 'open_transactions' ] != 0
2005-03-06 08:51:55 -05:00
ActiveRecord :: Base . connection . rollback_db_transaction
2006-10-09 03:48:27 -04:00
Thread . current [ 'open_transactions' ] = 0
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
2005-03-06 08:51:55 -05:00
alias_method :teardown , :teardown_with_fixtures
def self . method_added ( method )
case method . to_s
when 'setup'
unless method_defined? ( :setup_without_fixtures )
alias_method :setup_without_fixtures , :setup
define_method ( :setup ) do
setup_with_fixtures
setup_without_fixtures
end
end
when 'teardown'
unless method_defined? ( :teardown_without_fixtures )
alias_method :teardown_without_fixtures , :teardown
define_method ( :teardown ) do
teardown_without_fixtures
teardown_with_fixtures
end
2004-12-16 12:14:41 -05:00
end
end
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 )
@loaded_fixtures [ fixtures . table_name ] = fixtures
2005-10-24 12:45:33 -04:00
else
2005-10-25 14:14:09 -04:00
fixtures . each { | f | @loaded_fixtures [ f . table_name ] = f }
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
2005-03-06 08:51:55 -05:00
2004-12-16 12:14:41 -05:00
end
2005-01-01 14:22:16 -05:00
end