В приложении Rails 4.2.0, протестированном с помощью rspec-rails
, я предоставляю веб-API JSON ресурсом, подобным REST, с обязательным атрибутом mand_attr
.
Я бы хотел проверить, что этот API отвечает с кодом HTTP 400 (BAD REQUEST
), когда этот атрибут отсутствует в запросе POST. (см. второй пример). Мой контроллер пытается вызывают этот HTTP-код, бросая ActionController::ParameterMissing
, как показано первым примером RSpec ниже.
В других примерах RSpec, я хочу, чтобы исключения были спасены примерами (если они ожидаются) или попали в тестовый бегун, поэтому они отображаются разработчику (если ошибка неожиданна), поэтому я не хочу удалять
# Raise exceptions instead of rendering exception templates.
config.action_dispatch.show_exceptions = false
из config/environments/test.rb
.
Мой план состоял в следующем: request spec:
describe 'POST' do
let(:perform_request) { post '/my/api/my_ressource', request_body, request_header }
let(:request_header) { { 'CONTENT_TYPE' => 'application/json' } }
context 'without mandatory attribute' do
let(:request_body) do
{}.to_json
end
it 'raises a ParameterMissing error' do
expect { perform_request }.to raise_error ActionController::ParameterMissing,
'param is missing or the value is empty: mand_attr'
end
context 'in production' do
###############################################################
# How do I make this work without breaking the example above? #
###############################################################
it 'reports BAD REQUEST (HTTP status 400)' do
perform_request
expect(response).to be_a_bad_request
# Above matcher provided by api-matchers. Expectation equivalent to
# expect(response.status).to eq 400
end
end
end
# Below are the examples for the happy path.
# They're not relevant to this question, but I thought
# I'd let you see them for context and illustration.
context 'with mandatory attribute' do
let(:request_body) do
{ mand_attr: 'something' }.to_json
end
it 'creates a ressource entry' do
expect { perform_request }.to change(MyRessource, :count).by 1
end
it 'reports that a ressource entry was created (HTTP status 201)' do
perform_request
expect(response).to create_resource
# Above matcher provided by api-matchers. Expectation equivalent to
# expect(response.status).to eq 201
end
end
end
Я нашел два рабочих и один частично работающих решения, которые я выложу в качестве ответов. Но я не особенно доволен ни одним из них, поэтому, если вы можете придумать что-то лучшее (или просто другое), , я бы хотел увидеть ваш подход! Кроме того, если спецификация запроса неправильный тип спецификации, чтобы проверить это, я хотел бы знать это.
Я предвижу вопрос
Почему вы проверяете структуру Rails вместо своего приложения Rails? Рамка Rails имеет собственные тесты!
поэтому позвольте мне ответить на это упреждающе: я чувствую, что я не тестирую сам фрейм, но правильно ли я использую фреймворк. Мой контроллер не наследует от ActionController::Base
, а от ActionController::API
, и я не знал, использует ли ActionController::API
ActionDispatch::ExceptionWrapper
по умолчанию, или мне сначала пришлось бы сказать моему контроллеру сделать это как-то.