fixtures.rb 13.8 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
  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
149 150 151 152 153
      fixtures.each do |name, fixture|
        if model = fixture.find
          object.instance_variable_set "@#{name}", model
        end
      end
D
Initial  
David Heinemeier Hansson 已提交
154 155
    end
  end
156

D
Initial  
David Heinemeier Hansson 已提交
157 158 159 160 161 162
  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
163 164

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

168
      connection.transaction do
169 170
        fixtures.reverse.each { |fixture| fixture.delete_existing_fixtures }
        fixtures.each { |fixture| fixture.insert_fixtures }
D
Initial  
David Heinemeier Hansson 已提交
171
      end
172

173
      reset_sequences(connection, table_names) if connection.is_a?(ActiveRecord::ConnectionAdapters::PostgreSQLAdapter)
174

D
Initial  
David Heinemeier Hansson 已提交
175 176 177 178 179 180
      return fixtures.size > 1 ? fixtures : fixtures.first
    ensure
      ActiveRecord::Base.logger.level = old_logger_level
    end
  end

181 182 183 184 185 186 187 188 189 190 191 192 193 194 195 196
  # Work around for PostgreSQL to have new fixtures created from id 1 and running.
  def self.reset_sequences(connection, table_names)
    table_names.flatten.each do |table|
      table_class = Inflector.classify(table.to_s)
      if Object.const_defined?(table_class)
        pk = eval("#{table_class}::primary_key")
        if pk == 'id'
          connection.execute(
            "SELECT setval('public.#{table.to_s}_id_seq', (SELECT MAX(id) FROM #{table.to_s}), true)", 
            'Setting Sequence'
          )
        end
      end
    end
  end

197
  def initialize(connection, table_name, fixture_path, file_filter = DEFAULT_FILTER_RE)
D
Initial  
David Heinemeier Hansson 已提交
198 199 200 201
    @connection, @table_name, @fixture_path, @file_filter = connection, table_name, fixture_path, file_filter
    @class_name = Inflector.classify(@table_name)

    read_fixture_files
202 203 204
  end

  def delete_existing_fixtures
205
    @connection.delete "DELETE FROM #{@table_name}", 'Fixture Delete'
206 207 208 209
  end

  def insert_fixtures
    values.each do |fixture|
210
      @connection.execute "INSERT INTO #{@table_name} (#{fixture.key_list}) VALUES (#{fixture.value_list})", 'Fixture Insert'
211
    end
D
Initial  
David Heinemeier Hansson 已提交
212 213 214 215
  end

  private
    def read_fixture_files
216
      if File.file?(yaml_file_path)
217
        # YAML fixtures
218 219 220 221 222
        begin
          yaml = YAML::load(erb_render(IO.read(yaml_file_path)))
          yaml.each { |name, data| self[name] = Fixture.new(data, @class_name) } if yaml
        rescue Exception=>boom
          raise Fixture::FormatError, "a YAML error occured parsing #{yaml_file_path}"
D
Initial  
David Heinemeier Hansson 已提交
223
        end
224
      elsif File.file?(csv_file_path)
225 226 227 228 229 230 231 232 233
        # 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
234 235
      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 已提交
236
      else
237
        # Standard fixtures
238 239 240 241 242
        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 已提交
243 244 245 246 247
        end
      end
    end

    def yaml_file_path
248 249 250 251 252
      "#{@fixture_path}.yml"
    end

    def deprecated_yaml_file_path
      "#{@fixture_path}.yaml"
D
Initial  
David Heinemeier Hansson 已提交
253
    end
254 255 256 257

    def csv_file_path
      @fixture_path + ".csv"
    end
258

D
Initial  
David Heinemeier Hansson 已提交
259 260 261 262 263 264 265 266 267 268 269
    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
270 271 272 273
  class FixtureError < StandardError#:nodoc:
  end
  class FormatError < FixtureError#:nodoc:
  end
D
Initial  
David Heinemeier Hansson 已提交
274 275 276 277 278 279 280 281 282 283 284 285 286 287 288 289 290 291 292 293 294 295 296 297 298

  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
299

D
Initial  
David Heinemeier Hansson 已提交
300
  def find
301 302 303 304
    if Object.const_defined?(@class_name)
      klass = Object.const_get(@class_name)
      klass.find(self[klass.primary_key])
    end
D
Initial  
David Heinemeier Hansson 已提交
305
  end
306

D
Initial  
David Heinemeier Hansson 已提交
307 308 309 310
  private
    def read_fixture_file(fixture_file_path)
      IO.readlines(fixture_file_path).inject({}) do |fixture, line|
        # Mercifully skip empty lines.
311
        next if line =~ /^\s*$/
D
Initial  
David Heinemeier Hansson 已提交
312 313 314

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

        # Disallow duplicate keys to catch typos.
320
        raise FormatError, "#{fixture_file_path}: duplicate '#{key}' in fixture." if fixture[key]
D
Initial  
David Heinemeier Hansson 已提交
321 322 323 324 325 326
        fixture[key] = value.strip
        fixture
      end
    end
end

327 328 329 330
module Test#:nodoc:
  module Unit#:nodoc:
    class TestCase #:nodoc:
      include ClassInheritableAttributes
331

332 333
      cattr_accessor :fixture_path
      cattr_accessor :fixture_table_names
D
Initial  
David Heinemeier Hansson 已提交
334

335 336 337
      def self.fixtures(*table_names)
        require_fixture_classes(table_names)
        write_inheritable_attribute("fixture_table_names", table_names)
338 339
      end

340 341 342 343 344 345 346 347 348
      def self.require_fixture_classes(table_names)
        table_names.each do |table_name| 
          begin
            require(Inflector.singularize(table_name.to_s))
          rescue LoadError
            # Let's hope the developer is included it himself
          end
        end
      end
349

350
      def setup
D
Initial  
David Heinemeier Hansson 已提交
351 352 353
        instantiate_fixtures(*fixture_table_names) if fixture_table_names
      end

354 355 356 357 358 359 360 361 362
      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
363

364 365 366 367 368 369 370 371
      private
        def instantiate_fixtures(*table_names)
          Fixtures.instantiate_fixtures(self, fixture_path, *table_names)
        end

        def fixture_table_names
          self.class.read_inheritable_attribute("fixture_table_names")
        end
D
Initial  
David Heinemeier Hansson 已提交
372
    end
373
  end
374
end