CSC/ECE 517 Spring 2014/ch1a 1k wm: Difference between revisions
(Page creation) |
|||
Line 1: | Line 1: | ||
=Factory Girl= | =Factory Girl= | ||
[https://docs.google.com/document/d/1XzVu0zEuk7LP4smGZV9lRpSx5-wri6oClpUAmGYPN44/edit# Assignment Spec] | |||
==Background== | ==Background== | ||
From Factory Girl's Ruby Toolbox page<ref>https://www.ruby-toolbox.com/projects/factory_girl</ref>: | From Factory Girl's Ruby Toolbox page<ref>https://www.ruby-toolbox.com/projects/factory_girl</ref>: |
Revision as of 01:02, 15 February 2014
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 />