RSpec Best Friends
Mauro quem...
maurogeorge.com.br
RSpec
RSpec 
descrevendo melhor os testes
spec/models/pokemon_spec.rb 
it 'exibe o nome e o id nacional' do 
pokemon = Pokemon.new(nome: 'Charizard', id_nacional: 6) 
expect(pokemon.nome_completo).to eq('Charizard - 6') 
end
spec/models/pokemon_spec.rb 
describe '#nome_completo' do 
it 'exibe o nome e o id nacional' do 
pokemon = Pokemon.new(nome: 'Charizard', id_nacional: 6) 
expect(pokemon.nome_completo).to eq('Charizard - 6') 
end 
end
RSpec 
definindo o sujeito
spec/models/pokemon_spec.rb 
context 'quando possui nome e o id nacional' do 
before do 
@pokemon = Pokemon.new(nome: 'Charizard', id_nacional: 6) 
end 
! 
it 'exibe o nome e o id nacional' do 
expect(@pokemon.nome_completo).to eq('Charizard - 6') 
end 
end
spec/models/pokemon_spec.rb 
context 'quando possui nome e o id nacional' do 
let(:pokemon) do 
Pokemon.new(nome: 'Charizard', id_nacional: 6) 
end 
! 
it 'exibe o nome e o id nacional' do 
expect(pokemon.nome_completo).to eq('Charizard - 6') 
end 
end
spec/models/pokemon_spec.rb 
context 'quando possui nome e o id nacional' do 
subject do 
Pokemon.new(nome: 'Charizard', id_nacional: 6) 
end 
! 
it 'exibe o nome e o id nacional' do 
expect(subject.nome_completo).to eq('Charizard - 6') 
end 
end
RSpec 
ordem aleatória nos testes
spec/spec_helper.rb 
RSpec.configure do |config| 
# ... 
config.order = "random" 
end
Testes que acessam rede
Testes que acessam rede 
introdução 
Testes lentos 
Testes quebradiços 
Não poder testar sem rede
app/services/criador_pokemon.rb 
class CriadorPokemon 
# ... 
def criar 
Pokemon.create(nome: nome) 
end 
! 
private 
# ... 
def cria_info 
resposta = Net::HTTP.get(endpoint) 
@info = JSON.parse(resposta) 
end 
end
spec/services/criador_pokemon_spec.rb 
describe 'pokemon criado' do 
before do 
criador_pokemon.criar 
end 
! 
subject do 
Pokemon.last 
end 
! 
it 'possui o nome correto' do 
expect(subject.nome).to eq('Charizard') 
end 
end
Testes que acessam rede 
vcr
Testes que acessam rede 
vcr: configuração
spec/support/vcr.rb 
VCR.configure do |c| 
c.cassette_library_dir = 'spec/fixtures/vcr_cassettes' 
c.hook_into :webmock 
end
Testes que acessam rede 
vcr: feedback rápido
bash 
Failure/Error: CriadorPokemon.new(6) 
VCR::Errors::UnhandledHTTPRequestError: 
! 
! 
===================================================================== 
An HTTP request has been made that VCR does not know how to handle: 
GET http://pokeapi.co/api/v1/pokemon/6/ 
! 
There is currently no cassette in use. There are a few ways 
you can configure VCR to handle this request: 
! 
...
Testes que acessam rede 
vcr: forjando a resposta
spec/services/criador_pokemon_spec.rb 
describe 'pokemon criado' do 
before do 
VCR.use_cassette('CriadorPokemon/criar') do 
criador_pokemon.criar 
end 
end 
! 
#... 
! 
it 'possui o nome correto' do 
expect(subject.nome).to eq('Charizard') 
end 
end
Testes que acessam rede 
vcr: RSpec metadata
spec/support/vcr.rb 
VCR.configure do |c| 
! 
# ... 
c.configure_rspec_metadata! 
end 
spec/spec_helper.rb 
RSpec.configure do |config| 
! 
# ... 
config.treat_symbols_as_metadata_keys_with_true_values = true 
end
spec/services/criador_pokemon_spec.rb 
describe 'pokemon criado', :vcr do 
before do 
criador_pokemon.criar 
end 
! 
#... 
! 
it 'possui o nome correto' do 
expect(subject.nome).to eq('Charizard') 
end 
end
factory_girl
factory_girl 
fixtures X factories
factory_girl 
criando uma factory
spec/factories/usuarios.rb 
FactoryGirl.define do 
factory :usuario do 
nome 'Mauro' 
email 'mauro@helabs.com.br' 
end 
end
console rails 
FactoryGirl.create(:usuario) 
FactoryGirl.create(:usuario, email: 'mauro@helabs.com.br')
factory_girl 
configurando
spec/spec_helper.rb 
RSpec.configure do |config| 
# ... 
config.include FactoryGirl::Syntax::Methods 
end 
Em um teste qualquer 
let!(:artigo) do 
create(:artigo) 
end
factory_girl 
attributes_for
spec/controllers/posts_controller_spec.rb 
describe "POST 'create'" do 
let(:params) do 
{ 
artigo: { 
titulo: 'Meu titulo', 
conteudo: 'Conteudo do artigo' 
} 
} 
end 
end
spec/controllers/posts_controller_spec.rb 
describe "POST 'create'" do 
let(:params) do 
{ artigo: attributes_for(:artigo) } 
end 
end
factory_girl 
herança
spec/factories/artigos.rb 
factory :artigo do 
titulo 'Diversas dicas do RSpec' 
conteudo 'Conteúdo de Diversas dicas do RSpec' 
! 
factory :artigo_aprovado do 
aprovado true 
end 
! 
factory :artigo_nao_aprovado do 
aprovado false 
end 
end
console rails 
FactoryGirl.create(:artigo_aprovado) 
FactoryGirl.create(:artigo_nao_aprovado)
factory_girl 
traits
spec/factories/artigos.rb 
factory :artigo do 
titulo 'Diversas dicas do RSpec' 
conteudo 'Conteúdo de Diversas dicas do RSpec' 
! 
trait :aprovado do 
aprovado true 
end 
! 
trait :nao_aprovado do 
aprovado false 
end 
end
console rails 
FactoryGirl.create(:artigo, :aprovado) 
FactoryGirl.create(:artigo, :nao_aprovado)
factory_girl 
associações
console rails 
usuario = FactoryGirl.create(:usuario) 
FactoryGirl.create(:artigo, usuario: usuario)
spec/factories/artigos.rb 
factory :artigo do 
titulo 'Diversas dicas do RSpec' 
conteudo 'Conteúdo do artigo' 
usuario 
end
factory_girl 
strategies
console rails 
pokemon = FactoryGirl.build(:pokemon)
console rails 
pokemon = FactoryGirl.build_stubbed(:pokemon)
timecop
app/models/pokemon.rb 
class Pokemon < ActiveRecord::Base 
! 
! 
scope :escolhidos_ontem, -> do 
where(escolhido_em: 1.day.ago.midnight..Time.zone.now.midnight) 
end 
end
spec/models/pokemon_spec.rb 
describe '.escolhidos_ontem' do 
let!(:pokemon_escolhido_ontem) do 
create(:pokemon, escolhido_em: Time.zone.local(2014, 10, 16, 16, 50)) 
end 
! 
subject do 
Pokemon.escolhidos_ontem 
end 
! 
it 'tem o pokemon escolhido ontem' do 
expect(subject).to include(pokemon_escolhido_ontem) 
end 
end
spec/models/pokemon_spec.rb 
describe '.escolhidos_ontem' do 
! 
# ... 
! 
it 'tem o pokemon escolhido ontem' do 
Timecop.freeze(Time.zone.local((2014, 10, 16, 16, 50)) do 
expect(subject).to include(pokemon_escolhido_ontem) 
end 
end 
end
simplecov
simplecov 
verificando a cobertura
spec/spec_helper.rb 
require 'simplecov' 
SimpleCov.start 'rails' 
Primeira linha do 
spec_helper.rb
simplecov 
O falso 100%
app/models/pokemon.rb 
class Pokemon < ActiveRecord::Base 
! 
validates :nome, :id_nacional, presence: true 
scope :escolhidos_ontem, -> do 
where(escolhido_em: 1.day.ago.midnight..Time.zone.now.midnight) 
end 
end
simplecov 
Não teste associações, validações ou 
escopos do Active Record
simplecov 
teste associações, validações e escopos do 
Active Record
simplecov 
devo ter 100% de cobertura de testes?
shoulda-matchers
app/models/pokemon.rb 
class Pokemon < ActiveRecord::Base 
! 
validates :nome, :id_nacional, presence: true 
validates :id_nacional, numericality: { only_integer: true, greater_than: 0 } 
end
spec/models/pokemon_spec.rb 
describe 'validações' do 
! 
it { is_expected.to validate_presence_of(:nome) } 
it { is_expected.to validate_presence_of(:id_nacional) } 
it { is_expected.to validate_numericality_of(:id_nacional).only_integer 
.is_greater_than(0) } 
end
shoulda-matchers 
os matchers 
ActiveModel 
ActiveRecord 
ActionController
shoulda-matchers 
além do shoulda-matchers 
https://github.com/bmabey/email-spec 
https://github.com/philostler/rspec-sidekiq 
https://github.com/evansagge/mongoid-rspec
Obrigado!
maurogeorge.com.br

RSpec Best Friends @ Rupy Natal 2014