Подтвердить что ты не робот

Задайте формат запроса GET для Rspec по умолчанию для JSON

Я выполняю функциональные тесты для своих контроллеров с Rspec. Я установил свой формат ответа по умолчанию на моем маршрутизаторе в JSON, поэтому каждый запрос без суффикса вернет JSON.

Теперь в rspec, я получаю ошибку (406), когда я пытаюсь

get :index

Мне нужно сделать

get :index, :format => :json

Теперь, поскольку я в первую очередь поддерживаю JSON с моим API, очень избыточно указывать формат JSON для каждого запроса.

Можно ли каким-либо образом установить его по умолчанию для всех моих запросов GET? (или всех запросов)

4b9b3361

Ответ 1

before :each do
  request.env["HTTP_ACCEPT"] = 'application/json'
end

Ответ 2

Поместите это в 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

Ответ 3

Для меня работает rspec 3:

before :each do
  request.headers["accept"] = 'application/json'
end

Это устанавливает HTTP_ACCEPT.

Ответ 4

Вот решение, которое

  1. работает для спецификации запроса,
  2. работает с Rails 5 и
  3. не использует закрытый API Rails (например, 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 можно разместить:

  1. Прямо в spec/spec_helper.rb. Это не предлагается; файл будет загружен даже при тестировании библиотечных методов в lib/.

  2. Прямо в spec/rails_helper.rb.

  3. (мой любимый) В spec/support/default_format.rb и загружаться явно в spec/rails_helper.rb с помощью

    require 'support/default_format'
    
  4. В spec/support и быть загруженным

    Dir[Rails.root.join('spec/support/**/*.rb')].each { |f| require f }
    

    который загружает все файлы в spec/support.

Это решение вдохновлено ответом knoopx. Его решение не работает для спецификаций запросов, и alias_method_chain устарело в пользу Module#prepend.

Ответ 5

В 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/...

Ответ 6

Возможно, вы могли бы добавить первый ответ в 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 версии вообще.

Ответ 7

Запуск 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

Ответ 8

Для тех людей, которые работают с запросами, самый простой способ - переопределить метод #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)

Ответ 9

Согласно 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

Ответ 10

Почему методы RSpec, "get" , "post" , "put" , "delete" , работать в спецификации контроллера в драгоценном камне (или за пределами Rails)?

Исходя из этого вопроса, вы можете попробовать переопределить процесс() в 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