fixtures.rb 12.3 KB
Newer Older
D
Initial  
David Heinemeier Hansson 已提交
1 2
require 'erb'
require 'yaml'
3
require 'csv'
D
Initial  
David Heinemeier Hansson 已提交
4 5 6
require 'active_record/support/class_inheritable_attributes'
require 'active_record/support/inflector'

7
# Fixtures are a way of organizing data that you want to test against; in short, sample data. They come in 3 flavours:
D
Initial  
David Heinemeier Hansson 已提交
8
#
9 10 11
#   1.  YAML fixtures
#   2.  CSV fixtures
#   3.  Single-file fixtures
D
Initial  
David Heinemeier Hansson 已提交
12
#
13
# = YAML fixtures
D
Initial  
David Heinemeier Hansson 已提交
14
#
15 16
# 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+.
D
Initial  
David Heinemeier Hansson 已提交
17
#
18 19 20 21
# Unlike single-file fixtures, YAML fixtures are stored in a single file per model, which is place in the directory 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/). 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:
D
Initial  
David Heinemeier Hansson 已提交
22
#
23 24 25 26
#   rubyonrails:
#     id: 1
#     name: Ruby on Rails
#     url: http://www.rubyonrails.org
D
Initial  
David Heinemeier Hansson 已提交
27
#
28 29 30 31 32
#   google:
#     id: 2
#     name: Google
#     url: http://www.google.com
#
33
# This YAML fixture file includes two fixtures.  Each YAML fixture (ie. record) is given a name and is followed by an
34
# indented list of key/value pairs in the "key: value" format.  Records are separated by a blank line for your viewing
35
# pleasure.
36 37 38
#
# = CSV fixtures
#
D
David Heinemeier Hansson 已提交
39
# Fixtures can also be kept in the Comma Separated Value format. Akin to YAML fixtures, CSV fixtures are stored
40 41 42 43 44 45 46 47 48 49 50 51 52 53
# in a single file, but, instead end with the .csv file extension (Rails example: "<your-rails-app>/test/fixtures/web_sites.csv")
#
# The format of this tye of fixture file is much more compact than the others, but also a little harder to read by us
# 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
54
# fixture names are automatically generated by deriving the class name of the fixture file and adding an incrementing
55 56 57
# 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".
#
58
# Most databases and spreadsheets support exporting to CSV format, so this is a great format for you to choose if you
59
# have existing data somewhere already.
60 61
#
# = Single-file fixtures
62 63
#
# This type of fixtures was the original format for Active Record that has since been deprecated in favor of the YAML and CSV formats.
64
# Fixtures for this format are created by placing text files in a sub-directory (with the name of the model) to the directory
65 66 67 68 69 70 71 72 73 74 75 76 77 78 79 80 81 82 83 84 85 86 87 88 89 90 91 92 93
# 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
# fixtures, but first lets take a look at a sample unit test found:
#
#   require 'web_site'
#
#   class WebSiteTest < Test::Unit::TestCase
#     def test_web_site_count
#       assert_equal 2, WebSite.count
94
#     end
95
#   end
96 97
#
# As it stands, unless we pre-load the web_site table in our database with two records, this test will fail.  Here's the
98 99 100 101 102 103
# easiest way to add fixtures to the database:
#
#   ...
#   class WebSiteTest < Test::Unit::TestCase
#     fixtures :web_sites # add more by separating the symbols with commas
#   ...
104
#
105
# By adding a "fixtures" method to the test case and passing it a list of symbols (only one is shown here tho), we trigger
106
# the testing environment to automatically load the appropriate fixtures into the database before each test, and
107 108 109 110 111 112 113 114 115 116
# automatically delete them after each test.
#
# 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
D
Initial  
David Heinemeier Hansson 已提交
117
#   def test_find
118
#     assert_equal @web_sites["rubyonrails"]["name"], @rubyonrails.name
D
Initial  
David Heinemeier Hansson 已提交
119 120
#   end
#
121 122 123 124
# 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
# from a CSV fixture file would be accessible via @web_sites["web_site_1"]["name"] == "Ruby on Rails" and have the individual
# fixtures available as instance variables @web_site_1 and @web_site_2.
D
Initial  
David Heinemeier Hansson 已提交
125
#
126
# = Dynamic fixtures with ERb
D
Initial  
David Heinemeier Hansson 已提交
127
#
128 129
# 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:
D
Initial  
David Heinemeier Hansson 已提交
130
#
131 132 133 134 135
# <% for i in 1..1000 %>
# fix_<%= i %>:
#   id: <%= i %>
#   name: guy_<%= 1 %>
# <% end %>
D
Initial  
David Heinemeier Hansson 已提交
136
#
D
David Heinemeier Hansson 已提交
137
# This will create 1000 very simple YAML fixtures.
138 139
#
# Using ERb, you can also inject dynamic values into your fixtures with inserts like <%= Date.today.strftime("%Y-%m-%d") %>.
140
# This is however a feature to be used with some caution. The point of fixtures are that they're stable units of predictable
141 142
# 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.
D
Initial  
David Heinemeier Hansson 已提交
143
class Fixtures < Hash
144 145
  DEFAULT_FILTER_RE = /\.ya?ml$/

D
Initial  
David Heinemeier Hansson 已提交
146 147 148 149 150 151
  def self.instantiate_fixtures(object, fixtures_directory, *table_names)
    [ create_fixtures(fixtures_directory, *table_names) ].flatten.each_with_index do |fixtures, idx|
      object.instance_variable_set "@#{table_names[idx]}", fixtures
      fixtures.each { |name, fixture| object.instance_variable_set "@#{name}", fixture.find }
    end
  end
152

D
Initial  
David Heinemeier Hansson 已提交
153 154 155 156 157 158
  def self.create_fixtures(fixtures_directory, *table_names)
    connection = block_given? ? yield : ActiveRecord::Base.connection
    old_logger_level = ActiveRecord::Base.logger.level

    begin
      ActiveRecord::Base.logger.level = Logger::ERROR
159 160 161 162 163

      fixtures = table_names.flatten.map do |table_name|
        Fixtures.new(connection, table_name.to_s, File.join(fixtures_directory, table_name.to_s))
      end

164
      connection.transaction do
165 166
        fixtures.reverse.each { |fixture| fixture.delete_existing_fixtures }
        fixtures.each { |fixture| fixture.insert_fixtures }
D
Initial  
David Heinemeier Hansson 已提交
167
      end
168

D
Initial  
David Heinemeier Hansson 已提交
169 170 171 172 173 174
      return fixtures.size > 1 ? fixtures : fixtures.first
    ensure
      ActiveRecord::Base.logger.level = old_logger_level
    end
  end

175
  def initialize(connection, table_name, fixture_path, file_filter = DEFAULT_FILTER_RE)
D
Initial  
David Heinemeier Hansson 已提交
176 177 178 179
    @connection, @table_name, @fixture_path, @file_filter = connection, table_name, fixture_path, file_filter
    @class_name = Inflector.classify(@table_name)

    read_fixture_files
180 181 182
  end

  def delete_existing_fixtures
183
    @connection.delete "DELETE FROM #{@table_name}", 'Fixture Delete'
184 185 186 187
  end

  def insert_fixtures
    values.each do |fixture|
188
      @connection.execute "INSERT INTO #{@table_name} (#{fixture.key_list}) VALUES (#{fixture.value_list})", 'Fixture Insert'
189
    end
D
Initial  
David Heinemeier Hansson 已提交
190 191 192 193
  end

  private
    def read_fixture_files
194
      if File.file?(yaml_file_path)
195
        # YAML fixtures
D
Initial  
David Heinemeier Hansson 已提交
196 197 198
        YAML::load(erb_render(IO.read(yaml_file_path))).each do |name, data|
          self[name] = Fixture.new(data, @class_name)
        end
199
      elsif File.file?(csv_file_path)
200 201 202 203 204 205 206 207 208
        # 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
209 210
      elsif File.file?(deprecated_yaml_file_path)
        raise Fixture::FormatError, ".yml extension required: rename #{deprecated_yaml_file_path} to #{yaml_file_path}"
D
Initial  
David Heinemeier Hansson 已提交
211
      else
212
        # Standard fixtures
213 214 215 216 217
        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
D
Initial  
David Heinemeier Hansson 已提交
218 219 220 221 222
        end
      end
    end

    def yaml_file_path
223 224 225 226 227
      "#{@fixture_path}.yml"
    end

    def deprecated_yaml_file_path
      "#{@fixture_path}.yaml"
D
Initial  
David Heinemeier Hansson 已提交
228
    end
229 230 231 232

    def csv_file_path
      @fixture_path + ".csv"
    end
233

D
Initial  
David Heinemeier Hansson 已提交
234 235 236 237 238 239 240 241 242 243 244 245 246 247 248 249 250 251 252 253 254 255 256 257 258 259 260 261 262 263 264 265 266 267 268 269 270 271
    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
  class FixtureError < StandardError; end
  class FormatError < FixtureError; end

  def initialize(fixture, class_name)
    @fixture = fixture.is_a?(Hash) ? fixture : read_fixture_file(fixture)
    @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
    @fixture.keys.join(", ")
  end

  def value_list
    @fixture.values.map { |v| ActiveRecord::Base.connection.quote(v).gsub('\\n', "\n").gsub('\\r', "\r") }.join(", ")
  end
272

D
Initial  
David Heinemeier Hansson 已提交
273
  def find
274
    Object.const_get(@class_name).find(self[Object.const_get(@class_name).primary_key])
D
Initial  
David Heinemeier Hansson 已提交
275
  end
276

D
Initial  
David Heinemeier Hansson 已提交
277 278 279 280
  private
    def read_fixture_file(fixture_file_path)
      IO.readlines(fixture_file_path).inject({}) do |fixture, line|
        # Mercifully skip empty lines.
281
        next if line =~ /^\s*$/
D
Initial  
David Heinemeier Hansson 已提交
282 283 284

        # Use the same regular expression for attributes as Active Record.
        unless md = /^\s*([a-zA-Z][-_\w]*)\s*=>\s*(.+)\s*$/.match(line)
285
          raise FormatError, "#{fixture_file_path}: fixture format error at '#{line}'.  Expecting 'key => value'."
D
Initial  
David Heinemeier Hansson 已提交
286 287 288 289
        end
        key, value = md.captures

        # Disallow duplicate keys to catch typos.
290
        raise FormatError, "#{fixture_file_path}: duplicate '#{key}' in fixture." if fixture[key]
D
Initial  
David Heinemeier Hansson 已提交
291 292 293 294 295 296 297 298
        fixture[key] = value.strip
        fixture
      end
    end
end

class Test::Unit::TestCase #:nodoc:
  include ClassInheritableAttributes
299

D
Initial  
David Heinemeier Hansson 已提交
300 301
  cattr_accessor :fixture_path
  cattr_accessor :fixture_table_names
302

D
Initial  
David Heinemeier Hansson 已提交
303 304 305 306 307 308 309
  def self.fixtures(*table_names)
    write_inheritable_attribute("fixture_table_names", table_names)
  end

  def setup
    instantiate_fixtures(*fixture_table_names) if fixture_table_names
  end
310

D
Initial  
David Heinemeier Hansson 已提交
311 312 313 314 315 316 317 318 319 320 321 322 323 324
  def self.method_added(method_symbol)
    if method_symbol == :setup && !method_defined?(:setup_without_fixtures)
      alias_method :setup_without_fixtures, :setup
      define_method(:setup) do
        instantiate_fixtures(*fixture_table_names) if fixture_table_names
        setup_without_fixtures
      end
    end
  end

  private
    def instantiate_fixtures(*table_names)
      Fixtures.instantiate_fixtures(self, fixture_path, *table_names)
    end
325

D
Initial  
David Heinemeier Hansson 已提交
326 327 328
    def fixture_table_names
      self.class.read_inheritable_attribute("fixture_table_names")
    end
329
end