向Rails4应用程序

时间:2016-10-05 18:44:33

标签: ruby-on-rails ruby-on-rails-4 rspec rspec-rails

关注Railscast #385& #386我已将权限添加到我的Rails 4应用程序(rails 4.2.6,rspec-rails 3.4.2)。添加授权后,我的所有控制器规格都会失败。

我的功能规格仍然通过,因为在规范中我以admin身份登录以执行不允许访问者的操作(:edit,:update ...)。如果我允许所有操作允许访问者我的控制器规格通过。但是,只有管理员才能执行的任何操作都将在控制器规范中失败。

脚手架生成器创建一个空

  let(:valid_session) { { } }

它在评论中说:

  

#这应该返回应该在会话中的最小值集       为了传递MyController中定义的任何过滤器(例如身份验证)。

但是,我不知道如何配置:valid_session以允许身份验证通过。

当我尝试在此question中发布的解决方案时:

def valid_session
  controller.stub!(:signed_in?).and_return(true)
end

我收到此错误:

NoMethodError:
        undefined method `stub' for #<MyController

因为,逻辑的所有分支路径都应该在较低级别(即控制器规范)而不是功能规格上进行测试,如何让控制器规格通过?

如何配置valid_session或我可以使用哪种方法在控制器规范中传递授权?

2 个答案:

答案 0 :(得分:2)

好的,在blog的帮助下,我可以在向控制器添加授权后让我的控制器测试通过。

spec / support / helpers / controller_macros.rb 中我定义了以下方法:

module ControllerMacros
  def login_admin
    user = FactoryGirl.create(:admin)
    allow(controller).to receive(:current_user).and_return(user)
  end
  def login_member
    user = FactoryGirl.create(:member)
    allow(controller).to receive(:current_user).and_return(user)
  end
  def login_visitor
    allow(controller).to receive(:current_user).and_return(nil)
  end
end

spec / support / helpers.rb 中激活这些方法:

RSpec.configure do |config|
  config.include ControllerMacros, type: :controller
end

然后在控制器规范中实现它, spec / controllers / topics_controller_spec.rb

require 'rails_helper'
RSpec.describe TopicsController, type: :controller do
  let(:valid_attributes) {
    { :name => "MyTopic" }
  }
  let(:invalid_attributes) {
    { :name => nil }
  }
  before(:each) do
    login_admin
  end
  describe "GET #index" do
    it "assigns all topics as @topics" do
      login_visitor
      topic = Topic.create! valid_attributes
      get :index, {}
      expect(assigns(:topics)).to eq([topic])
    end
  end
  describe "GET #show" do
    it "assigns the requested topic as @topic" do
      login_visitor
      topic = Topic.create! valid_attributes
      get :show, {:id => topic.to_param}
      expect(assigns(:topic)).to eq(topic)
    end
  end
  describe "GET #new" do
    it "assigns a new topic as @topic" do
      get :new, {}
      expect(assigns(:topic)).to be_a_new(Topic)
    end
  end
[...]
end

这将在每次测试前以管理员身份登录。 请注意,您应该在具有权限的最低级别测试每个操作,因此show和index将作为访问者进行测试(未登录)。如果所有操作都需要管理员权限,那么您可以使用{{ 1}}而不是before do在测试中节省一点时间。

最后,您必须从规范中删除所有before(:each) do的提及。

答案 1 :(得分:0)

你可以只为工厂用户。

module ControllerMacros
  def login_user(user = nil, options = {})
    before(:each) do
    @request.env["devise.mapping"] = Devise.mappings[:user]
    @user = user || FactoryGirl.create(options[:user] || :user)
    sign_in @user
  end
end