- Generate a members index page, and a book club index page
Member
- first_name (string, req)
- last_name (string, req)
- email (string, req)
- bio (long string)
- favorite_book (string)
- leader (boolean, req)
- book_club (integer, req)
Book Club
- name (string, req)
- location (string)
Your views have been created for you!
Use the tests to get your features to pass.
Be sure to run bundle exec rake:db:test:prepare
after running any migrations.
The commented out code in the user_views_book_club_details spec will require knowledge of ActiveRecord Associations in order to complete, as well as an addition to the provided erb
file.
FactoryBot is a gem that makes testing easier. It allows you to create "factories" for the objects your app is concerned with, thereby allowing you to more quickly set up tests for a feature or method. Once FactoryBot is set up, instead of needing to type this:
let(:leader) { Member.create(first_name: "Emily", last_name: "Dickinson", email: "[email protected]", bio: "I don't see what's so great about leaving the house.", favorite_book: "Aurora Leigh", leader: true) }
I can use something more simple, like this:
let(:leader) { FactoryBot.create(:club_leader) }
The following has already been done, but will be the steps to add FactoryBot to your apps in the future:
First, add FactoryBot to your Gemfile:
group :development, :test do
# ... other gems ...
gem 'factory_bot'
end
- Run
bundle install
to install it! - Next, make a file at
spec/support/factories.rb
. This is where you'll be creating your factories. - Now open the spec_helper.rb file and add these lines to the top:
require 'factory_bot'
require_relative 'support/factories'
-
To create your database, open
config/database.yml
and update the file with database names appropriate for your app. Then runrake db:create
. -
Lastly,
require spec_helper
at the top of spec test files.
Open up spec/support/factories.rb
. Add the following code:
FactoryBot.define do
# Your factories will go here!
end
You are now ready to write up your factories inside of this block!
Writing a Simple Factory (Documentation)
If we're using a Book Club example, we'll have Book Club
objects and Member
objects. Let's set these tables up in our database and write their associated models:
rake db:create_migration NAME=create_book_clubs
class CreateBookClubs < ActiveRecord::Migration
def change
create_table :book_clubs do |t|
t.string :name, null: false
t.string :location
end
end
end
class BookClub < ActiveRecord::Base
# requires knowledge of associations
# has_many :members
end
rake db:create_migration NAME=create_members
class CreateMembers < ActiveRecord::Migration
def change
create_table :members do |t|
t.string :first_name, null: false
t.string :last_name, null: false
t.string :email, null: false
t.text :bio
t.string :favorite_book
t.belongs_to :book_club
t.boolean :leader, null: false, default: false
end
end
end
class Member < ActiveRecord::Base
# requires knowledge of associations
# belongs_to :book_club
end
Let's start with our Member
object. Here's what creating a factory for that object would look like:
factory :member do
first_name "Emily"
last_name "Dickinson"
email "[email protected]"
bio "I don't see what's so great about leaving the house."
favorite_book "Aurora Leigh"
end
Note that each of lines is actually a method being called, with an argument of the value you want your object to be created with.
It's that simple! Now every time I call FactoryBot.create(:member)
, I'll have a standardized book club member (in this case, Emily Dickinson).
Using a Factory (Documentation)
Let's look at an example of how I would use this.
feature 'book club member directory' do
let!(:emily_dickinson) { FactoryBot.create(:member) }
scenario "view list of all book club members" do
visit '/members'
expect(page).to have_content("All Book Club Members")
expect(page).to have_content(emily_dickinson.first_name)
expect(page).to have_content(emily_dickinson.email)
end
end
But what if we want to customize some part of the Factory Bot created object? Let's try that out:
feature "book club member directory" do
let!(:emily_dickinson) { FactoryBot.create(:member)}
let!(:walt_whitman) { FactoryBot.create(:member, first_name: "Walt", last_name: "Whitman", bio: "Yawp") }
scenario "view list of all book club members" do
visit '/members'
expect(page).to have_content("All Book Club Members")
expect(page).to have_content(emily_dickinson.first_name)
expect(page).to have_content(emily_dickinson.email)
expect(page).to have_content(walt_whitman.first_name)
expect(page).to have_content(walt_whitman.email)
end
end
You can override any of the attributes that your factory sets by passing in an argument to modify that attribute. In this example, we're changing the first name, last name, and bio of the factory to suit our needs. This allows us to avoid duplicating some of our work in mocking up the data needed to test things that book club members can do, but with custom values when needed.
Objects with Associations (Documentation)
Note: you will need to be fluent in basic ActiveRecord associations in order to use the following code snippets.
In our set up, a Book Club
has many Member
s, and a Member
belongs to a Book Club
. Usually when we create a member object, we'll know we want a book club object also. So we can add a factory for this purpose to our factories.rb
file:
factory :book_club do
name "(Actually) Dead Poets Society"
location "Amherst, MA"
end
Once that factory is set up, we can go back to our member
factory and just add the word book_club
on its own line in the factory definition. FactoryBot will look for a factory of that name and create the associated object for us:
factory :member do
first_name "Emily"
last_name "Dickinson"
email "[email protected]"
bio "I don't see what's so great about leaving the house."
favorite_book "Aurora Leigh"
book_club
end
Now every time we create a new book club member, there will be a book club to go with it!
Here's an example:
feature "view a particular book club's members" do
let!(:emily_dickinson) { FactoryBot.create(:member)}
let(:book_club) { emily_dickinson.book_club }
let!(:walt_whitman) { FactoryBot.create(:member, first_name: "Walt", last_name: "Whitman", bio: "Yawp", book_club: book_club ) }
scenario "see all members of a particular book club" do
visit "/book_clubs/#{book_club.id}"
expect(page).to have_content(book_club.name)
expect(page).to have_content(emily_dickinson.first_name)
expect(page).to have_content(walt_whitman.first_name)
end
end
Or, let's say we want to add a book club member to a specific, pre-existing club. We can just overwrite this default information like before:
book_club = FactoryBot.create(:book_club, name: "Cranky Poet's Society")
emily_dickinson = FactoryBot.create(:member, book_club: book_club)
Now Emily will belong to the book club we already created, and the factory won't create a new book club when it creates her membership.
Different Kinds of Objects (aka Inheritance)
Let's say when our book club members sign up, they can either check a box to offer to lead their book club during its meetings or not. Passing override values to our member
factory might get old after a while if we need to write 15 feature tests where a book club leader is required. Let's go ahead and make a permanent factory for this kind of object. We can do this by inheriting most of the properties we want from the parent member
object:
factory :member do
first_name "Emily"
last_name "Dickinson"
email "[email protected]"
bio "I don't see what's so great about leaving the house."
favorite_book "Aurora Leigh"
factory :club_leader do
leader true
end
end
If we call FactoryBot.create(:club_leader)
, all of the default traits we set up in the normal member
factory will still be there, except the one we explicitly overrode in the book_club_leader
factory. Nesting things this way makes creating new factories for explicit uses very simple.
Let's update our previous feature test to leverage this improved factory set up:
feature "view a particular book club's members" do
scenario "see all members of a particular book club" do
let!(:emily_dickinson) { FactoryBot.create(:club_leader)}
let(:book_club) { emily_dickinson.book_club }
let!(:walt_whitman) { FactoryBot.create(:member, first_name: "Walt", last_name: "Whitman", bio: "Yawp", book_club: book_club) }
visit "/book_clubs/#{book_club.id}"
expect(page).to have_content(book_club.name)
expect(page).to have_content("#{emily_dickinson.first_name} (Leader)")
expect(page).to have_content(walt_whitman.first_name)
end
end
Objects with Uniqueness Constraints (aka Sequencing)
As of right now, all book club members I create will have the same email (unless otherwise specified). What if I have a uniqueness constraint on book club members' emails, so I can't have duplicate emails in my database?
Let's write the migration that would modify this:
rake db:create_migration NAME=make_member_emails_unique
class MakeMemberEmailsUnique < ActiveRecord::Migration
def change
change_column :members, :email, :string, unique: true, null: false
end
end
Let's also add a uniqueness constraint on the class/model itself:
class Member < ActiveRecord::Base
belongs_to :book_club
validates :email, uniqueness: true
end
Now we can return to our testing and set up a sequence, meaning that each time you use the factory to create a new book club member, the n
in the block you wrote will increment by 1. So first we'll have "[email protected]", then "[email protected]", etc!
# factories.rb
factory :member do
# ..other attributes
sequence(:email) { |n| "nobody#{n}@nobodytoo.org" }
end
Let's run the feature test we wrote before to make sure this is working as expected:
feature "book club member directory" do
scenario "view list of all book club members" do
emily_dickinson = FactoryBot.create(:member)
walt_whitman = FactoryBot.create(:member,first_name: "Walt", last_name: "Whitman", bio: "Yawp")
visit '/members'
expect(page).to have_content("All Book Club Members")
expect(page).to have_content(emily_dickinson.first_name)
expect(page).to have_content(emily_dickinson.email)
expect(page).to have_content(walt_whitman.first_name)
expect(page).to have_content(walt_whitman.email)
end
end
Lists of Objects
Now, so far our book club tests have only had a couple of members, but maybe we want to test a more realistic scenario wherein our book club has 15 members, 3 of whom are club leaders. Factory Bot can help us do this pretty easily with create_list
. Here's how it might look:
feature "view a book club's members" do
scenario 'see all members of a particular book club' do
book_club = FactoryBot.create(:book_club)
members = FactoryBot.create_list(:member, 12, book_club: book_club)
leaders = FactoryBot.create_list(:club_leader, 3, book_club: book_club)
visit "/book_clubs/#{book_club.id}"
members.each do |member|
expect(page).to have_content(member.first_name)
end
leaders.each do |leader|
expect(page).to have_content("#{leader.first_name} (Leader)")
end
end
end
Listed below are some additional resources on using Factory Bot. Note: These resources will presume you use Rails with ActiveRecord and FactoryBot. It works the same way as in Sinatra with ActiveRecord.