Logo Questions Linux Laravel Mysql Ubuntu Git Menu
 

Accessing controller instance variables from within an rspec controller spec

Shouldn't I be able to see instance variables which are created in a controller action from within my rspect tests?

# /app/controllers/widget_controller.rb
...
def show
  @widget = ...
  puts "in controller: #{@widget}"
end
...

--

# /spec/controllers/widget_controller_spec.rb
RSpec.describe WidgetController, type: :controller do
...
describe "GET #show" do
  it "assigns the requested widget as @widget" do
    get :show, { :id => 1 } # this is just an example - I'm not hardcoding the id

    puts "in spec: #{@widget}"
  end
end
...

Here is the output I get when I run that spec:

controller: #<Widget:0x007f9d02aff090>
in spec:

Am I wrong in thinking that I should have access to @widget in my controller spec?

like image 742
RobertJoseph Avatar asked Sep 17 '15 16:09

RobertJoseph


3 Answers

Use the assigns method (*note: assigns is now deprecated. See bottom of my answer for info):

it "assigns the @widget"
  expect(assigns(:widget)).not_to be_nil
end

Of course, you can inspect widget as you'd like, but without seeing what @widget is from your provided controller code, I just checked if it was nil

If you're wanting to puts the widget, like in your example, just print it with assigns

puts assigns(:widget)

EDIT: assigns is now deprecated (see: https://apidock.com/rails/ActionController/TestProcess/assigns)

If you want to continue using assigns you will need to install the rails-controller-testing gem.

Otherwise, you could use what rails-controller-testing gem uses internally: @controller.view_assigns[]

like image 196
Wes Foster Avatar answered Oct 08 '22 19:10

Wes Foster


#assigns is deprecated. Here's a solution that avoids pulling in the rails-controller-testing gem

Though it can be a code smell to test instance variables in controllers, to upgrade older apps you can leverage #instance_variable_get.

Rspec example: expect(@controller.instance_variable_get(:@person).id).to eq(person_id)

like image 31
stevepentler Avatar answered Oct 08 '22 17:10

stevepentler


In Rails 5, assigns has now been split out into a new rails-controller-testing gem.

You can either install the gem to continue using assigns, or you could use what the rails-controller-testing gem uses internally, which@controller.view_assigns[].

like image 35
Will Clarke Avatar answered Oct 08 '22 17:10

Will Clarke