Аутентификация в байте

Я ищу способ сделать этот, но в спецификациях запросов. Мне нужно войти в систему и выйти из системы double или instance_double в Devise вместо фактического ActiveModel/ActiveRecord.

Используя код на странице wiki:

module RequestSpecHelpers
    def sign_in(user = double('user'))
      if user.nil?
        allow(request.env['warden']).to receive(:authenticate!).and_throw(:warden, {:scope => :user})
        allow(controller).to receive(:current_user).and_return(nil)
      else
        allow(request.env['warden']).to receive(:authenticate!).and_return(user)
        allow(controller).to receive(:current_user).and_return(user)
      end
    end
  end

Я получаю эту ошибку: undefined method 'env' for nil:NilClass

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

Тестирование:

RSpec.describe 'new shipment', type: :request do
  describe 'authenticated as user' do
    before do
      @user = double(:user, id: 1, email: '[email protected]', password: 'password',
                      id_card: '4163649-1', first_name: 'Jane', last_name: 'Doe')

      sign_in @user
    end
  end
end

Если я включаю:

RSpec.configure do |config|
  config.include Devise::TestHelpers, :type => :requests
end

Я получаю эту ошибку:

Failure/Error: @request.env['action_controller.instance'] = @controller

     NoMethodError:
       undefined method `env' for nil:NilClass
     # /root/.rbenv/versions/2.4.2/lib/ruby/gems/2.4.0/gems/devise-4.3.0/lib/devise/test/controller_helpers.rb:40:in `setup_controller_for_warden'

Проблема с ответом Фредерика Чунга

Если я это сделаю, метод login_as не сбой, но на самом деле он не регистрирует пользователя. Поэтому, когда я пытаюсь получить доступ к пути с обратным вызовом before_action :authenticate_user!, он терпит неудачу.

Вот мой код, основанный на его ответе:

require 'rails_helper'

RSpec.describe 'new shipment', type: :request do
  describe 'authenticated as user' do
    include Warden::Test::Helpers

    before(:each) do
      Warden.test_mode!
      #stub more methods as needed by the pages you are testing
      user = instance_double(User, to_key: 1, authenticatable_salt: 'example')
      login_as(user, scope: 'user')
    end

    it 'returns 200 Ok' do
      get new_shipment_path
      expect(response).to have_http_status(:ok)
    end
  end
end

И это ответ при запуске rspec:

 1) new shipment authenticated as user returns 200 Ok
     Failure/Error: expect(response).to have_http_status(:ok)
       expected the response to have status code :ok (200) but it was :found (302)
     # ./spec/requests/shipments_requests_spec.rb:41:in `block (3 levels) in <top (required)>'

Как вы можете видеть вместо того, чтобы разрешить мне доступ к пути, который он перенаправляет меня, это обычное поведение, когда пользователю не разрешен доступ к пути.

Это я изменяю instance_double для реального User, сохраненного в базе данных, этот подход работает правильно:

# only changed this line in the before hook
user = User.create(email: '[email protected]', password: 'password',id_card: '4163649-1', first_name: 'Jane', last_name: 'Doe')

Результат:

Finished in 3.23 seconds (files took 33.47 seconds to load)
1 example, 0 failures

Ответ 1

Похоже, вы используете Devise 3.x(поскольку Devise::TestHelpers был переименован в приложение 4), Devise::TestHelpers предназначен только для работы с спецификациями контроллера.

Если вы можете выполнить обновление до 4, у него есть отдельные помощники для спецификаций запросов и тестов контроллера. Это всего лишь очень тонкая обертка вокруг того, что надзиратель предоставляет, который скрывает все беспорядки с помощью env.

Есть несколько дополнительных осложнений при использовании двойника - вам нужно разбить различные методы, которые вы не понимаете.

Следующие работали для меня

describe 'example' do
  include Warden::Test::Helpers

  before(:each) do
    Warden.test_mode!
    #stub more methods as needed by the pages you are testing
    user = instance_double(User, to_key: 1, authenticatable_salt: 'example')
    login_as(user, scope: 'user')
  end
end