Как я могу использовать omniauth в rspec для sinatra? - PullRequest
4 голосов
/ 07 декабря 2011

Сокращенная версия:

Используя гем omniauth для sinatra, я не могу заставить работать протокол rspec и сохранять сеанс для последующих запросов.

На основена предложениях от http://benprew.posterous.com/testing-sessions-with-sinatra, и отключении сеансов я выделил проблему следующим образом:

  app.send(:set, :sessions, false)    # From http://benprew.posterous.com/testing-sessions-with-sinatra
  get '/auth/google_oauth2/callback', nil, {"omniauth.auth" => OmniAuth.config.mock_auth[:google_oauth2] }
  # last_request.session => {"uid"=>"222222222222222222222", :flash=>{:success=>"Welcome"}}
  # last_response.body => ""

  follow_redirect!
  # last_request.session => {:flash=>{}}
  # last_response.body => Html for the homepage, which is what I want

Как мне заставить rspec следовать перенаправлению и сохранять переменные сеанса?Возможно ли это в Синатре?

Начиная с http://benprew.posterous.com/testing-sessions-with-sinatra, кажется, что мне придется отправлять переменные сеанса при каждом запросе get / post, для которого мне требуется вход в систему, но это не сработает вслучай перенаправлений.


Подробности:

Я пытаюсь использовать камень omniauth в sinatra со следующей настройкой:

spec_helper.rb

ENV['RACK_ENV'] = 'test'

# Include web.rb file
require_relative '../web'
# Include factories.rb file
require_relative '../test/factories.rb'

require 'rspec'
require 'rack/test'
require 'factory_girl'
require 'ruby-debug'

# Include Rack::Test in all rspec tests
RSpec.configure do |conf|
  conf.include Rack::Test::Methods
  conf.mock_with :rspec
end

web_spec.rb

describe "Authentication:" do
  before do
    OmniAuth.config.test_mode = true
    OmniAuth.config.add_mock(:google_oauth2, {
      :uid => '222222222222222222222',
      :info => {
        :email => "someone@example.com",
        :name => 'Someone'
      }
    })
  end

  describe "Logging in as a new user" do
    it "should work" do
      get '/auth/google_oauth2/'

      last_response.body.should include("Welcome")
    end
  end
end

При попытке аутентификации я получаю <h1>Not Found</h1> ответ.Чего мне не хватает?

На странице Интеграционное тестирование в документации omniauth упоминается добавление двух переменных среды:

before do 
  request.env["devise.mapping"] = Devise.mappings[:user] 
  request.env["omniauth.auth"] = OmniAuth.config.mock_auth[:twitter] 
end

Но, похоже, только для рельсов, как я добавил

request.env["omniauth.auth"] = OmniAuth.config.mock_auth[:google_oauth2] 

к моему before блоку в моей спецификации, и я получаю эту ошибку:

Failure/Error: request.env["omniauth.auth"] = OmniAuth.config.mock_auth[:google_oauth2]
 ArgumentError:
   wrong number of arguments (0 for 1)

Редактировать:

Вызов get с

get '/auth/google_oauth2/', nil, {"omniauth.auth" => OmniAuth.config.mock_auth[:google_oauth2]}

, кажется, дает мне last_request.env["omniauth.auth"], равный

 {"provider"=>"google_oauth2", "uid"=>"222222222222222222222", "info"=>{"email"=>"someone@example.com", "name"=>"Someone"}}

, что кажется правильным, но last_response.body все еще возвращает

<h1>Not Found</h1>

Ответы [ 2 ]

3 голосов
/ 07 декабря 2011

Частичный ответ ...

URL обратного вызова работает лучше, с добавленными переменными среды запроса:

get '/auth/google_oauth2/callback', nil, {"omniauth.auth" => OmniAuth.config.mock_auth[:google_oauth2]}
follow_redirect!

last_response.body.should include("Welcome")

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

0 голосов
/ 08 декабря 2011

Используя этот гист (происходящий из https://stackoverflow.com/a/3892401/111884) для хранения данных сеанса, я получил свои тесты для хранения сеанса, что позволило мне передать сеанс для дальнейших запросов.

Возможно, существует более простой способ.

Код настройки:

# Omniauth settings
OmniAuth.config.test_mode = true
OmniAuth.config.add_mock(:google_oauth2, {
  :uid => '222222222222222222222',
  :info => {
    :email => "someone@example.com",
    :name => 'Someone'
  }
})


# Based on https://gist.github.com/375973 (from https://stackoverflow.com/a/3892401/111884)
class SessionData
  def initialize(cookies)
    @cookies = cookies
    @data = cookies['rack.session']
    if @data
      @data = @data.unpack("m*").first
      @data = Marshal.load(@data)
    else
      @data = {}
    end
  end

  def [](key)
    @data[key]
  end

  def []=(key, value)
    @data[key] = value
    session_data = Marshal.dump(@data)
    session_data = [session_data].pack("m*")
    @cookies.merge("rack.session=#{Rack::Utils.escape(session_data)}", URI.parse("//example.org//"))
    raise "session variable not set" unless @cookies['rack.session'] == session_data
  end
end

def login!(session)
  get '/auth/google_oauth2/callback', nil, { "omniauth.auth" => OmniAuth.config.mock_auth[:google_oauth2] }
  session['uid'] = last_request.session['uid']

  # Logged in user should have the same uid as login credentials
  session['uid'].should == OmniAuth.config.mock_auth[:google_oauth2]['uid']
end

# Based on Rack::Test::Session::follow_redirect!
def follow_redirect_with_session_login!(session)
  unless last_response.redirect?
    raise Error.new("Last response was not a redirect. Cannot follow_redirect!")
  end

  get(last_response["Location"], {}, { "HTTP_REFERER" => last_request.url, "rack.session" => {"uid" => session['uid']} })
end

def get_with_session_login(path)
  get path, nil, {"rack.session" => {"uid" => session['uid']}}
end

Пример кода rspec:

describe "Authentication:" do

  def session
    SessionData.new(rack_test_session.instance_variable_get(:@rack_mock_session).cookie_jar)
  end

  describe "Logging in as a new user" do
    it "should create a new account with the user's name" do
      login!(session)
      last_request.session[:flash][:success].should include("Welcome")

      get_with_session_login "/"
      follow_redirect_with_session_login!(session)
      last_response.body.should include("Someone")
    end
  end
end
...