如何在RSpec中使用HTTP状态代码符号?

时间:2013-08-16 13:39:26

标签: ruby-on-rails rspec http-status-codes human-readable

我在控制器中的代码中使用HTTP status code symbols,例如:

render json: {
    auth_token: user.authentication_token, 
    user: user
  }, 
  status: :created

render json: {
    errors: ["Missing parameter."]
  }, 
  success: false, 
  status: :unprocessable_entity

在我的请求规范的代码中,我也想使用符号:

post user_session_path, email: @user.email, password: @user.password
expect(last_response.status).to eq(201)

...

expect(last_response.status).to eq(422)

然而,我使用符号而不是整数的每个测试都失败了:

Failure/Error: expect(last_response.status).to eq(:created)

  expected: :created
       got: 201

  (compared using ==)

以下是HTTP status code symbols in Rack的最新列表。

4 个答案:

答案 0 :(得分:21)

response对象作为消息响应多个符号类型。所以你可以这样做:

expect(response).to be_success
expect(response).to be_error
expect(response).to be_missing
expect(response).to be_redirect

对于其他类型,例如:created,您可以为此创建一个包含assert_response的简单自定义匹配器:

RSpec::Matchers.define :have_status do |type, message = nil|
  match do |_response|
    assert_response type, message
  end
end

expect(response).to have_status(:created)
expect(response).to have_status(404)

对于具有正确状态设置的控制器规范,这应该可以正常工作。它将用于功能规范。我没有尝试过请求规格,所以你的milage可能会有所不同。

这样做的原因是它利用了RSpec控制器规范在幕后具有类似状态设置的事实。因此,当assert_response访问@response时,它就可用了。

只需将assert_response使用的代码复制到匹配器中,就可以改进此匹配器:

RSpec::Matchers.define :have_status do |type, message = nil|
  match do |response|
    if Symbol === type
      if [:success, :missing, :redirect, :error].include?(type)
        response.send("#{type}?")
      else
        code = Rack::Utils::SYMBOL_TO_STATUS_CODE[type]
        response.response_code == code
      end
    else
      response.response_code == type
    end
  end

  failure_message do |response|
    message or
      "Expected response to be a <#{type}>, but was <#{response.response_code}>"
  end
end

更新时间:2014-07-02

现在可以使用RSpec Rails 3开箱即用:https://www.relishapp.com/rspec/rspec-rails/v/3-0/docs/matchers/have-http-status-matcher

答案 1 :(得分:20)

这对我有用:

expect(response.response_code).to eq(Rack::Utils::SYMBOL_TO_STATUS_CODE[:not_found])

答案 2 :(得分:11)

一方面,响应是通过以下方法构建的:

  • 成功?

  • 重定向?

  • 不可处理?

  • 完整列表:response.methods.grep(/\?/)

另一方面,Rspec谓词将每个foo?方法转换为be_foo匹配器。

不幸的是,不确定你是否能以这种方式获得201,但创建自定义匹配器非常容易。

注意Rails仅测试rely on a few statuses

答案 3 :(得分:8)

使用 rspec-rails (从 rspec 3开始)可以使用

expect(response).to have_http_status(:created)

更新2018-06-11

从Rails 6开始,一些匹配器将被替换(例如success successful}。