Ответ 1
before :each do
request.env["HTTP_ACCEPT"] = 'application/json'
end
Я выполняю функциональные тесты для своих контроллеров с Rspec. Я установил свой формат ответа по умолчанию на моем маршрутизаторе в JSON, поэтому каждый запрос без суффикса вернет JSON.
Теперь в rspec, я получаю ошибку (406), когда я пытаюсь
get :index
Мне нужно сделать
get :index, :format => :json
Теперь, поскольку я в первую очередь поддерживаю JSON с моим API, очень избыточно указывать формат JSON для каждого запроса.
Можно ли каким-либо образом установить его по умолчанию для всех моих запросов GET? (или всех запросов)
before :each do
request.env["HTTP_ACCEPT"] = 'application/json'
end
Поместите это в spec/support
:
require 'active_support/concern'
module DefaultParams
extend ActiveSupport::Concern
def process_with_default_params(action, parameters, session, flash, method)
process_without_default_params(action, default_params.merge(parameters || {}), session, flash, method)
end
included do
let(:default_params) { {} }
alias_method_chain :process, :default_params
end
end
RSpec.configure do |config|
config.include(DefaultParams, :type => :controller)
end
И затем просто переопределите default_params
:
describe FooController do
let(:default_params) { {format: :json} }
...
end
Для меня работает rspec 3:
before :each do
request.headers["accept"] = 'application/json'
end
Это устанавливает HTTP_ACCEPT
.
Вот решение, которое
process
).Вот конфигурация RSpec:
module DefaultFormat
extend ActiveSupport::Concern
included do
let(:default_format) { 'application/json' }
prepend RequestHelpersCustomized
end
module RequestHelpersCustomized
l = lambda do |path, **kwarg|
kwarg[:headers] = {accept: default_format}.merge(kwarg[:headers] || {})
super(path, **kwarg)
end
%w(get post patch put delete).each do |method|
define_method(method, l)
end
end
end
RSpec.configure do |config|
config.include DefaultFormat, type: :request
end
Проверено с помощью
describe 'the response format', type: :request do
it 'can be overridden in request' do
get some_path, headers: {accept: 'text/plain'}
expect(response.content_type).to eq('text/plain')
end
context 'with default format set as HTML' do
let(:default_format) { 'text/html' }
it 'is HTML in the context' do
get some_path
expect(response.content_type).to eq('text/html')
end
end
end
FWIW, конфигурацию RSpec можно разместить:
Прямо в spec/spec_helper.rb
. Это не предлагается; файл будет загружен даже при тестировании библиотечных методов в lib/
.
Прямо в spec/rails_helper.rb
.
(мой любимый) В spec/support/default_format.rb
и загружаться явно в spec/rails_helper.rb
с помощью
require 'support/default_format'
В spec/support
и быть загруженным
Dir[Rails.root.join('spec/support/**/*.rb')].each { |f| require f }
который загружает все файлы в spec/support
.
Это решение вдохновлено ответом knoopx. Его решение не работает для спецификаций запросов, и alias_method_chain
устарело в пользу Module#prepend
.
В RSpec 3 вам нужно, чтобы тесты JSON запрашивали спецификации, чтобы визуализировать представления. Вот что я использую:
# spec/requests/companies_spec.rb
require 'rails_helper'
RSpec.describe "Companies", :type => :request do
let(:valid_session) { {} }
describe "JSON" do
it "serves multiple companies as JSON" do
FactoryGirl.create_list(:company, 3)
get 'companies', { :format => :json }, valid_session
expect(response.status).to be(200)
expect(JSON.parse(response.body).length).to eq(3)
end
it "serves JSON with correct name field" do
company = FactoryGirl.create(:company, name: "Jane Doe")
get 'companies/' + company.to_param, { :format => :json }, valid_session
expect(response.status).to be(200)
expect(JSON.parse(response.body)['name']).to eq("Jane Doe")
end
end
end
Что касается установки формата во всех тестах, мне нравится подход из этого другого ответа: fooobar.com/questions/139509/...
Возможно, вы могли бы добавить первый ответ в spec/spec_helper или spec/rails_helper с этим:
config.before(:each) do
request.env["HTTP_ACCEPT"] = 'application/json' if defined? request
end
если в тесте модели (или какой-либо не существует контекста методов запросов) этот код просто игнорируется. он работал с rspec 3.1.7 и рельсами 4.1.0 он должен работать со всеми рельсами 4 версии вообще.
Запуск Rails 5 и Rspec 3.5 Мне пришлось установить заголовки для выполнения этого.
post '/users', {'body' => 'params'}, {'ACCEPT' => 'application/json'}
Thi соответствует тому, что пример в docs выглядит следующим образом:
require "rails_helper"
RSpec.describe "Widget management", :type => :request do
it "creates a Widget" do
headers = {
"ACCEPT" => "application/json", # This is what Rails 4 accepts
"HTTP_ACCEPT" => "application/json" # This is what Rails 3 accepts
}
post "/widgets", { :widget => {:name => "My Widget"} }, headers
expect(response.content_type).to eq("application/json")
expect(response).to have_http_status(:created)
end
end
Для тех людей, которые работают с запросами, самый простой способ - переопределить метод #process
в ActionDispatch::Integration::Session
и установить параметр as
по умолчанию :json
следующим образом:
module DefaultAsForProcess
def process(method, path, params: nil, headers: nil, env: nil, xhr: false, as: :json)
super
end
end
ActionDispatch::Integration::Session.prepend(DefaultAsForProcess)
Согласно Rspec docs, поддерживаемый метод через заголовки:
require "rails_helper"
RSpec.describe "Widget management", :type => :request do
it "creates a Widget" do
headers = {
"ACCEPT" => "application/json", # This is what Rails 4 and 5 accepts
"HTTP_ACCEPT" => "application/json", # This is what Rails 3 accepts
}
post "/widgets", :params => { :widget => {:name => "My Widget"} }, :headers => headers
expect(response.content_type).to eq("application/json")
expect(response).to have_http_status(:created)
end
end
Исходя из этого вопроса, вы можете попробовать переопределить процесс() в ActionController:: TestCase из https://github.com/rails/rails/blob/32395899d7c97f69b508b7d7f9b7711f28586679/actionpack/lib/action_controller/test_case.rb.
Вот мое обходное решение.
describe FooController do
let(:defaults) { {format: :json} }
context 'GET index' do
let(:params) { defaults }
before :each do
get :index, params
end
# ...
end
context 'POST create' do
let(:params) { defaults.merge({ name: 'bar' }) }
before :each do
post :create, params
end
# ...
end
end