CSC/ECE 517 Spring 2014/ch1a 1k wm
Factory Girl
Background
From Factory Girl's Ruby Toolbox page<ref>https://www.ruby-toolbox.com/projects/factory_girl</ref>:
factory_girl provides a framework and DSL for defining and using factories - less error-prone, more explicit, and all-around easier to work with than fixtures.
Also from the Factory Girl's github page<ref>https://github.com/thoughtbot/factory_girl</ref>:
factory_girl is a fixtures replacement with a straightforward definition syntax, support for multiple build strategies (saved instances, unsaved instances, attribute hashes, and stubbed objects), and support for multiple factories for the same class (user, admin_user, and so on), including factory inheritance.
Factory Girl's purpose is to expedite good testing by creating test fixtures for the developer, eliminating the need to create large fixtures by hand. There are two versions, one for use with regular Ruby (factory_girl) and one for use with Ruby on Rails (factory_girl_rails<ref>https://github.com/thoughtbot/factory_girl_rails</ref>)
Test fixtures are used to create an application state through which you can test code.<ref>http://guides.rubyonrails.org/testing.html#the-low-down-on-fixtures</ref> This can be as simple as constructing two objects with which to perform a comparison (testing the comparison operator) or as complicated as can be imagined.
Generic Examples
All examples taken from the github project<ref>https://github.com/thoughtbot/factory_girl/blob/master/GETTING_STARTED.md</ref>
Factories
Factories are the basic unit of Factory Girl, they wrap up a constructor into a single method call that can be used in multiple test fixtures. A factory looks pretty similar to a normal constructor but can be used multiple times without having to specify any parameters, as shown below.
FactoryGirl.define do
factory :user do
first_name "John"
last_name "Doe"
admin false
end
end
The above creates a factory for class User (inferred by Factory Girl from the name of the factory) that can then be called in many different ways. You can also define factories using specific classes, not only inferred from the name of the factory, as below (in the same word as the above example).
FactoryGirl.define do
factory :admin, class: User do
first_name "Admin"
last_name "User"
admin true
end
end
Using Factories
You can use factories in a variety of ways, as shown. You can also overwrite predefined attributes on the fly by passing in a hash of attributes and values.
# Returns a User instance that's not saved user = build(:user) # Returns a saved User instance user = create(:user) # Returns a hash of attributes that can be used to build a User instance attrs = attributes_for(:user) # Returns an object with all defined attributes stubbed out stub = build_stubbed(:user) # Passing a block to any of the methods above will yield the return object create(:user) do |user| user.posts.create(attributes_for(:post)) end # Build a User instance and override the first_name property user = build(:user, first_name: "Joe") user.first_name # => "Joe"
Aliasing
You can utilize aliases to make somewhat more specialized factories out of existing factories, as shown.
factory :user, aliases: [:author, :commenter] do
first_name "John"
last_name "Doe"
date_of_birth { 18.years.ago }
end
factory :post do
author
# instead of
# association :author, factory: :user
title "How to read a book effectively"
body "There are five steps involved."
end
factory :comment do
commenter
# instead of
# association :commenter, factory: :user
body "Great article!"
end
The above code creates a factory called user and then uses that factory to add users as attributes of post and comment with a different name (author and commenter).
Lazy, Dependent, and Transient Attributes
Attributes need not be statically defined. There are several other ways to define attributes of factories, all termed lazy attributes because they acquire a value after the object has been instantianted. The simplest case of lazy attributes is the simple lazy attribute, determined by a block rather than given as a static value.
factory :user do
# ...
activation_code { User.generate_activation_code }
date_of_birth { 21.years.ago }
end
Dependent attributes are lazy attributes that use the values of other attributes in the factory to determine their value.
factory :user do
first_name "Joe"
last_name "Blow"
email { "#{first_name}.#{last_name}@example.com".downcase }
end
create(:user, last_name: "Doe").email
# => "joe.doe@example.com"
Finally, transient attributes allow you to further modify factories and prevent repetition of code, in a way passing arguments to the factory creation function.
factory :user do
ignore do
rockstar true
upcased false
end
name { "John Doe#{" - Rockstar" if rockstar}" }
email { "#{name.downcase}@example.com" }
after(:create) do |user, evaluator|
user.name.upcase! if evaluator.upcased
end
end
create(:user, upcased: true).name
#=> "JOHN DOE - ROCKSTAR"
Further Reading
Factory Girl contains far more than can be expressed in so verbose a fashion in a summary page, so be sure to read the entirety of the Getting Started guide. This guide goes through all of the above and more, in detail, with code examples.
Using with Ruby on Rails
As mentioned earlier, Factory Girl can be used in Ruby on Rails through the factory_girl_rails gem. All of the same benefits apply, including those not mentioned above.
Example
A typical test scenario is benchmarking the time a sort takes to complete on a random assortment of objects. A stock Rake test fixture implementation might look something like the following:
person1:
id: 1
name: Bob
dob: 10-11-1990
person2:
id: 2
name: George
dob: 9-8-1990
person3:
id: 3
name: Stanley
dob: 1-9-1990
This could get tedious with more than 3 values, so you could use ERb (Embedded Ruby)
<% for i in 1..1000 %>
fix_<%= i %>:
id: <%= i %>
name: guy_<%= 1 %>
dob: <%= Date.today.strftime("%Y-%m-%d") %>
<% end %>
However, this is harder to maintain and is not very DRY-friendly. Instead, we could use Factory Girl and expedite the process, also making it more DRY while we're at it (using some Factory Girl features not shown above)!
FactoryGirl.define do
sequence :email do |n|
"person#{n}@example.com"
end
sequence :id do |n|
#{n}
end
sequence :dob do |n|
(20.years.ago)+(Random.rand(20).months)#give random DOB from 20 years ago + 0..20 months
end
factory :user do
id
first_name "John"
last_name "Doe"
email
dob
end
end
built_users_2500 = build_list(:user, 2500) #make 2500 users with unique email addresses and id's and random DOB
built_users_10 = build_list(:user, 10) #make 10 more users
References
<references />